《電子技術(shù)應(yīng)用》
您所在的位置:首頁 > 模擬設(shè)計(jì) > 設(shè)計(jì)應(yīng)用 > WinCE USB驅(qū)動(dòng)開發(fā)
WinCE USB驅(qū)動(dòng)開發(fā)
摘要: 隨著USB2.0設(shè)備的不斷增加,USB設(shè)備驅(qū)動(dòng)開發(fā)在嵌入式開發(fā)中變的越來越重要。WindowsCE支持USB2.0更是對(duì)這一波新技術(shù)浪潮產(chǎn)生巨大的推動(dòng)。近期我負(fù)責(zé)一個(gè)這樣的項(xiàng)目,在WinCE下開發(fā)USB接口的外圍設(shè)備驅(qū)動(dòng)。
關(guān)鍵詞: 接口IC WinCE USB 驅(qū)動(dòng)
Abstract:
Key words :

  隨著USB2.0設(shè)備的不斷增加,USB設(shè)備驅(qū)動(dòng)開發(fā)在嵌入式開發(fā)中變的越來越重要。Windows CE支持USB 2.0更是對(duì)這一波新技術(shù)浪潮產(chǎn)生巨大的推動(dòng)。近期我負(fù)責(zé)一個(gè)這樣的項(xiàng)目,在WinCE下開發(fā)USB接口的外圍設(shè)備驅(qū)動(dòng)。當(dāng)時(shí)做這個(gè)項(xiàng)目花費(fèi)了我相當(dāng)多的時(shí)間和精力,錯(cuò)走許多冤枉路使我精疲力盡。

  項(xiàng)目需求是在已調(diào)好的ARM9板子上開發(fā)USB WiFi無線網(wǎng)卡的驅(qū)動(dòng)程序,具體要求是驅(qū)動(dòng)程序平臺(tái)是WinCE,CPU類型支持ARM構(gòu)架,要能比較方便地移植到X86;驅(qū)動(dòng)接口類型是USB2.0和Wlan 802.11b。后來因?yàn)檫B接效率一直有問題,就東改西改,最后改的是一塌糊涂。幸好老板比較寬容,給了我充裕的時(shí)間和支持,這里將關(guān)于USB驅(qū)動(dòng)開發(fā)的點(diǎn)滴理解與大家分享。

  1. 什么是WinCE設(shè)備驅(qū)動(dòng)程序?

 ?。?)從驅(qū)動(dòng)加載方式來區(qū)分

  在深入探討Windows CE所支持的外圍設(shè)備驅(qū)動(dòng)程序之前,先了解在WinCE平臺(tái)上使用的兩種設(shè)備:內(nèi)建設(shè)備和可安裝設(shè)備。因此,從驅(qū)動(dòng)加載方式來看WinCE可分為本機(jī)設(shè)備驅(qū)動(dòng)(Built-In Driver)、可加載驅(qū)動(dòng)(Loadable Driver)以及混合型驅(qū)動(dòng)。

 ?、俦緳C(jī)設(shè)備驅(qū)動(dòng)

  本機(jī)設(shè)備驅(qū)動(dòng)即Native Device Drivers。WinCE設(shè)計(jì)成可直接使用內(nèi)建設(shè)備,這些設(shè)備由本機(jī)驅(qū)動(dòng)過程控制,而本機(jī)驅(qū)動(dòng)程序又與WinCE的核心組件緊密相連。這些驅(qū)動(dòng)對(duì)應(yīng)的設(shè)備通常在系統(tǒng)啟動(dòng)時(shí),在GWES的進(jìn)程空間內(nèi)被加載,因此它們不是以獨(dú)立的DLL形式存在,也因此要求每一個(gè)本機(jī)驅(qū)動(dòng)程序都必須與稱為設(shè)備驅(qū)動(dòng)程序接口(DDI)的特定接口一致。

  本機(jī)設(shè)備是指整合進(jìn)平臺(tái)的設(shè)備,其中包括顯示、觸摸面板、音頻、串行埠、LED、電池和PC卡插座等。如果沒有這些本機(jī)設(shè)備整個(gè)系統(tǒng)就不能和用戶信息交流,例如觸摸面板和顯示等。本機(jī)驅(qū)動(dòng)程序一般設(shè)計(jì)為動(dòng)態(tài)鏈接庫,但有兩個(gè)例外:電池和LED驅(qū)動(dòng)程序由于小而設(shè)計(jì)為靜態(tài)庫(當(dāng)建立CE圖像時(shí)與GWES模塊鏈接)。這些設(shè)備相應(yīng)的驅(qū)動(dòng)程序是在WinCE平臺(tái)開發(fā)過程中由OEM開發(fā)的,它們儲(chǔ)存在ROM或閃存內(nèi)。通常只有OEM才會(huì)對(duì)本機(jī)設(shè)備驅(qū)動(dòng)程序進(jìn)行修改,其它自由設(shè)備生產(chǎn)商只提供附加的硬件設(shè)備,對(duì)本機(jī)設(shè)備驅(qū)動(dòng)程序不會(huì)有過多涉及。

 ?、诳杉虞d設(shè)備驅(qū)動(dòng)

  可加載設(shè)備是指可與平臺(tái)連接和分離的第三方接口設(shè)備,可由用戶隨時(shí)安裝和卸載。這種外圍設(shè)備的驅(qū)動(dòng)也被稱為流驅(qū)動(dòng),這些驅(qū)動(dòng)可以在系統(tǒng)啟動(dòng)時(shí)或者和啟動(dòng)后的任何時(shí)候由設(shè)備管理器動(dòng)態(tài)加載。通常這類驅(qū)動(dòng)是以DLL動(dòng)態(tài)鏈接庫的形式存在,系統(tǒng)加載后這些驅(qū)動(dòng)程序也只是以用戶態(tài)的角色運(yùn)行??杉虞d驅(qū)動(dòng)程序是通過文件操作API來從設(shè)備管理器和應(yīng)用程序獲得命令。在WinCE典型的可加載驅(qū)動(dòng)有:PCMCIA driver(PCMCIA.dll)、Serial driver(SERIAL.dll)、ATAFLASH driver(ATA.dll)、Ethernet driver(NE2000.dll,SMSC100FD.dll)。

  與本機(jī)驅(qū)動(dòng)程序不同的是,所有可加載流驅(qū)動(dòng)程序都共享一個(gè)公用接口。該接口由每個(gè)驅(qū)動(dòng)程序內(nèi)的10個(gè)功能或記錄點(diǎn)組成,這些功能與應(yīng)用程序所用的文件API中的功能匹配。因此,控制可加載設(shè)備的流接口驅(qū)動(dòng)程序一般由應(yīng)用程序存取,流接口驅(qū)動(dòng)程序由一個(gè)特殊文件來將設(shè)備功能展現(xiàn)給應(yīng)用程序的,該文件可被打開、讀取、寫入和關(guān)閉。例如,用戶將一個(gè)GPS設(shè)備與平臺(tái)相連后,就可啟動(dòng)有GPS功能的應(yīng)用程序來存取并使用該設(shè)備。WinCE是使用已有的API來讓應(yīng)用程序存取這些驅(qū)動(dòng)程序,而不是建立新的API。

 ?。?)從驅(qū)動(dòng)程序?qū)哟紊戏诸?/p>

  一般可以分為獨(dú)立驅(qū)動(dòng)和層次型驅(qū)動(dòng)兩類。獨(dú)立驅(qū)動(dòng)程序是指將驅(qū)動(dòng)程序編寫成同時(shí)包含Model Device Driver(MDD)和Platform Dependent Driver(PDD)層的獨(dú)立驅(qū)動(dòng)。使用獨(dú)立驅(qū)動(dòng)的好處在于可以省去MDD和PDD層驅(qū)動(dòng)之間的信息傳遞,這一點(diǎn)在實(shí)時(shí)處理中非常重要。獨(dú)立驅(qū)動(dòng)的代碼包括中斷服務(wù)例程和平臺(tái)相關(guān)處理函數(shù)。另外,如果設(shè)備的操作和MDD驅(qū)動(dòng)層的接口描述相吻合,用獨(dú)立驅(qū)動(dòng)程序可以提高處理性能。

  層次型驅(qū)動(dòng)是指分為兩層,較上層的MDD和比較下層的PDD。MDD實(shí)現(xiàn)的是和平臺(tái)無關(guān)的功能,它描述了一個(gè)通用的驅(qū)動(dòng)程序框架;而PDD是和硬件以及平臺(tái)相關(guān)的代碼組成。MDD調(diào)用PDD中特定的接口來獲取硬件相關(guān)的信息。當(dāng)使用層次型驅(qū)動(dòng)的時(shí)候,一般只需要基于相近的樣列驅(qū)動(dòng)程序,針對(duì)特定的硬件只修改PDD程序,MDD建立的框架可繼續(xù)使用。但由于層次間接口的層層調(diào)用以及消息的傳遞,使得處理速度相對(duì)于獨(dú)立驅(qū)動(dòng)程序要慢。因此,在嵌入式實(shí)時(shí)要求苛刻的環(huán)境下,層次型驅(qū)動(dòng)顯得不是很適合。

  簡(jiǎn)單的說,獨(dú)立驅(qū)動(dòng)是把PDD與MDD寫在一起,沒有做嚴(yán)格的區(qū)分,通常這種驅(qū)動(dòng)比較簡(jiǎn)單,比如ATADISK。至于本機(jī)驅(qū)動(dòng)和加載式流驅(qū)動(dòng)是從驅(qū)動(dòng)與系統(tǒng)其它模塊(調(diào)用者)的接口形式上做的分類。所以,一個(gè)加載式驅(qū)動(dòng)程序可以是獨(dú)立的流式驅(qū)動(dòng),例如ATADISK;也可以是分層的流式驅(qū)動(dòng),例如OHCI。也就是說,獨(dú)立和分層是驅(qū)動(dòng)實(shí)現(xiàn)方式上的分類,而本機(jī)和加載流式則是驅(qū)動(dòng)模型上的分類。所謂本機(jī)驅(qū)動(dòng)就是操作系統(tǒng)有保留專門的接口,而加載流式驅(qū)動(dòng)是指編寫DLL文件導(dǎo)出各種流式接口函數(shù)的接口。

  2. USB加載式流接口驅(qū)動(dòng)要點(diǎn)分析

  為了支持不同類型的外圍設(shè)備,WinCE平臺(tái)提供了具有定制接口的流接口驅(qū)動(dòng)程序模型。因?yàn)榇蟛糠諹SB外圍設(shè)備由于功能性更適合流接口驅(qū)動(dòng)的結(jié)構(gòu),所以一般都采用加載式流接口驅(qū)動(dòng)程序模型來開發(fā)USB設(shè)備驅(qū)動(dòng)程序。

  (1)USB系統(tǒng)結(jié)構(gòu)分析

  WinCE下USB系統(tǒng)軟件由兩層組成:較高USB設(shè)備驅(qū)動(dòng)程序?qū)雍洼^低的USB函數(shù)層。較低的USB函數(shù)層本身又由兩部分組成:較高的通用串行總線驅(qū)動(dòng)程序(USBD)模塊和較低的主控制器驅(qū)動(dòng)程序(HCD)模塊。通過HCD模塊功能和USBD模塊實(shí)現(xiàn)高層的USBD接口函數(shù),USB設(shè)備驅(qū)動(dòng)程序就能與外圍設(shè)備進(jìn)行通訊。

  在數(shù)據(jù)傳輸?shù)倪^程中,操作流程通常按下列的次序進(jìn)行:①USB設(shè)備驅(qū)動(dòng)程序進(jìn)行數(shù)據(jù)傳輸?shù)某跏蓟?,即通過USBD接口函數(shù)給USBD模塊發(fā)送數(shù)據(jù)傳輸?shù)恼?qǐng)求。②USBD模塊將該請(qǐng)求分成一些單獨(dú)的事務(wù)。③HCD模塊排出事務(wù)次序。④主控制器硬件執(zhí)行事務(wù)。這里需要提醒的是,所有的事務(wù)都是從主機(jī)發(fā)出的,外圍設(shè)備完全是被動(dòng)接受型的。

 ?。?)USB設(shè)備驅(qū)動(dòng)程序入口點(diǎn)函數(shù)

  從結(jié)構(gòu)分析我們可知,所有的USB設(shè)備驅(qū)動(dòng)程序必須在它們的DLL庫設(shè)置一定的入口點(diǎn)與USBD模塊進(jìn)行適當(dāng)?shù)慕换?。設(shè)置入口點(diǎn)函數(shù)有兩個(gè)作用:一是使得 USBD 模塊能與外部設(shè)備交互;二是使得驅(qū)動(dòng)程序能創(chuàng)建和管理任何可能需要的注冊(cè)鍵。

  下面簡(jiǎn)要介紹相關(guān)函數(shù)的作用:USBDeviceAttach是當(dāng) USB 設(shè)備連接到主計(jì)算機(jī)時(shí)運(yùn)行,USBD模塊會(huì)調(diào)用這個(gè)函數(shù)初始化USB設(shè)備,取得USB設(shè)備信息和配置USB設(shè)備,并且申請(qǐng)必需的資源。USBInstallDrive是在第一次加載USB設(shè)備驅(qū)動(dòng)程序時(shí)首先被調(diào)用,它使得驅(qū)動(dòng)程序能創(chuàng)建需要的注冊(cè)鍵,用于將一個(gè)驅(qū)動(dòng)程序所需的注冊(cè)表信息寫入到HKEY_LOCAL_MACHINE\Drivers\USB\ClientDrivers目錄下,例如設(shè)備名稱等。需要注意的是,USB設(shè)備驅(qū)動(dòng)程序不使用標(biāo)準(zhǔn)的注冊(cè)表函數(shù),而是使用RegisterClientDriverID()、RegisterClientSettings()函數(shù)來注冊(cè)相應(yīng)的設(shè)備信息。

  USBUninstallDriver是在用戶刪除USB設(shè)備驅(qū)動(dòng)程序時(shí)調(diào)用,負(fù)責(zé)刪除注冊(cè)鍵并釋放其它相關(guān)資源。它通過調(diào)用UnRegisterClientSettings()和UnRegisterClientDriverID()函數(shù)來刪除由驅(qū)動(dòng)程序的USBInstallDriver()函數(shù)創(chuàng)建的所有注冊(cè)鍵。因此,我們?cè)隍?qū)動(dòng)程序中就需要嚴(yán)格按照這三個(gè)函數(shù)的原型來實(shí)現(xiàn),否則就不能為設(shè)備管理器所識(shí)別。

  3.USB設(shè)備流接口驅(qū)動(dòng)的實(shí)現(xiàn)步驟

  從WinCE USB設(shè)備驅(qū)動(dòng)模型及結(jié)構(gòu)分析中,我們可以清晰的看到主機(jī)和外設(shè)之間的實(shí)現(xiàn)方式。在主機(jī)端,通過USBD模塊和HCD模塊使用默認(rèn)的PIPE訪問一個(gè)通用的邏輯設(shè)備,實(shí)際上就是說USBD和HCD是一組訪問所有USB設(shè)備的邏輯接口,它們負(fù)責(zé)管理所有USB設(shè)備的連接、加載、移除、數(shù)據(jù)傳輸和通用配置。其中HCD是主機(jī)控制驅(qū)動(dòng),是為USBD提供底層的功能訪問服務(wù),USBD是USB總線驅(qū)動(dòng),位于HCD的上層,利用HCD的服務(wù)提供較高層次的功能。因此,實(shí)現(xiàn)USB加載流驅(qū)動(dòng)程序大致需要完成以下步驟:

 ?。?)選擇代表設(shè)備的文件名前綴。前綴非常重要,設(shè)備管理器在注冊(cè)表中通過前綴來識(shí)別設(shè)備。同時(shí),在流接口命名時(shí)也將這個(gè)前綴作為入口點(diǎn)函數(shù)的前綴,如果設(shè)備前綴為XXX,那么流接口對(duì)應(yīng)為XXX_Close,XXX_Init等。

  (2)設(shè)置驅(qū)動(dòng)的各個(gè)入口點(diǎn)函數(shù)。所謂入口點(diǎn)是指提供給設(shè)備管理器的標(biāo)準(zhǔn)文件I/O接口。在生成一個(gè)DLL后,就用設(shè)備文件名前綴替換名字中的XXX。因此,每個(gè)加載式流接口驅(qū)動(dòng)程序必須實(shí)現(xiàn)XXX_Init()、XXX_IOControl()以及XXX_PowerUp()等一組標(biāo)準(zhǔn)的函數(shù),用來完成標(biāo)準(zhǔn)的文件I/O函數(shù)和電源管理等。

  (3)建立.DEF文件。當(dāng)設(shè)備管理器初始化USB設(shè)備編譯出來的流接口函數(shù)后,還必須建立一個(gè).def文件。DEF文件定義了DLL要導(dǎo)出的接口集,而且加載式流驅(qū)動(dòng)大多是以DLL形式存在的,所以應(yīng)將DLL和DEF的文件名統(tǒng)一起來。DEF文件告訴鏈接程序需要輸出什么樣的函數(shù),最后將驅(qū)動(dòng)程序編譯到內(nèi)核中去,這樣這個(gè)USB設(shè)備流接口驅(qū)動(dòng)程序就可以被應(yīng)用程序調(diào)用。

 ?。?)在注冊(cè)表中為驅(qū)動(dòng)程序建立表項(xiàng)。在注冊(cè)表中建立驅(qū)動(dòng)程序入口點(diǎn),這樣設(shè)備管理器才能識(shí)別和管理這個(gè)驅(qū)動(dòng)。此外,注冊(cè)表中還能存儲(chǔ)額外的信息,這些信息可以在驅(qū)動(dòng)運(yùn)行之后被使用到。

  在這次USB驅(qū)動(dòng)開發(fā)過程中,錯(cuò)走許多冤枉路使我叫苦連天。我感受最深的是由于WinCE提供了通用串行總線驅(qū)動(dòng)程序(USBD)模塊、USBD接口函數(shù)全集、樣本主機(jī)控制器驅(qū)動(dòng)程序(HCD)模塊。所以,我們只需要根據(jù)USB設(shè)備硬件特性,利用USBD提供的不同函數(shù),實(shí)現(xiàn)流接口函數(shù)與外圍設(shè)備的交互。在沒有特別的情況下,我最大的收獲經(jīng)驗(yàn)是把這些公用的源程序照搬過來,能極大的縮短開發(fā)周期,從而能更快速地進(jìn)行嵌入式開發(fā)。

此內(nèi)容為AET網(wǎng)站原創(chuàng),未經(jīng)授權(quán)禁止轉(zhuǎn)載。