劉士勛,劉滿國,唐同斌,喻 戈,岳 超
(西安現代控制技術研究所, 西安 710065)
RTX系統憑借其兼容了Windows良好的人機交互特性的原因,逐漸走入各大高校、研究所的實驗室內。然而,板卡出廠時大多不提供RTX驅動,即使有些板卡提供RTX驅動,這些驅動往往沒有源代碼,同時使用起來也未必方便,出現問題也不好定位[3]。對于大量使用RTX系統的研究生產單位而言,了解、掌握和開發板卡RTX驅動程序是相當必要的。當前對RTX驅動的研究工作還不是很多,已有的成果大多也是基于Rfm2g反射內存卡的。因此,文中以Moxa CP-118U板卡為例,對RTX驅動程序開發問題進行討論,以求該技術能被更多探索。
Windows操作系統的架構相對復雜,用戶使用板卡API訪問底層硬件,需從頂層開始,層層穿越,跨越用戶模式和內核模式,才能訪問硬件,驅動程序可以理解為Windows內核模式下的一塊“補丁”,所有驅動程序都需要按照該模式開發。
相比Windows驅動程序模型,RTX程序就相對靈活,RTX應用程序可繞過Windows內核直接通過RTX內核訪問RTX硬件擴展層,直接通往硬件進行訪問[5,8]。RTX系統構架示意如圖1所示。

圖1 RTX框架簡圖
當板卡安裝好Windows驅動后,在RTX管理器對該板卡增加RTX支持,然后在設備管理器轉換驅動為RTX下的驅動后,即可進行RTX驅動程序的開發工作。
PCI板卡驅動的開發實質是操縱板卡上芯片的寄存器。每個寄存器有其特定的作用,只要了解每個寄存器的作用及其地址,通過寄存器地址對板卡上的寄存器置數取數,即可完成數據交互等功能。
PCI板卡的各類寄存器的訪問地址均存放于一個稱作“PCI配置空間”的256 Byte的內存中[10]。這個空間的前64 Byte是PCI協議標準預定義的,所有PCI板卡都會使用該標準。查詢板卡硬件手冊,即可得到板卡寄存器地址。有些板卡如果使用了其他芯片,其硬件手冊會告訴用戶查找對應芯片的手冊以尋找寄存器的作用及地址。
板卡驅動程序的第一步便是驗證該板卡是否存在于當前計算機系統中。
為了獲得指定板卡的PCI配置空間內存,RTX系統為編程人員提供了RtGetBusDataByOffset函數。該函數的參數有總線類型、PCI總線編號、插槽編號,傳入正確參數即可獲得PCI配置空間的參數,比較DeviceID和VendorID即可判斷該板卡是否為指定板卡。如果遍歷所有總線以及所有總線插槽仍無結果,則搜尋該板卡失敗,程序退出或返回失敗。
Moxa CP-118U板卡給用戶提供I/O映射的方式來訪問寄存器。寄存器的基地址存儲在PCI配置空間的基址寄存器2中,保存該值方便后續對各個串口通道的相關寄存器進行訪問。
操作板卡的核心便是讀寫板卡上的各個寄存器。深刻理解寄存器的意義和用法對操縱板卡有相當重要的幫助。
Moxa CP-118U板卡的配置包括各個串口通道的波特率的設置、數據位長度、奇偶校驗位的設置等。
板卡一般都提供了中斷標識寄存器。中斷服務函數中就是對中斷標識寄存器中的各個中斷二進制位進行判斷,如果某個二進制位為1則對該中斷進行響應。
Moxa CP-118U板卡有8路RS 422/485串口,每路串口都配備了一塊TL16C550C異步串并行收發器轉換芯片,通過操作對應串口的芯片,可以順利的完成數據的傳輸。其中,每路串口的寄存器共占有8 Byte,8路串口總共占用64 Byte,如圖2所示。

圖2 板卡寄存器組分配示意
對于每路串口,其8 Byte寄存器分配如圖3所示。

