Windows CE下驅動(dòng)程序開(kāi)發(fā)基礎(二)
ULONG inIoSpace = 1; ///1表示是IO空間 PHYSICAL_ADDRESS ioPhysicalBase = {iobase, 0}; ///相當于ioPhysicalBase.LowPart = iobase |
在地址轉換后就要將轉換后的地址映射到驅動(dòng)程序(一般IST和應用程序一樣運行在用戶(hù)模式)能夠訪(fǎng)問(wèn)的虛擬地址空間(0x80000000以下)和ISR能夠訪(fǎng)問(wèn)的靜態(tài)虛擬地址空間中(0x80000000以上)。例如:
////如果地址屬于物理地址空間 ioPortBase = (PUCHAR)MmMapIoSpace(ioPhysicalBase, Size, FALSE); TransBusAddrToStatic(Isa, 0, ioPhysicalBase, Size, inIoSpace, ppStaticAddress); |
MmMapIoSpace函數負責將物理地址映射到驅動(dòng)程序能夠訪(fǎng)問(wèn)的虛擬地址空間中,通過(guò)源碼分析MmMapIoSpace在內部分別調用:
pVirtualAddress =VirtualAlloc(0, SourceSize, MEM_RESERVE, PAGE_NOACCESS); VirtualCopy(pVirtualAddress, (PVOID)(SourcePhys >> 8), SourceSize, PAGE_PHYSICAL | PAGE_READWRITE | (CacheEnable ? 0 : PAGE_NOCACHE)); |
VirtualAlloc分配一塊和MemLen一樣大小的虛擬地址空間,因為參數1為0,所以?xún)群俗詣?dòng)分配。一般MemLen小于2MB,所以會(huì )在應用程序的地址空間中分配。VirtualCopy負責將硬件設備寄存器的物理地址與VirtualAlloc分配的虛擬地址做一個(gè)映射關(guān)系,這樣驅動(dòng)程序訪(fǎng)問(wèn)PvirtualAddress實(shí)際上就是訪(fǎng)問(wèn)第一個(gè)寄存器。因為硬件設備寄存器的物理地址一定是在512MB(CE支持RAM的最大值)以上,所以除了最后的參數要加PAGE_PHYSICAL外,第二個(gè)參數物理地址也要右移8位(或者除以256)。
映射硬件寄存器當然PAGE_NOCACHE是必須加的。TransBusAddrToStatic函數負責將物理地址映射到ISR能夠訪(fǎng)問(wèn)的靜態(tài)虛擬地址空間中,當出現中斷共享時(shí),ISR要負責訪(fǎng)問(wèn)硬件設備的某一個(gè)寄存器來(lái)判斷中斷源,所以將寄存器的物理地址映射到靜態(tài)虛擬地址空間中是必要的(ISR只能訪(fǎng)問(wèn)靜態(tài)的虛擬地址空間)。所謂靜態(tài)虛擬地址空間是指在OEMAddressTable中定義的虛擬地址空間(當然是0x80000000以上)。在x86平臺一般這個(gè)表只定義RAM的物理地址與虛擬地址對應關(guān)系,而硬件設備的寄存器地址并不在該表中定義,所以如果要創(chuàng )建一塊靜態(tài)的虛擬地址空間供ISR訪(fǎng)問(wèn),必須在此之前調用CreateStaticMapping函數在0xC4000000到0xE0000000虛擬地址空間中分配。TransBusAddrToStatic函數在內部就是調用了CreateStaticMapping函數。注:硬件設備的寄存器地址也可以在OEMAddressTable中定義。
////如果地址屬于IO空間 ioPortBase = (PUCHAR)ioPhysicalBase.LowPart; *ppStaticAddress=ioPortBase |
這種情況只屬于x86平臺,是IO空間就可以直接訪(fǎng)問(wèn),即使是用戶(hù)模式。
SerInit函數接著(zhù)初始化SER_INFO結構體成員,之后調用SL_Init函數,這個(gè)函數在ser16550中定義,負責初始化SER16550_INFO結構體,在這個(gè)結構體中保存串口8個(gè)寄存器的地址。SerInit函數執行完畢后COM_Init函數創(chuàng )建接收緩沖區,然后調用StartDispatchThread函數初始化中斷并且創(chuàng )建IST。StartDispatchThread函數在內部調用InterruptInitialize函數關(guān)聯(lián)SysIntr和Event,然后調用InterruptDone函數告訴內核當前串口可以中斷處理,接著(zhù)調用CreateThread函數創(chuàng )建IST線(xiàn)程。(over吧,再往下說(shuō)就和串口硬件有關(guān)了,看多了沒(méi)注釋的代碼我也煩?。。?/p>
評論