S3c2440A平台HIVE注册表+binfs的实现
先來回味一下底層的東東,我們的Image主要由兩部分組成:XIPKERNEL.bin和NK.bin,XIPKERNEL.bin中的東西就是那些WinCE中比較核心的又需要經常加載的一些程序和DLL文件,這些文件會被Boot Loader在剛啟動的時候拷貝到RAM中去,這樣就可以在RAM中XIP(Excute in place)了。在NK.bin中的基本上是需要但不至于要常駐內存的一些程序和DLL了,比如我們BuildIn下的大部分驅動,比如微軟的IE,mediaplayer等應用程序,甚至連設備管理器device.exe也可以放到這里面,這些文件只有在需要的時候才被復制到內存中去執行,節約了內存并且也加快了啟動的時間。嘿,到這里大概知道binfs的工作原理和重要性吧。
binfs的建立工作是在用UT(OEM自己的一種底層的工具集)實現的,UT在燒image的時候會自動把XIPKERNEL和NK分別保存到flash的特定的邏輯扇區上.啟動的時候Boot Loader會先把XIPKERNEL復制到RAM中,然后跳到RAM中的XIPKERNEL的入口點去執行,這個時候會跑一些OEMinit之類的CPU,內存,中短等初始化的過程,接著OS會從注冊表中找到binfs的一些設置,然后加載binfs的驅動使binfs分區對OS來講是可用的,假如device.exe是在NK.bin中的話,那么在這個時候就可以用\binfs\device.exe(\binfs是假設的裝載路徑)來調用它了,如果這個時候binfs沒有初始化成功那么,device.exe得不到執行,那么系統肯定就起不來了。
現在來講講HIVE,其實HIVE是個很簡單的東西,都怪和binfs牽到一起搞得很多問題都走錯了方向,本來一天就能搞定結果搞了四五天,NND。這么說吧,WinCE下面就兩種注冊表,一種是RAM based,另外就是HIVE based了,缺省用的是前者,如果用前者PB會在編譯的時候把common.reg和platform.reg的內容做到一個叫reginit.ini的文件然后壓縮成default.***(忘記擴展名了,有過老跡象了哈)的文件放到XIPKERNEL中去,image在起來的時候會把這個文件解壓到RAM中形成RAM based注冊表,既然是RAM based那么所有的改動都會在斷電后蒸發,哈哈。怎么辦呢?其實再笨你也能想出來,保存到磁盤上不就結了嗎!?對你太聰明了,但是你想如果你把注冊表全放到磁盤(SDMMC或HDD或Flash)上WinCE怎么在沒有加載你磁盤的驅動的情況下讀到注冊表呢?而一般情況加載磁盤的驅動程序也是要注冊表的支持啊!嘿,對了,這就是HIVE想到的,看它怎么做,它把注冊表分成兩部分(其實是三部分,當時大體還是兩步分,把user.hv和system.hv做一部分),第一部分就是叫做boot.hv的注冊表,里面的東西就是一些在沒有拿到保存在磁盤的注冊表之前引導時需要的一些設置,這部分的注冊表和RAM based的是一樣的,改了之后斷電就沒了,所以這部分的注冊表項都是不需要改動的,需要改動的都放到第二部分就是了,這第二部分就是system.hv和user.hv了,也就是一直提到的要放到磁盤上的注冊表. 編譯的時候PB會根據platform.reg和Common.reg中的標簽判斷哪些表項放到boot.hv中,這個標簽就是;HIVE BOOT SECTION ;END BOOT SECTION,夾在這個標簽之間的表項PB在編譯的時候會把它們塞到boot.hv中去(boot.hv是二進制文件,要看里面到底放了哪些表項用一個老外寫的工具吧,好像叫d_readvol.exe,到google上找得到的),其他的內容會分別塞到default.hv和user.hv中去,最后會把這三個hv文件統統塞到XIPKERNEL中去,這樣WinCE在引導的第一階段就把所有的hv扔到RAM中去了,然后打開boot.hv拿到必要的資料,這其中包括如何加載放置system.hv的磁盤的驅動,所以那些和加載這個磁盤相關的驅動要統統放到boot.hv中,比如FAT文件系統驅動,mspart分區驅動等等,這里有一點很重要就是假如你用binfs而且device.exe在NK.bin中,那么一定在第一階段要保證binfs可用,否則這里就不可能為system.hv創造條件了。WinCE第一次啟動時候磁盤上沒有東東,這個時候WinCE會將內存中的default.hv和user.hv復制到注冊表BootVars指定的地方,default.hv往往會被重命名為system.hv,第二次啟動會先檢查磁盤上的hv是不是和內存中的一致,不一致就加載磁盤上的表項。
整個過程就是這樣子,但要注意一點,HIVE注冊表也是在內存中運行的,不同的是啟動的時候會從磁盤上去讀改動的表項,因為這樣才能保證速度,所以你做的的注冊表改動也是在內存中做的,這個時候如果你不掉用FlushRegister去將內存中的數值保存到磁盤上那么這些改動還是會丟失的。兩種方法來避免丟失,一種是認為去調用FlushRegister,令一種就設置一個flag讓WinCE在每次改動注冊表后自動Flush.
最后總結一下我到底做了哪些事情:
1)在PB中將HIVEbased Registers拉到項目的WorkSpaces中來。
2)把Platform.reg中的下列表項加到boot.hv中
3)Build Image了
附上我的注冊表設置做參考:
;-----------------------------------------------------------------------------------------
;ALL these entries below will be add to boot.hv when hive register is enabled!
;HIVE BOOT SECTION
[HKEY_LOCAL_MACHINE\init\BootVars]
"SYSTEMHIVE"="Documents and Settings\\system.hv" ? ? ;system.hv會保存到\HDD\Documents and Settings\system.hv
"PROFILEDIR"="Documents and Settings" ? ;user.hv會保存到\HDD\Documents and Settings\default\user.hv
"Flags"=dword:3? ; 這個應該是wince 5.0下決定在哪個階段啟動device.exe的表項
"DefaultUser"="default" ? ? ? ;咱們只有一個用戶default,基本上就是決定user.hv的路徑了
"RegistryFlags"=dword:1 ? ;這個就是設置注冊表每次改動后自動flush到system.hv
;###########################################################################
;這個部分是binfs的注冊表項,如果你不是用的binfs那么不用將它們拉到boot.hv中
[HKEY_LOCAL_MACHINE\System\StorageManager\AutoLoad\SMFlash]
"DriverPath"="Drivers\\BlockDevice\\SMFlash"
"LoadFlags"=dword:1
"MountFlags"=dword:11
"BootPhase"=dword:0
"Flags"=dword:1000
[HKEY_LOCAL_MACHINE\Drivers\BlockDevice\SMFlash]
"Prefix"="DSK"
"Dll"="BIBDrv.dll"? ;這個binfs的驅動DLL一定要在XIPKERNEL內部
"Order"=dword:0
"Ioctl"=dword:4
"Profile"="SMFlash"
"FriendlyName"="Samsung Flash Driver"
"MountFlags"=dword:11
"BootPhase"=dword:0
"Flags"=dword:1000
; Bind BINFS to the block driver
[HKEY_LOCAL_MACHINE\System\StorageManager\Profiles\SMFlash]
"DefaultFileSystem"="BINFS" ? ;binfs的路徑為/BINFS
"PartitionDriver"="mspart.dll"?? ;這個分區的驅動DLL一定要在XIPKERNEL內部
"AutoMount"=dword:1
"AutoPart"=dword:1
"MountFlags"=dword:11
"Folder"="ResidentFlash"
"Name"="Samsung Flash Disk"
"BootPhase"=dword:0??;要在第一階段加載binfs
"Flags"=dword:1000
"MountHidden"=dword:0 ? ;有了這個你就可以在/BINFS目錄下看到所有的NK.bin的東東了
;###########################################################################
;###########################################################################
;這個部分是設置保存system.hv的磁盤的驅動程序,每個人不一樣了,但是大同小異
;我這里用的是PoketStroeII15的Flash驅動,system.hv保存在第一個Flash分區上
IF BSP_POCKETSTORE
[HKEY_LOCAL_MACHINE\Drivers\BuiltIn\PocketStore]
? "Prefix"="DSK"
? "Dll"="ONDisk.dll"?? ;這個是在binfs之后加載,所以可以放在NK.bin中
? "Order"=dword:1
? "Profile"="PocketStore"
? "IClass"=multi_sz:"{A4E7EDDA-E575-4252-9D6B-4195D48BB865}"
? "BmlVolumeId"=dword:0 ; BML volume ID = 0
? "BmlPartitionId"=dword:8 ; BML parition ID = PARTITION_ID_FILESYSTEM
? "Index"=dword:2 ? ?
? "Flags"=dword:1000 ? ;這個flag指定這個驅動只在boot.hv中加載一次
[HKEY_LOCAL_MACHINE\System\StorageManager\Profiles\PocketStore]
? "DefaultFileSystem"="FATFS"
? "PartitionDriver"="mspart.dll"
? "AutoMount"=dword:1
? "AutoPart"=dword:1
? "AutoFormat"=dword:1
? "MountAsBootable"=dword:1 ? ;這個是wince 5.0下指定這個分區保存system.hv的關鍵
? "Folder"="HDD"
? "Name"="NAND Drive"
? "Ioctl"=dword:4
[HKEY_LOCAL_MACHINE\System\StorageManager\Profiles\PocketStore\FATFS]
? "EnableCacheWarm"=dword:0
?
ENDIF
;###########################################################################
;END HIVE BOOT SECTION
補充:
我的Flash劃分了四個區,第一個區是放4k的只能以nor模式運行的bootloader,然后第二個區放置UT,第三個區放置XIPKERNEL和BINFS,第四個區是將剩下的所有的扇區格式化為一個FAT分區作為文件系統,system.hv就是放置在最后一個分區中,在wince起來之后可以看到有個document and setting文件夾,里面的hv文件都是隱藏的。
我們用的是三星的ONENAND的flash,其實就是擁有4K NOR的NAND flash,我們的bootloader也分成三部份,第一部分bootloader主要是映射到0x00000000地址的一些跳轉指令,這個部分會被燒寫到flash的前4K里面,然后三星的ONENAND會自動復制前4K的數據到一個類似NOR的物理模塊中,這個NOR模塊支持CPU的直接尋址;第二部份叫IPL,它的功能是加載在NAND flash中的image或UT,然后在加載后跳轉到其RAM中的入口去執行,因為CPU的數據線和地址線在這個時候還只能直接訪問NOR flash,要訪問NAND flash的話必須要有NAND的接口驅動,所以在IPL的部分會有NAND的接口驅動的代碼,這就導致IPL的代碼一般有幾十到上百K,我們的flash一個塊是128K,前兩部分一共占了兩個塊;第三個部分是UT,就是一些通用的工具,比如燒image,燒bootloader,格式化flash等常用的維護和image升級工具,這個部分的數據最多包括了很多的驅動程序,體積也很大,有300K的樣子。
最后我們的三個部分的bootloader一共占據了flash的頭10個塊(block),128K*10,但這三個部分在三星的flash的分區中是兩個BML分區(三星的flash驅動PocetStoreII里面的概念,你就把它看成普通的磁盤分區好了),等下后面給的圖示可以看到。
接下來的塊會放置一個MBR,然后從11個塊開始我們放置wince的image了,這個區是第三個BML分區,大小一般在40-200個塊左右,因為wince的image也就在4M-20M左右,這些劃分分區的工作都是由上面提到的UT去做的,我也沒有仔細看源代碼,只是看到有BMLFormat之類的函數,其參數就是后面看到的圖示。如果是binfs那么這個區會放置兩個模塊:XIPKERNEL和NK,NK的區域會被binfs的驅動識別并且加載成FAT分區的樣子,可以在wince的資源管理器中看到的,具體是怎么被識別的還沒有認真的搞懂。
大概算下來,前面所有的空間也就被占用20多M的樣子,剩下的空間了你可以隨意利用了,分區的方法還是調用三星的flash驅動中的BMLFormat和STLformat,分區的參數定好了它們就會自動把分區建好,這部分的工作我們在BT中完成的。然后WinCE中僅僅需要將這些分區讀出來并顯示成磁盤就行了,這個就和SD、HDD的驅動相似了,參考上面的介于“IF BSP_POCKETSTORE”部分的注冊表寫法就行了。
總結
以上是生活随笔為你收集整理的S3c2440A平台HIVE注册表+binfs的实现的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 教你第一时间升级更新!iOS 16 Be
- 下一篇: WinCE的I2C驱动程序设计