Windows CE下驅動(dòng)程序開(kāi)發(fā)基礎
也就是說(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指定的寄存器的值運算的結果如果為真,則證明當前設備是中斷源。
函數SerInit接著(zhù)調用函數Ser_InternalMapRegisterAddresses轉換IO地址并且映射地址,Ser_InternalMapRegisterAddresses在內部調用系統提供的HalTranslateBusAddress(Isa, 0, ioPhysicalBase, inIoSpace, ioPhysicalBase)函數將與總線(xiàn)相關(guān)的地址轉換為系統地址,參數1為總線(xiàn)類(lèi)型,參數2為總線(xiàn)號,參數3為要轉換的地址(PHYSICAL_ADDRESS類(lèi)型,實(shí)際是LARGE_INTEGER型),參數4指定寄存器地址屬于IO地址空間還是物理地址空間,參數5返回轉換后的物理地址。觀(guān)察HalTranslateBusAddress的源碼得知如果是在x86平臺,這個(gè)函數除了把參數3賦給了參數5其余什么都沒(méi)有做,而非x86平臺將inIoSpace的值置為0,表示一定是物理地址。在調用HalTranslateBusAddress前要確定從注冊表中得到的寄存器地址到底是屬于哪個(gè)地址空間的,例如:
ULONG inIoSpace = 1; ///1表示是IO空間 PHYSICAL_ADDRESS ioPhysicalBase = {iobase, 0}; ///相當于ioPhysicalBase.LowPart = iobase |
評論