龔松顯,王光輝
(總參通信訓練基地教研部,宣化075100)
WinCE下GPIO端口驅動大多采用流接口驅動的方式進行開發(fā),在實現(xiàn)上主要依托Platform Builder編譯環(huán)境將驅動程序編譯成動態(tài)鏈接庫,并整合到WinCE內核中。但在實際項目的開發(fā)中,Platform Builder在驅動程序的開發(fā)和調試上不夠便捷,且在整合動態(tài)鏈接庫和WinCE內核時,需要進行繁瑣的參數(shù)配置,整個過程較為復雜。筆者以飛凌S3C2440開發(fā)板為例,給出在Visual Studio編譯環(huán)境中采用動態(tài)加載方式來設計GPIO端口驅動程序的思路,較好地改進了目前常用的GPIO端口驅動設計方法,提高了開發(fā)效率。
關于GPIO端口流驅動的設計原理,大量論文及書籍中均有介紹[1-2],這里不再贅述。其設計步驟也遵循固定的設計模式,按照編寫驅動實現(xiàn)代碼、驅動程序配置、編譯調試的順序進行。
1.1.1 編寫驅動實現(xiàn)代碼
WinCE流式驅動實質上是按照固定的流函數(shù)接口實現(xiàn)的一個動態(tài)鏈接庫[2]。在實現(xiàn)時,創(chuàng)建相關的.cpp文件、.h文件和.def文件,根據(jù)需要實現(xiàn)xxx_Init、xxx_Read、xxx_Write等流接口函數(shù)即可。
1.1.2 驅動程序配置
WinCE配置文件分為源代碼配置文件和鏡像配置文件,通過這兩種配置文件,Platform Builder編譯環(huán)境實現(xiàn)對驅動代碼和內核的編譯、整合。具體包括:
① 創(chuàng)建相關提供注冊信息的.reg文件以供在WinCE注冊表中注冊GPIO驅動程序。
② 在Platform Builder的相關目錄下,配置Dirs文件、Makefile文件、Sources文件和Platform.bib文件中的相關參數(shù),Platform Builder在編譯時按照配置的參數(shù)編譯驅動程序并將其整合到內核鏡像中。
1.1.3 編譯調試
使用Platform Builder編譯環(huán)境編譯驅動程序和WinCE內核,編譯成功之后即得到按照硬件平臺定制的含有驅動程序動態(tài)鏈接庫的內核鏡像文件,當WinCE啟動時,該驅動會被自動加載。
基于以上描述,流接口驅動程序設計方法中存在以下兩個問題:
① 在整個設計過程中,驅動程序的編譯、調試全部依托的是Platform Builder編譯環(huán)境。Platform Builder編譯環(huán)境雖然也能實現(xiàn)對驅動程序的編譯和后期調試,但在實際工程項目的開發(fā)中適用面不廣,在使用操作、功能提供等方面均不如Visual Studio,其在實際工程項目中主要是用來定制WinCE內核。另外,為了將驅動程序整合到WinCE內核中,需要開發(fā)者較為熟悉Platform Builder編譯環(huán)境中對源代碼配置文件和內核鏡像配置文件的修改及原理[3],而大多數(shù)開發(fā)者對此基本上處于“比葫蘆畫瓢”的模仿層次,學習入門門檻較高。
② 在設計中,出于整合資源的目的,將端口驅動程序的動態(tài)鏈接庫編譯到WinCE內核中,但實際上這種方法導致驅動程序的動態(tài)鏈接庫與內核結合過于緊密,每次對驅動程序的任何改動都需要隨后使用Platform Builder編譯環(huán)境對內核重新進行編譯,來回反復、效率低下。從程序設計的角度講,驅動程序和內核分屬不同功能類型的模塊,與程序設計中功能模塊內緊外松的耦合思想不一致。
在前文中已經(jīng)介紹過,在基于流驅動的驅動設計方法中,需要在Platform Builder編譯環(huán)境中配置相關的參數(shù),將驅動程序的動態(tài)鏈接庫文件編譯到WinCE內核中。實際上,WinCE內核具有動態(tài)加載(卸載)驅動文件的能力,驅動文件可以在需要的時候動態(tài)加載進WinCE內核中,在不需要的時候卸載掉[4]。開發(fā)者可以使用Visual Studio編譯環(huán)境替代Platform Builder編譯環(huán)境,在Visual Studio中通過調用相關的WinCE API來實現(xiàn)對驅動文件的加載、卸載。利用這一特性,對驅動文件的調用方式更加靈活,項目的可擴展性得到了提高。
在WinCE中,主要通過RegisterDevice和Deregister-Device兩個函數(shù)實現(xiàn)WinCE內核對GPIO端口驅動的動態(tài)加載和卸載,其中RegisterDevice函數(shù)負責加載設備驅動,DeregisterDevice函數(shù)負責卸載設備驅動[5]。在程序設計時,端口驅動程序的開發(fā)仍然采用流驅動的設計方式,但編譯環(huán)境使用Visual Studio。生成驅動動態(tài)鏈接庫文件后將其部署在開發(fā)板適當?shù)穆窂较?,然后在WinCE程序中動態(tài)加載(卸載)該dll文件。筆者使用飛凌S3C2440開發(fā)板,創(chuàng)建了一個WinCE對話框程序,通過點擊相關按鈕可以實現(xiàn)加載(卸載)GPIO端口驅動、點亮(熄滅)LED燈。圖1是本程序的運行界面。
圖1 動態(tài)加載驅動的程序運行界面
相關的要點代碼及解釋如下(暫不考慮程序的異常處理)。
(1)端口驅動動態(tài)鏈接庫的創(chuàng)建及生成
使用Visual Studio編譯環(huán)境創(chuàng)建一個常規(guī)的動態(tài)鏈接庫工程,并編譯生成dll文件即可。具體代碼這里不再給出。需要注意的是,在動態(tài)鏈接庫工程的.cpp文件中,驅動的編寫需要嚴格遵循流驅動定義的接口標準。在本程序中,流驅動的前綴名為GPI,生成的動態(tài)鏈接庫文件名為 GPI.dll。
(2)端口驅動動態(tài)鏈接庫的注冊及加載
調用RegisterDevice函數(shù)后,即將步驟1中生成的動態(tài)鏈接庫文件注冊到注冊表中,并加載進WinCE內核。在使用該函數(shù)時需要注意,在WinCE應用程序中需要獲取該驅動動態(tài)鏈接庫文件的絕對路徑。因此,開發(fā)者在部署WinCE應用程序和端口驅動動態(tài)鏈接庫時,需要注意兩者之間的位置關系,一般情況下,將兩者部署在同一路徑下即可。驅動程序動態(tài)鏈接庫的注冊及加載代碼如下:
(3)端口訪問
端口訪問的方式與常規(guī)方式一致,根據(jù)需要調用CreateFile、WriteFile、ReadFile等函數(shù)即可,相關代碼這里不再給出。
(4)端口驅動動態(tài)鏈接庫的卸載
當不需要該設備驅動時,調用DeregisterDevice函數(shù)即可實現(xiàn)對動態(tài)鏈接庫dll文件的卸載。如果程序再次需要該設備驅動時,按照步驟2再次加載設備驅動即可。
在WinCE內核動態(tài)加載GPIO端口驅動的設計方法中,內核調用端口驅動的方式比較靈活,GPIO端口驅動文件是通過相關WinCE API直接注冊和調用的,整個開發(fā)過程不再涉及Platform Builder編譯環(huán)境的使用。開發(fā)者只需要將注意力集中在Visual Studio創(chuàng)建端口驅動文件和WinCE應用程序的開發(fā)上,對驅動程序的修改和調試都獨立于WinCE內核,而且在實際的使用中,端口驅動程序、WinCE內核和WinCE應用程序各自獨立,便于各功能模塊的獨立開發(fā)和調試。
本文介紹了基于ARM+WinCE進行項目開發(fā)過程中動態(tài)加載GPIO端口驅動程序的設計方法,較好地避開了對Platform Builder編譯環(huán)境的使用,能夠替代常用的基于流驅動的驅動程序的設計方法。筆者在工程項目中驗證了這兩種設計方法,實際可行且快捷有效,對于其他ARM類的GPIO驅動程序開發(fā),具有一定的借鑒意義。
[1]羅家兵,滕少華,張巍,等.WinCE.NET下流接口驅動研究與實現(xiàn)[J].微計算機信息,2007,23(9):229-292.
[2]楊澤輝,徐燕玲,劉碧君,等.基于嵌入式 WinCE的GPIO驅動開發(fā)[J].太原科技大學學報,2010,31(6):446-448.
[3]周建設.Windows CE設備驅動及BSP開發(fā)指南[M].北京:中國電力出版社,2009.
[4]林濤.嵌入式操作系統(tǒng) Windows CE的研究[J].微計算機信息,2006(6-2):91-93.
[5]張歡,鈕文良.Windows CE系統(tǒng)開發(fā)基礎與實例[M].北京:中國電力出版社,2010.