当Wince使用了HIVE注册表后,每次用户的注册表改动将得到保存,但是在某些应用场合需要将注册表还原成为出厂的默认设置,通常要求能够通过一个跳线来标志是否清除Hive注册表。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即可。
提供相关代码作为参考:
//------------------------------------------------------------------------------
//
// Function: OALIoCtlBGetHiveCleanFlag
//
// This function is Get the Clean Hive Clean Flage
//
BOOL OALIoCtlBGetHiveCleanFlag(
UINT32 code, VOID *lpInBuf , UINT32 nInBufSize, VOID *lpOutBuf,
UINT32 nOutBufSize , UINT32 *pOutSize)
{
DWORD *pdwFlags;
BOOL *pfClean;
volatile XLLP_GPIO_T *pGPIORegs = (volatile XLLP_GPIO_T *) OALPAtoVA(BULVERDE_BASE_REG_PA_GPIO, FALSE);
BOOL Flage = 0;
if (!lpInBuf || (nInBufSize != sizeof(DWORD)) || !lpOutBuf || (nOutBufSize != sizeof(BOOL)))
{
NKSetLastError(ERROR_INVALID_PARAMETER);
return FALSE;
}
pdwFlags = (DWORD *)lpInBuf;
pfClean = (BOOL*)lpOutBuf;
pGPIORegs->GAFR1_L &= ~(0X03<<26); // Config gp45 as GPIO
pGPIORegs->GPDR1 &= ~(0x01<<13); // dir in
if((pGPIORegs->GPLR1) & (0x01<<13))
Flage = 0;
else
Flage = 1;
if (*pdwFlags == HIVECLEANFLAG_SYSTEM)
{
if(Flage) //判断是否清除system.hv
RETAILMSG(1, (TEXT("OEM: cleaning system hive/r/n")));
else
RETAILMSG(1, (TEXT("OEM: Not cleaning system hive/r/n")));
*pfClean = Flage;
}
else if (*pdwFlags == HIVECLEANFLAG_USERS) {
if(Flage) //判断是否清除user.hv
RETAILMSG(1, (TEXT("OEM: cleaning user hive/r/n")));
else
RETAILMSG(1, (TEXT("OEM: Not cleaning user hive/r/n")));
*pfClean = Flage;
}
return TRUE;
}
//
// Function: OALIoCtlBGetHiveCleanFlag
//
// This function is Get the Clean Hive Clean Flage
//
BOOL OALIoCtlBGetHiveCleanFlag(
UINT32 code, VOID *lpInBuf , UINT32 nInBufSize, VOID *lpOutBuf,
UINT32 nOutBufSize , UINT32 *pOutSize)
{
DWORD *pdwFlags;
BOOL *pfClean;
volatile XLLP_GPIO_T *pGPIORegs = (volatile XLLP_GPIO_T *) OALPAtoVA(BULVERDE_BASE_REG_PA_GPIO, FALSE);
BOOL Flage = 0;
if (!lpInBuf || (nInBufSize != sizeof(DWORD)) || !lpOutBuf || (nOutBufSize != sizeof(BOOL)))
{
NKSetLastError(ERROR_INVALID_PARAMETER);
return FALSE;
}
pdwFlags = (DWORD *)lpInBuf;
pfClean = (BOOL*)lpOutBuf;
pGPIORegs->GAFR1_L &= ~(0X03<<26); // Config gp45 as GPIO
pGPIORegs->GPDR1 &= ~(0x01<<13); // dir in
if((pGPIORegs->GPLR1) & (0x01<<13))
Flage = 0;
else
Flage = 1;
if (*pdwFlags == HIVECLEANFLAG_SYSTEM)
{
if(Flage) //判断是否清除system.hv
RETAILMSG(1, (TEXT("OEM: cleaning system hive/r/n")));
else
RETAILMSG(1, (TEXT("OEM: Not cleaning system hive/r/n")));
*pfClean = Flage;
}
else if (*pdwFlags == HIVECLEANFLAG_USERS) {
if(Flage) //判断是否清除user.hv
RETAILMSG(1, (TEXT("OEM: cleaning user hive/r/n")));
else
RETAILMSG(1, (TEXT("OEM: Not cleaning user hive/r/n")));
*pfClean = Flage;
}
return TRUE;
}
在ioctl_tab.h文件中加入以下代码
{IOCTL_HAL_GET_HIVE_CLEAN_FLAG,
0, OALIoCtlBGetHiveCleanFlag },