對于WinCE5.0:
1. pbxmlutils.exe
在 “Windows CE Platform. Builder5.00CEPBBIN”下面可以找到,這個命令可以從PB的文檔中查到。主要作用是通過解析你的工程文件
pbxmlutils /getbuildenv /workspace "%_WINCEROOT%PBWorkspaces
看著可能有點暈,我來舉個例子:
pbxmlutils /getbuildenv /workspace "%_WINCEROOT%PBWorkspacesMyWorkspaceMyWorkspace.pbxml" /config "CEPC: x86_Release" > SetEnv.bat
其實就兩個參數,第一個參數很多人都知道如何設置,第二個參數config可能有些人不太了解,你可以打開你的工程,然后查環(huán)境變量PBCONFIG的值,用這個值就可以了。
2. Blddemo.exe
這個應該比較熟悉。編譯WinCE的時候,實際上就是調用Blddemo -q,不多說了。
3. buildsdk.exe
這個是用來編譯SDK的。格式如下;
buildsdk [MyOSDesign.pbxml]
很簡單,不過還是給個例子:
buildsdk "%_WINCEROOT%PBWorkspacesMyWorkspaceMyWorkspace.pbxml"
在某些應用場合需要將注冊表還原成為出廠的默認設置。以前當Wince使用了HIVE注冊表后,每次用戶的注冊表改動將得到保存,而不會恢復默認注冊表。通常要求能夠在AP中通過點擊一個按鈕來實現(xiàn)這種clean boot。當Wince使用了HIVE注冊表后,每次用戶的注冊表改動將得到保存,但是在某些應用場合需要將注冊表還原成為出廠的默認設置,通常要求能夠在AP中通過點擊一個按鈕來實現(xiàn)這種clean boot。
方法一:
使用HIVE注冊表,系統(tǒng)在完成了第一階段也就是加載完了boot.hv+binfs之后和加載系統(tǒng)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設備上的注冊表文件。
默認WINCE并沒有實現(xiàn)這個IOCTL,所以OEM要刪除注冊表文件就必須先編寫這個IOCTL代碼。代碼的例子可參考標題為“IOCTL_HAL_GET_HIVE_CLEAN_FLAG”的幫助文檔。另外 必須在ioctl.h和ioctl.c兩個文件中編寫該代碼。
要進一步了解這個全局數組,參見標題為“IOCTL Library”的幫助文檔。
可以使用共享內存空間來實現(xiàn),我們可以在物理內存中保留出一塊不會被其他模塊占用的空間,在這個空間放置兩個BOOL變量分別來保存system和user的hv清除的標志符,缺省它們都為False,OALIoCtlBGetHiveCleanFlag()讀到Flase則認為不清注冊表,AP在需要的時候將這兩個標志符置為True,接下來就是要重新啟動到OALIoCtlBGetHiveCleanFlag()函數被調用的地方,由于標志符號是保存在RAM中的,斷電會丟失,還好有個方法可以讓系統(tǒng)復位而又能保存RAM中的內容,那就是Reset,所以讓AP在設置完標志符后馬上調用Reset指令就可以完美實現(xiàn)Clean boot了。
1 在ioctl.c文件中找到 const OAL_IOCTL_HANDLER g_oalIoCtlTable[],添加IOCTL和對應的處理函數OALIoCtlBGetHiveCleanFlag
{ IOCTL_HAL_GET_HIVE_CLEAN_FLAG, 0, OALIoCtlBGetHiveCleanFlag },
2 在ioctl.c文件中實現(xiàn)該函數
//added by aulyp for 恢復出廠設置
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(pArgs->bClearSystemHive) //判斷是否清除system.hv
{
RETAILMSG(1, (TEXT("OEM: cleaning system hivern")));
}
else
{
RETAILMSG(1, (TEXT("OEM: Not cleaning system hivern")));
}
*pfClean = pArgs->bClearSystemHive;
pArgs->bClearSystemHive=FALSE; //一定在執(zhí)行完后設置為默認的false否則常規(guī)reset都會清空注冊表
}
else if (*pdwFlags == HIVECLEANFLAG_USERS)
{
if(pArgs->bClearUserHive) //判斷是否清除user.hv
{
RETAILMSG(1, (TEXT("OEM: cleaning user hivern")));
}
else
{
RETAILMSG(1, (TEXT("OEM: Not cleaning user hivern")));
}
*pfClean = pArgs->bClearUserHive
pArgs->bClearUserHive=FALSE; //restore to default
}
}
return TRUE;
}
3 Args.h文件中,找到BSP_ARGS結構體定義
加入如下:
BOOL bClearSystemHive=FALSE; //TRUE ? clear system.hv
BOOL bClearUserHive=FALSE; //TRUE ? clear user.hv
4 在Oal_ioctl.h中加入函數聲明
BOOL OALIoCtlBGetHiveCleanFlag(UINT32 code, VOID *lpInBuf , UINT32 nInBufSize,[!--empirenews.page--]
VOID *lpOutBuf,UINT32 nOutBufSize , UINT32 *pOutSize);
5在應用中可以如下使用:
Void On_CleanBoot()
{
BSP_ARGS* pArgs = (BSP_ARGS*)IMAGE_SHARE_ARGS_UA_START;
//保留的共享RAM空間的虛擬地址
pArgs-> bClearSystemHive =TRUE; //設置system.hv清空標志符
pArgs-> bClearUserHive =TRUE; //設置user.hv清空標志符
ReSet();
}
使用這種方法每次系統(tǒng)在啟動到加載HIVE系統(tǒng)注冊表之前都會先檢查保存在/HDD中的文件的存在和合法性,如果不滿足要求系統(tǒng)將會用binfs中的缺省文件創(chuàng)建新的system.hv和user.hv文件于/HDD中,根據這個特性我在WinCE運行起來后刪除這兩個hv文件,但是由于WinCE已經事先加載了它們,刪除被禁止,只有采用其他的方法來解決。
基于HIVE注冊表的WinCE的啟動過程我發(fā)現(xiàn),系統(tǒng)在完成了第一階段也就是加載完了boot.hv+binfs之后和加載系統(tǒng)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設備上的注冊表文件。
所以現(xiàn)在要做的就是實現(xiàn)和IOCTL_HAL_GET_HIVE_CLEAN_FLAG相對應的OEMIoControl源碼(假設由OALIoCtlBGetHiveCleanFlag()這個function來實現(xiàn)),加入對是否需要清除注冊表的判定條件并告知filesys.exe即可?,F(xiàn)在的問題是如何讓AP通知OALIoCtlBGetHiveCleanFlag()該不該清除注冊表,因為OALIoCtlBGetHiveCleanFlag()只能在指定的時候由filesys.exe去調用,AP的運行只能在OALIoCtlBGetHiveCleanFlag()運行完之后。
后來想到可以使用共享內存空間來實現(xiàn),可以在物理內存中保留出一塊不會被其他模塊占用的空間,在這個空間放置兩個BOOL變量分別來保存system和user的hv清除的標志符,缺省它們都為False,OALIoCtlBGetHiveCleanFlag()讀到Flase則認為不清注冊表,AP在需要的時候將這兩個標志符置為True,接下來就是要重新啟動到OALIoCtlBGetHiveCleanFlag()函數被調用的地方,由于標志符號是保存在RAM中的,斷電會丟失,還好有個方法可以讓系統(tǒng)復位而又能保存RAM中的內容,那就是Reset,所以讓AP在設置完標志符后馬上調用Reset指令就可以完美實現(xiàn)Clean boot了。
提供相關代碼作為參考:
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í)行完后設置為默認的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; //設置system.hv清空標志符
pArgs->bcleanuserhive=TRUE; //設置user.hv清空標志符
ReSet();