原创 wince5.0 注册表还原(转)

2008-6-19 11:42 2559 2 2 分类: MCU/ 嵌入式
当Wince使用了HIVE注册表后,每次用户的注册表改动将得到保存,但是在某些应用场合需要将注册表还原成为出厂的默认设置,通常要求能够在AP中通过点击一个按钮来实现这种clean boot。使用我前面的文章的方法配置的HIVE系统注册表和HIVE用户注册表分别保存在/HDD/Document and Setting/system.hv 中和 /HDD/Document and Setting/default/user.hv中,每次系统在启动到加载HIVE系统注册表之前都会先检查保存在/HDD中的文件的存在和合法性,如果不满足要求系统将会用binfs中的缺省文件创建新的system.hv和user.hv文件于/HDD中,根据这个特性我先试图在WinCE运行起来后删除这两个hv文件,但是由于WinCE已经事先加载了它们,删除被禁止,只有采用其他的方法。

重新研究了基于HIVE注册表的WinCE的启动过程发现,系统在完成了第一阶段也就是加载完了boot.hv+binfs之后和加载系统HIVE注册表之前,filesys.exe都会调用OEMIoControl来查询是否需要清除保存在block设备上的hv文件,其CODE代码为IOCTL_HAL_GET_HIVE_CLEAN_FLAG,它的输入参数lpInBuf固定为HIVECLEANFLAG_SYSTEM或HIVECLEANFLAG_USERS,filesys.exe会分别用这两种参数调用两次IOCTL_HAL_GET_HIVE_CLEAN_FLAG,第一次用HIVECLEANFLAG_SYSTEM来问OEM是否需要清除system.hv,第二次用HIVECLEANFLAG_USERS做参数来查询是否要清除user.hv,如果返回的lpOutBuf中的值为TRUE则做清除操作,如果为False则保留block设备上的注册表文件。

所以我们要做的就是实现和IOCTL_HAL_GET_HIVE_CLEAN_FLAG相对应的OEMIoControl源码(假设由OALIoCtlBGetHiveCleanFlag()这个function来实现),加入对是否需要清除注册表的判定条件并告知filesys.exe即可。现在棘手的问题是如何让AP通知OALIoCtlBGetHiveCleanFlag()该不该清除注册表,因为OALIoCtlBGetHiveCleanFlag()只能在指定的时候由filesys.exe去调用,AP的运行只能在OALIoCtlBGetHiveCleanFlag()运行完之后。

后来终于想到可以使用共享内存空间来实现,我们可以在物理内存中保留出一块不会被其他模块占用的空间,在这个空间放置两个BOOL变量分别来保存system和user的hv清除的标志符,缺省它们都为False,OALIoCtlBGetHiveCleanFlag()读到Flase则认为不清注册表,AP在需要的时候将这两个标志符置为True,接下来就是要重新启动到OALIoCtlBGetHiveCleanFlag()函数被调用的地方,由于标志符号是保存在RAM中的,断电会丢失,还好有个方法可以让系统复位而又能保存RAM中的内容,那就是Reset,所以让AP在设置完标志符后马上调用Reset指令就可以完美实现Clean boot了。

提供相关代码作为参考:

BOOL OALIoCtlBGetHiveCleanFlag(   // 一般在IOCTL.C中实现

       UINT32 code, VOID *lpInBuf , UINT32 nInBufSize, VOID *lpOutBuf,

       UINT32 nOutBufSize , UINT32 *pOutSize)

{

BSP_ARGS *pArgs = (BSP_ARGS*)IMAGE_SHARE_ARGS_UA_START; //保留的共享RAM空间的虚拟地址

if (!lpInBuf || (nInBufSize != sizeof(DWORD)) || !lpOutBuf || (nOutBufSize != sizeof(BOOL)))

 {

     SetLastError(ERROR_INVALID_PARAMETER);

     return FALSE;

 }

else

{

     DWORD *pdwFlags = (DWORD*)lpInBuf;

     BOOL *pfClean = (BOOL*)lpOutBuf;

    if (*pdwFlags == HIVECLEANFLAG_SYSTEM) {

          if(*pfClean = pArgs->bcleansystemhive) //判断是否清除system.hv

                   RETAILMSG(1, (TEXT("OEM: cleaning system hive\r\n")));

               else

                   RETAILMSG(1, (TEXT("OEM: Not cleaning system hive\r\n")));

          *pfClean = pArgs->bcleansystemhive;

               pArgs->bcleansystemhive=FALSE; //一定在执行完后设置为默认的false否则常规reset都会清空注册表

     } else if (*pdwFlags == HIVECLEANFLAG_USERS) {

     if(*pfClean = pArgs->bcleanuserhive) //判断是否清除user.hv

                   RETAILMSG(1, (TEXT("OEM: cleaning user hive\r\n")));

               else

                   RETAILMSG(1, (TEXT("OEM: Not cleaning user hive\r\n")));

                 *pfClean = pArgs->bcleanuserhive;

                   pArgs->bcleanuserhive=FALSE; //restore to default

     }

 }

return TRUE;

}

 

AP中的实现代码如下:

Void On_CleanBoot()

{

BSP_ARGS* pArgs = (BSP_ARGS*)IMAGE_SHARE_ARGS_UA_START; //保留的共享RAM空间的虚拟地址

pArgs->bcleansystemhive=TRUE;         //设置system.hv清空标志符

pArgs->bcleanuserhive=TRUE;            //设置user.hv清空标志符

ReSet();

}

 

引用本文请注明出处:http://blog.csdn.net/fredzeng

文章评论0条评论)

登录后参与讨论
我要评论
0
2
关闭 站长推荐上一条 /2 下一条