Windows CE下驅動(dòng)程序開(kāi)發(fā)基礎(一)
這是我從1月6日開(kāi)始主持天極網(wǎng)論壇嵌入式開(kāi)發(fā)版以來(lái)第一次發(fā)表文章,加上以前瑣碎的文章共計30篇。研究的越多就越感覺(jué)自己懂的太少,其實(shí)在驅動(dòng)開(kāi)發(fā)方面我還是個(gè)菜鳥(niǎo),我是想再次拋磚引玉,讓做驅動(dòng)有N年經(jīng)驗的人奉獻一點(diǎn)出來(lái),讓大家減少一些研究驅動(dòng)源碼而又缺少注釋所帶來(lái)的痛苦。
我想即使讀者看過(guò)微軟的關(guān)于驅動(dòng)開(kāi)發(fā)的培訓教材和CE幫助文檔中的驅動(dòng)部分,頭腦中仍然一片茫然。要想真正了解驅動(dòng)程序必須結合一些驅動(dòng)程序源碼,在此我以串口驅動(dòng)程序(COM16550)中初始化過(guò)程為線(xiàn)索簡(jiǎn)單講一講驅動(dòng)開(kāi)發(fā)的基礎知識。
Windows CE下的串口驅動(dòng)程序能夠處理所有I/O行為類(lèi)似串口的設備,包括基于16450、16550 UART(通用異步收發(fā)芯片)的設備和一些采用DMA的設備,常見(jiàn)的有9針串口、紅外I/O口、Modem等。在%_WINCEROOT%PublicCommonOAKDriversSerial目錄下,COM_MDD2子目錄包含新的串口驅動(dòng)MDD層函數代碼。COM16550子目錄包含串口驅動(dòng)PDD層代碼。SER16550子目錄包含的一系列函數專(zhuān)用于控制與16550兼容的UART,這樣PDD層的主要工作就是調用SER16550中的函數。還有一個(gè)ISR16550子目錄包含的是串口驅動(dòng)程序專(zhuān)用的可安裝ISR(中斷服務(wù)例程),而很多硬件設備驅動(dòng)程序采用CE默認的可安裝ISR giisr.dll。一般串口設備相應的注冊表設置例子及意義如下:
[HKEY_LOCAL_MACHINEDriversBuiltInSerial_1] |
鍵 | 意義 |
SysIntr=dword:13 | 串口1的中斷ID為十進(jìn)制13 |
IoBase=dword:02F8 | 串口1的IO空間首地址為十六進(jìn)制2F8 |
IoLen=dword:8 | 串口1的IO空間長(cháng)度為8個(gè)字節 |
DeviceArrayIndex=dword:0 | 串口1的索引,是1的由來(lái) |
Order=dword:0 | 串口1驅動(dòng)的加載順序 |
DeviceType=dword:0 | 串口1的設備類(lèi)型 |
DevConfig=hex: 10,00 .... | 串口1在與Modem設備通訊時(shí)的配置,如波特率、奇偶校檢等 |
FriendlyName=COM1: | 串口1在撥號程序中顯示的名字 |
Tsp=Unimodem.dll | 串口1 被用于與Modem設備通訊的時(shí)候要加載的TSP(TAPI Service provider)DLL |
Prefix=COM | 串口1的流接口的前綴 |
Dll=com16550.Dll | 串口1的驅動(dòng)程序DLL |
下面從MDD層函數COM_Init開(kāi)始探索串口驅動(dòng)的初始化過(guò)程。COM_Init是在串口設備被檢測后由設備管理器device.exe調用的,主要的作用是初始化設備,它的唯一參數Identifier是由device.exe傳遞的,其類(lèi)型是一個(gè)字符串指針,字符串的內容是HLMDriversActivexx,xx是一個(gè)十進(jìn)制數(device.exe會(huì )跟蹤系統中每個(gè)驅動(dòng)程序,把加載的驅動(dòng)程序記錄在A(yíng)ctive鍵下)。
COM_Init先分配一個(gè)HW_INDEP_INFO結構體,這個(gè)結構體是獨立于串口硬件的頭信息(MDD、PDD、SER16550都包含自己獨特的結構體,具體的結構體定義請參見(jiàn)串口驅動(dòng)源碼),分配之后再初始化結構體中每個(gè)成員,初始化結構體后調用 OpenDeviceKey((LPCTSTR)Identifier)打開(kāi)HLMDriversActivexxKey包含的注冊表路徑,在這里路徑一般為HLMDriversBuiltInSerial,即串口的驅動(dòng)程序信息在注冊表中所處的位置。COM_Init接著(zhù)在HLMDriversBuiltInSerial下查詢(xún)DeviceArrayIndex、Priority256的值,Priority256指定了驅動(dòng)程序的優(yōu)先級,如果沒(méi)有就用默認的優(yōu)先級。接下來(lái)調用GetSerialObject(DeviceArrayIndex),這個(gè)函數由PDD層定義,返回HWOBJ結構體,這個(gè)結構體主要包含PDD層和SER16550定義的函數的指針。
也就是說(shuō)MDD通過(guò)調用這個(gè)函數才能調用底層實(shí)現的函數。接下來(lái)的大多數工作都是調用底層函數實(shí)現初始化。第一個(gè)調用的底層函數SerInit主要設置由用戶(hù)設置的硬件配置,例如線(xiàn)路控制、波特率。它調用Ser_GetRegistryData函數得到保存在注冊表中的硬件信息,Ser_GetRegistryData在內部調用系統提供的DDKReg_GetIsrInfoDDK和DDKReg_GetWindowInfo函數得到在HLMDriversBuiltInSerial下保存的IRQ、SysIntr、IsrDll、IsrHandler、IoBase、IoLen。IRQ是邏輯中斷號,IsrDll表示當前驅動(dòng)程序的可安裝ISR所在的DLL名稱(chēng),IsrHandler 表示可安裝ISR的函數名稱(chēng)。
在這里順便提一下可安裝ISR,讀者在我以前發(fā)表的關(guān)于OAL的文章中可以了解到OEM在OEMInit函數中關(guān)聯(lián)IRQ和SysIntr,當硬件設備發(fā)生中斷時(shí),ISR會(huì )禁止同級和低級中斷,然后根據IRQ返回關(guān)聯(lián)的SysIntr,內核根據ISR返回的SysIntr喚醒相應的IST(SysIntr與IST創(chuàng )建的Event關(guān)聯(lián)),IST處理中斷之后調用InterruptDone解除中斷禁止。在OEMInit中關(guān)聯(lián)的缺點(diǎn)是一旦編譯了CE內核后就無(wú)法添加這種關(guān)聯(lián)了,而一些硬件設備會(huì )隨時(shí)插拔或者共享中斷,要關(guān)聯(lián)這樣的硬件設備解決方法就是可安裝ISR,可安裝ISR專(zhuān)用于處理指定的硬件設備發(fā)出的中斷,所以如果硬件設備需要可安裝ISR必須在注冊表中添加IsrDll、IsrHandler。多數硬件設備采用CE默認的可安裝ISR giisr.dll,格式如下:
IsrDll=giisr.dll IsrHandler=ISRHandler |
如果一個(gè)硬件驅動(dòng)程序需要可安裝ISR而開(kāi)發(fā)者又不想自己寫(xiě)一個(gè),那么可以利用giisr.dll來(lái)實(shí)現。除了在注冊表中添加如上所示外,還要在驅動(dòng)程序中調用相關(guān)函數注冊可安裝ISR。偽代碼如下:
g_IsrHandle = LoadIntChainHandler(IsrDll, IsrHandler, (BYTE)Irq); GIISR_INFO Info; PHYSICAL_ADDRESS PortAddress = {PhysAddr, 0}; TransBusAddrToStatic(BusType, dwBusNumber, PortAddress, dwAddrLen, dwIOSpace, (PVOID)PhysAddr) Info.SysIntr = dwSysIntr; Info.CheckPort = TRUE; Info.PortIsIO = (dwIOSpace) ? TRUE : FALSE; Info.UseMaskReg = TRUE; Info.PortAddr = PhysAddr + 0x0C; Info.PortSize = sizeof(DWORD); Info.MaskAddr = PhysAddr + 0x10; KernelLibIoControl(g_IsrHandle, IOCTL_GIISR_INFO, Info, sizeof(Info), NULL, 0, NULL); |
LoadIntChainHandler函數負責注冊可安裝ISR,參數1為DLL名稱(chēng),參數2為ISR函數名稱(chēng),參數3為IRQ。TransBusAddrToStatic函數在后面講。如果要利用giisr.dll作為可安裝ISR,必須先填充GIISR_INFO結構體,CheckPort=TRUE表示giisr要檢測指定的寄存器來(lái)確定當前發(fā)出中斷的是否是這個(gè)設備。PortIsIO表示寄存器地址屬于哪個(gè)地址空間,FALSE表示是內定空間,TRUE表示IO空間。UseMaskReg=TRUE表示設備有一個(gè)掩碼寄存器,專(zhuān)用于指定當前設備是否是中斷源,也就是發(fā)出中斷,而MaskAddr表示掩碼寄存器的地址。如果對Info.Mask賦值,那么PortAddr表示一個(gè)特殊的寄存器地址,這個(gè)寄存器的值與Mask的值運算的結果如果為真,則證明當前設備是中斷源,否則返回SYSINTR_CHAIN(表示當前ISR沒(méi)有處理中斷,內核將調用ISR鏈中下一個(gè)ISR),如果UseMaskReg=TRUE,那么MaskReg寄存器的值與PortAddr指定的寄存器的值運算的結果如果為真,則證明當前設備是中斷源。
評論