張德迪
[摘 要]Windows CE是一種嵌入式實(shí)時(shí)性的操作系統(tǒng),在內(nèi)存管理方面必須要比其它Windows操作系統(tǒng)更節(jié)約物理內(nèi)存和虛擬地址空間。
[關(guān)鍵詞]Windows CE 內(nèi)存管理
[中圖分類號(hào)]TP311 [文獻(xiàn)標(biāo)識(shí)碼]A[文章編號(hào)]1007-9416(2009)11-0022-01
Windows CE.NET支持32位虛擬內(nèi)存機(jī)制、按需分配內(nèi)存和內(nèi)存映射文件,但是與其它Windows操作系統(tǒng)又有明顯的不同。畢竟Windows CE是一種嵌入式實(shí)時(shí)性的操作系統(tǒng),在內(nèi)存管理方面必須要比其它Windows操作系統(tǒng)更節(jié)約物理內(nèi)存和虛擬地址空間。
對于早期采用的存儲(chǔ)設(shè)備一般采用ROM+RAM,在ROM中存放的所有文件可以是壓縮的也可以是不壓縮的,這取決于OEM(原始設(shè)備制造商)。OEM在定制CE內(nèi)核時(shí)可以設(shè)置一個(gè)標(biāo)志告訴ROM鏡像制作工具(romimage.exe)是否壓縮文件。對于ROM中存放的模塊(DLL、EXE文件)來說,如果是壓縮的,模塊在運(yùn)行前先解壓=并全部存放到RAM中。如果是不壓縮的,并且ROM介質(zhì)支持線性訪問(line-accessed),就可以本地執(zhí)行(executed in place,縮寫為XIP)。利用本地執(zhí)行方式運(yùn)行應(yīng)用程序、DLL的優(yōu)點(diǎn)是:采用這種技術(shù)在加載EXE或DLL時(shí),其中的代碼段數(shù)據(jù)不加載到物理內(nèi)存中,內(nèi)核只是分配虛擬地址空間給代碼段,當(dāng)執(zhí)行代碼時(shí)內(nèi)核會(huì)到實(shí)際存放在ROM存儲(chǔ)設(shè)備上的文件中尋找代碼并執(zhí)行。采用這樣的技術(shù)既可以節(jié)省可用內(nèi)存又可以減少加載的時(shí)間。但是這種技術(shù)有一定的局限性,如果要讓CPU到ROM中去尋找代碼執(zhí)行,那么ROM介質(zhì)必須支持線性訪問,這就要求ROM介質(zhì)支持線性訪問,而不是塊訪問。XIP這種加載方式的缺點(diǎn)是執(zhí)行速度相對較慢,CPU訪問ROM的速度要慢于訪問RAM的速度。
基于Windows CE的產(chǎn)品開始采用FLASH、IDE等永久存儲(chǔ)設(shè)備時(shí),內(nèi)核鏡像和其它應(yīng)用程序文件開始存放到永久存儲(chǔ)設(shè)備中而不是ROM中,這不僅僅是因?yàn)橛脖P或者FLASH的I/O速度比ROM快,更因?yàn)楝F(xiàn)在的內(nèi)核包含的功能多并且文件數(shù)量增加,因而需要的存儲(chǔ)空間很大,一般都在20MB左右。再加上其它開發(fā)商開發(fā)的應(yīng)用程序文件,要求的空間就更大了。CE啟動(dòng)時(shí)內(nèi)核鏡像由加載程序解壓并將系統(tǒng)文件加載到RAM的NK,NK是在config.bib中定義的一段RAM區(qū)域,專用于保存內(nèi)核鏡像解壓出來的所有文件。Windows CE將NK看作是ROM,當(dāng)執(zhí)行一個(gè)應(yīng)用程序時(shí),CE內(nèi)核將這個(gè)應(yīng)用程序需要的系統(tǒng)DLL(在NK中保存)加載到Slot 1(地址范圍0x0200 0000-0x03FF FFFF,在Windows CE.NET中Slot 1專用于XIP DLL使用)。Slot 1是一段虛擬地址,當(dāng)CPU執(zhí)行DLL的代碼時(shí),CPU會(huì)根據(jù)地址映射關(guān)系到NK中尋找實(shí)際的代碼執(zhí)行,因?yàn)镹K是一段實(shí)際的物理內(nèi)存,I/O速度非???所以相對于在ROM中執(zhí)行,DLL的運(yùn)行效率得到很大提高。
RAM和ROM文件系統(tǒng)是Windows CE默認(rèn)的文件系統(tǒng)。RAM文件系統(tǒng)的優(yōu)點(diǎn)是支持文件壓縮、支持事務(wù)機(jī)制(和數(shù)據(jù)庫中的事務(wù)機(jī)制相似)、數(shù)據(jù)I/O較快。Windows CE.NET啟動(dòng)時(shí)把除了NK以外的RAM分為對象存儲(chǔ)(object store)區(qū)域和應(yīng)用程序內(nèi)存(program memory)區(qū)域,并且默認(rèn)各使用一半RAM。在基于Windows CE的設(shè)備沒有采用永久存儲(chǔ)器之前,對象存儲(chǔ)的作用相當(dāng)于永久存儲(chǔ)器,對象存儲(chǔ)區(qū)域采用RAM文件系統(tǒng)來保存文件,對象存儲(chǔ)中可以存儲(chǔ)的對象類型有文件、目錄、數(shù)據(jù)庫、記錄、數(shù)據(jù)庫卷。默認(rèn)在對象存儲(chǔ)中存儲(chǔ)的對象全部是壓縮的。當(dāng)整個(gè)系統(tǒng)關(guān)閉時(shí),設(shè)備的電源還繼續(xù)提供電力給RAM,這樣對象存儲(chǔ)中保存的所有數(shù)據(jù)就不會(huì)丟失。應(yīng)用程序內(nèi)存區(qū)域留給所有應(yīng)用程序運(yùn)行時(shí)使用?;赪indows CE的設(shè)備采用永久存儲(chǔ)器后,對象存儲(chǔ)的作用就被永久存儲(chǔ)器替代了,所以采用永久存儲(chǔ)器后,應(yīng)該減小對象存儲(chǔ)區(qū)域的大小。如果定制的Windows CE的內(nèi)核包含了資源管理器,那么打開“控制面板”,在“系統(tǒng)”-“內(nèi)存”中,可以調(diào)節(jié)這兩個(gè)存儲(chǔ)區(qū)域的比例。滑塊向左,則釋放對象存儲(chǔ)區(qū)域的一些可用內(nèi)存并將這些內(nèi)存劃到應(yīng)用程序內(nèi)存區(qū)域中?;瑝K向右則相反。
Windows CE.NET只能管理512MB的物理內(nèi)存和4GB大小的虛擬地址空間。不同的CPU內(nèi)存管理方法也不同。對于MIPS和SHX系列CPU來說,物理地址映射是由CPU完成的,CE內(nèi)核可以直接訪問512MB的物理內(nèi)存。對于x86系列和ARM系列的CPU來說,在內(nèi)核啟動(dòng)過程中它會(huì)將現(xiàn)有物理內(nèi)存地址全部映射到0x8000 0000以上的虛擬地址空間中供內(nèi)核以后使用。OEM可以通過OEMAddressTable來詳細(xì)定義虛擬地址和物理地址的映射關(guān)系。OEMAddressTable本身并不是一個(gè)文件,它只是存在于其它文件中描述虛擬地址和實(shí)際物理地址的映射關(guān)系的數(shù)據(jù)。比如文件oem init.asm中包含一段代碼:dd 80000000h, 0, 04000000h。它表示將整個(gè)物理地址(0x0400 0000=64MB)共64MB映射到虛擬地址從0x8000 0000到0x8400 0000中。
當(dāng)一個(gè)應(yīng)用程序啟動(dòng)時(shí),內(nèi)核為這個(gè)程序選擇一個(gè)空閑的槽(Slot),并且加載所有的代碼、資源,并分配堆棧,加載DLL等。當(dāng)這個(gè)進(jìn)程得到CPU使用權(quán)時(shí),它的整個(gè)地址空間被內(nèi)核映射到Slot0,也就是當(dāng)前進(jìn)程使用的地址空間,然后開始運(yùn)行。圖中給出的地址實(shí)際上是經(jīng)過映射到Slot 0之后的結(jié)構(gòu)。從圖中可以看出,進(jìn)程首先加載代碼段,因?yàn)槊總€(gè)進(jìn)程最低部64KB作為保留區(qū)域,所以代碼段從0x0001 0000開始,內(nèi)核為代碼段分配足夠的虛擬地址空間后,接著分配空間為只讀數(shù)據(jù)和可讀/可寫數(shù)據(jù),接著分配空間為資源數(shù)據(jù),之后分配空間為默認(rèn)堆和棧。非XIP DLL從進(jìn)程最高地址向下開始加載。非XIP DLL的加載按如下規(guī)則:內(nèi)核先檢查要加載的DLL是否被其它進(jìn)程加載過,如果加載過,就做一個(gè)地址的重定位。這樣就避免了整個(gè)系統(tǒng)內(nèi)多次加載相同DLL。如果沒有加載過,就按照從槽的高地址到槽的低地址的順序查找空閑地址空間。然后分配足夠的地址空間用于加載DLL。因?yàn)槊總€(gè)進(jìn)程在執(zhí)行前都要映射到Slot 0,而且進(jìn)程使用的所有DLL可能來自不同的槽(Slot),為避免所有使用的DLL在映射到Slot 0中出現(xiàn)地址空間沖突的現(xiàn)象,內(nèi)核的加載器(Loader)在加載DLL時(shí)會(huì)查找所有槽中加載的DLL的地址,保證在映射到Slot 0時(shí)不會(huì)發(fā)生地址沖突。假如系統(tǒng)內(nèi)有兩個(gè)進(jìn)程,進(jìn)程A只加載了DLL A,進(jìn)程B需要加載DLL A和DLL B,那么進(jìn)程B會(huì)留出DLL A的地址空間,然后加載DLL B,也就是說進(jìn)程B映射到Slot 0時(shí),DLL A的地址空間和DLL B的地址空間是相鄰的,不會(huì)發(fā)生沖突。實(shí)際情況是Windows CE下DLL都很小,而且一個(gè)應(yīng)用程序使用DLL多數(shù)是系統(tǒng)的DLL(存在于Slot 1)。所以目前來看進(jìn)程的地址空間是夠用的。
[參考文獻(xiàn)]
[1]基于Windows CE 平臺(tái)的嵌入式GIS 開發(fā)與應(yīng)用,河南,2003.
[2]劉文峰,李程遠(yuǎn),李善平.D嵌入式L inux 操作系統(tǒng)的研究.
[3]錢靜,蘆東昕.嵌入式軟件虛擬內(nèi)存管理技術(shù)的研究和實(shí)現(xiàn),2008.
[4]付曉軍,夏應(yīng)清,何軒.嵌入式LwIP協(xié)議棧的內(nèi)存管理,湖北,2008.