wince5.0 注冊表還原
當(dāng)Wince使用了HIVE注冊表后,每次用戶的注冊表改動將得到保存,但是在某些應(yīng)用場合需要將注冊表還原成為出廠的默認(rèn)設(shè)置,通常要求能夠在AP中通過點擊一個按鈕來實現(xiàn)這種clean boot。使用我前面的文章的方法配置的HIVE系統(tǒng)注冊表和HIVE用戶注冊表分別保存在/HDD/Document and Setting/system.hv 中和 /HDD/Document and Setting/default/user.hv中,每次系統(tǒng)在啟動到加載HIVE系統(tǒng)注冊表之前都會先檢查保存在/HDD中的文件的存在和合法性,如果不滿足要求系統(tǒng)將會用binfs中的缺省文件創(chuàng)建新的system.hv和user.hv文件于/HDD中,根據(jù)這個特性我先試圖在WinCE運行起來后刪除這兩個hv文件,但是由于WinCE已經(jīng)事先加載了它們,刪除被禁止,只有采用其他的方法。
重新研究了基于HIVE注冊表的WinCE的啟動過程發(fā)現(xiàn),系統(tǒng)在完成了第一階段也就是加載完了boot.hv+binfs之后和加載系統(tǒng)HIVE注冊表之前,filesys.exe都會調(diào)用OEMIoControl來查詢是否需要清除保存在block設(shè)備上的hv文件,其CODE代碼為IOCTL_HAL_GET_HIVE_CLEAN_FLAG,它的輸入?yún)?shù)lpInBuf固定為HIVECLEANFLAG_SYSTEM或HIVECLEANFLAG_USERS,filesys.exe會分別用這兩種參數(shù)調(diào)用兩次IOCTL_HAL_GET_HIVE_CLEAN_FLAG,第一次用HIVECLEANFLAG_SYSTEM來問OEM是否需要清除system.hv,第二次用HIVECLEANFLAG_USERS做參數(shù)來查詢是否要清除user.hv,如果返回的lpOutBuf中的值為TRUE則做清除操作,如果為False則保留block設(shè)備上的注冊表文件。
所以我們要做的就是實現(xiàn)和IOCTL_HAL_GET_HIVE_CLEAN_FLAG相對應(yīng)的OEMIoControl源碼(假設(shè)由OALIoCtlBGetHiveCleanFlag()這個function來實現(xiàn)),加入對是否需要清除注冊表的判定條件并告知filesys.exe即可?,F(xiàn)在棘手的問題是如何讓AP通知OALIoCtlBGetHiveCleanFlag()該不該清除注冊表,因為OALIoCtlBGetHiveCleanFlag()只能在指定的時候由filesys.exe去調(diào)用,AP的運行只能在OALIoCtlBGetHiveCleanFlag()運行完之后。
后來終于想到可以使用共享內(nèi)存空間來實現(xiàn),我們可以在物理內(nèi)存中保留出一塊不會被其他模塊占用的空間,在這個空間放置兩個BOOL變量分別來保存system和user的hv清除的標(biāo)志符,缺省它們都為False,OALIoCtlBGetHiveCleanFlag()讀到Flase則認(rèn)為不清注冊表,AP在需要的時候?qū)⑦@兩個標(biāo)志符置為True,接下來就是要重新啟動到OALIoCtlBGetHiveCleanFlag()函數(shù)被調(diào)用的地方,由于標(biāo)志符號是保存在RAM中的,斷電會丟失,還好有個方法可以讓系統(tǒng)復(fù)位而又能保存RAM中的內(nèi)容,那就是Reset,所以讓AP在設(shè)置完標(biāo)志符后馬上調(diào)用Reset指令就可以完美實現(xiàn)Clean boot了。
提供相關(guān)代碼作為參考:
BOOL OALIoCtlBGetHiveCleanFlag( // 一般在IOCTL.C中實現(xiàn)
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 hivern")));
else
RETAILMSG(1, (TEXT("OEM: Not cleaning system hivern")));
*pfClean = pArgs->bcleansystemhive;
pArgs->bcleansystemhive=FALSE;//一定在執(zhí)行完后設(shè)置為默認(rèn)的false否則常規(guī)reset都會清空注冊表
} else if (*pdwFlags == HIVECLEANFLAG_USERS) {
if(*pfClean = pArgs->bcleanuserhive) //判斷是否清除user.hv
RETAILMSG(1, (TEXT("OEM: cleaning user hivern")));
else
RETAILMSG(1, (TEXT("OEM: Not cleaning user hivern")));
*pfClean = pArgs->bcleanuserhive;
pArgs->bcleanuserhive=FALSE;//restore to default
}
}
return TRUE;
}
AP中的實現(xiàn)代碼如下:
Void On_CleanBoot()
{
BSP_ARGS* pArgs = (BSP_ARGS*)IMAGE_SHARE_ARGS_UA_START; //保留的共享RAM空間的虛擬地址
pArgs->bcleansystemhive=TRUE; //設(shè)置system.hv清空標(biāo)志符
pArgs->bcleanuserhive=TRUE; //設(shè)置user.hv清空標(biāo)志符
ReSet();
linux操作系統(tǒng)文章專題:linux操作系統(tǒng)詳解(linux不再難懂)
評論