圖3 每路串口寄存器分配示意
欲操作一路串口,需對該路串口對應芯片的寄存器有如下了解:
1)配置鏈路控制寄存器LCR:完成數據位長度設置(5~8 bit),停止位長度設置(1,1/2,2),奇偶校驗位設置;
2)中斷使能寄存器IER:完成各類中斷的使能與關閉;
3)中斷標識寄存器IIR:對當前是否發生中斷與當前中斷類型進行標識;
4)發送保持寄存器THR與接收緩沖區寄存器RBR:串口發數和接數會分別頻繁使用這兩個寄存器;
5)波特率分辨率寄存器DLL、DLM:配置這兩位寄存器完成串口波特率設置。
有了以上了解,即可開始驅動程序的開發。
在打開板卡函數的實現中,主要操作為根據DeviceID和VendorID查找板卡,如果搜尋到板卡,保存板卡的I/O映射地址,方便后續操作寄存器使用。
示例代碼如下:
for ( bus=0; bFlag; bus++ )
for(deviceNumber=0;deviceNumber for(functionNumber=0;functionNumber { bytesWritten = RtGetBusDataByOffset(…) if((PciData->VendorID==vendorID) && (PciData->DeviceID==deviceID)) …;//找到板卡 } 其中,PciData是通過RTX系統提供的RtGetBusDataByOffset接口所獲得的PCI配置空間的內存指針,將該內存中的DeviceID和VendorID成員與118U板卡的進行對比,即可驗證當前所遍歷板卡是否為118U板卡。對于Moxa CP-118U而言,DeviceID為0x1180,VendorID為0x1393。 在打開串口函數中,主要操作就是對欲使用的串口進行參數設置,其核心就是操縱對應串口的LCR寄存器。 1)波特率設置 Moxa CP-118U板卡上有參考輸入脈沖,其頻率為XIN=14.745 6 MHz,對每路串口的波特率設置就是對該脈沖進行分頻,分頻因子計算方法如公式(1)所示: (1) 例如,欲設置波特率為115 200 bit/s,則分頻因子divisor=14.745 6×106/115 200/16=8。 下面的工作就是將分頻因子寫入到分頻寄存器DLL和DLM中即可,需要注意的是,讀寫DLL和DLM,需先置LCR的BIT7為1。示例代碼如下: RtWritePortUchar(moxa118uBaseAdd+(port-1)*8+LCR, (UCHAR)(c|0x80)); RtWritePortUchar(moxa118uBaseAdd+(port-1)*8+DLL,(UCHAR)(divisor&0x00FF)); RtWritePortUchar(moxa118uBaseAdd+(port-1)*8+DLM,(divisor>>8)&0x00FF); 2)數據位設置 配置LCR的BIT0和BIT1可以設置數據位長度,示例代碼如下: setting=WordLength-5; 3)停止位 配置LCR的BIT2可以設置停止位長度,示例代碼如下: setting |= 0x00;//1位停止位 4)校驗位 配置LCR的BIT3-BIT5可以設置校驗位使能及校驗位類型。 if(parity ==PARITY_DISABLE_118){setting |=0x00;} else if(parity == PARITY_ODD_118){setting |=0x08;} else if (parity == PARITY_EVEN_118){setting |=0x18;} RtWritePortUchar(moxa118uBaseAdd+(port-1)*8+LCR,(UCHAR)setting); 使用RtAttachInterruptVector函數掛接中斷服務函數。 使用中斷模式向外發數,具體實現為向發數寄存器寫數,當數據發送成功后,會進入2號中斷(“發送保持寄存器空”中斷),在該中斷發生時繼續給發數寄存器寫入待發送的數據。循環這個過程直至欲發送的數據全部發送結束。 a)WriteMoxaPort的核心代碼示例 WriteCounter[port-1] = 0;//清理已發送數據計數器 WriteDataLength[port-1] = iLength;//記錄發送數據長度 RtWritePortUchar(moxa118uBaseAdd+(port-1)*8+THR,WriteBuffer[port-1][WriteCounter[port-1]++]);//向發送寄存器寫數,當前已發送計數器+1 b)中斷服務函數相關實現示例 if(WriteCounter[port-1] { RtWritePortUchar(moxa118uBaseAdd+(port-1)*8+THR,WriteBuffer[port-1][WriteCounter[port-1]++]); //繼續向緩沖區寫數當前已發送計數器+1 } else { WriteDataLength[port-1]=0; //結束本次發送,清空標志位 WriteCounter[port-1] =0; } 收數是一個被動的過程,當串口總線有數據到來并且超過中斷觸發深度時會自動觸發收數中斷,觸發之前,數據會存儲在板卡的硬件緩沖區內。由于板卡的硬件緩存只有128 Byte,因此需要在驅動程序層增加一層緩沖區。文中設計的ReadMoxaPort函數的參數列表中有一項為用戶提供欲接收數據的長度,當數據接收量大于該值時返回,具體流程如下: 1)當數據來臨時存儲在驅動層緩沖區內; 2)定時器檢測數據長度到達用戶欲得到的數據長度; 3)當數據長度滿足要求后,將數據填入用戶傳進的緩沖區內存,返回結束。 具體實現如下: a)中斷服務函數的核心實現 ReadBuffer[port-1][ReadCounter[port-1]++]=RtReadPortUchar(moxa118uBaseAdd+RBR); b)定時器函數的核心實現 if(ReadCounter[port-1]-ReadPostion[port-1]>=iDrvTriggerLevel[port-1]) RtSetEvent(hRecWaitEvent[port-1]); c)ReadMoxaPort的核心實現 if(RtWaitForSingleObject(hRecWaitEvent[port-1],INFINITE)==WAIT_FAILED) return-3; for(counter=0;counter Buffer[counter]=ReadBuffer[port-1][ReadPostion[port-1]++]; 使用RTX Application Wizard創建RTX應用程序,加載使用上述方法創建的驅動程序靜態庫對驅動程序性能進行測試,測試結果如表1所示。 表1 驅動性能測試結果 由于高波特率條件下中斷頻繁觸發,過于占用計算機資源,同時也受計算機CPU性能等因素影響,921 600 bit/s波特率的條件下最多容許4路串口穩定工作,超過4個串口會丟失數據;230 400 bit/s及以下波特率條件下8路串口可以穩定工作。該性能滿足了絕大多數工作的條件,可以用于日常仿真研究工作。 文中以Moxa CP-118U板卡為例,介紹了在RTX實時系統下PCI板卡的驅動編寫方法,出色的實現了板卡提供的所有功能,驅動性能可以滿足絕大多數工業、生產、仿真的要求。同時對于Moxa公司其他類型的多串口卡,亦可借鑒文中列舉的方法和框架進行開發驅動。3.2 打開串口——OpenMoxa118UPort

3.3 打開中斷模式——OpenInterruptMode
3.4 中斷發數——WriteMoxaPort
3.5 中斷收數——ReadMoxaPort
4 RTX驅動程序測試

5 結束語