VxWorks實(shí)時(shí)操作系統(tǒng)的USB驅(qū)動(dòng)程序原理與分析
在棧的最底部是USB主控制器(USB HC, 即USB Host Controller), 這是主系統(tǒng)中控制每一個(gè)USB設(shè)備的硬件部分。
目前,市場(chǎng)上主要有兩大類(lèi)USB主控制器,一種是支持由ime1公司最先提出的通用主控制器接口(Universal Host Controller Interface,簡(jiǎn)稱UHCI),另一種是支持由微軟、康柏和國(guó)家半導(dǎo)體公司聯(lián)合設(shè)計(jì)提出的開(kāi)放主控制器接口(Open Host Controller Interface,簡(jiǎn)稱OHCI)。硬件廠商一般根據(jù)這兩個(gè)規(guī)范設(shè)計(jì)他們的USB主控制器。
對(duì)于每一類(lèi)型的主控制器都有一個(gè)與硬件獨(dú)立的USB主控制器驅(qū)動(dòng)(Host Controller Driver,簡(jiǎn)稱HCD)。WindRiver提供了兩個(gè)驅(qū)動(dòng):usbHcdUhciLib (UHCI 主控制器庫(kù))和usbHedOheiLib (OHCI主控制器庫(kù))。USB主驅(qū)動(dòng)(USB host driver,簡(jiǎn)稱USBD)和HCD之間的接口允許一個(gè)或超過(guò)一個(gè)的底層主控制器。而且,WindRiver的USBD能夠同時(shí)連接多個(gè)USB HCD。這樣的設(shè)計(jì)特點(diǎn)可以使開(kāi)發(fā)者建立復(fù)雜的USB系統(tǒng)。
USBD是在HCD之上的與硬件獨(dú)立的模塊。USBD管理每一個(gè)與主機(jī)相連的USB設(shè)備,向更高層次提供了可與USB設(shè)備通信的路徑。它還負(fù)責(zé)自動(dòng)處理USB電力管理以及USB帶寬管理。而且,USBD還管理 USB hub,Hub功能是一個(gè)驅(qū)動(dòng)能否對(duì)USB正確操作的評(píng)價(jià)之一。因此WindRiver的USBD設(shè)計(jì)者要使USBD透明地處理hub的功能。這意味著, USBD 還能處理USB hub和設(shè)備的動(dòng)態(tài)插拔。
USB Client模塊在USB主驅(qū)動(dòng)棧的頂端。USB類(lèi)驅(qū)動(dòng)(USB Class Driver)是Client模塊的典型例子。USB類(lèi)驅(qū)動(dòng)負(fù)責(zé)管理連接到USB上的不同類(lèi)型的設(shè)備;它們依靠USBD來(lái)提供與每個(gè)設(shè)備的通信路徑。 USB client模塊的其他例子就是那些利用USBD與USB設(shè)備通信的應(yīng)用程序。
4、 USBD驅(qū)動(dòng)詳解
這一部分將要描述USBD(USB Host Driver)的典型應(yīng)用。例如初始化,client注冊(cè),動(dòng)態(tài)連接注冊(cè),設(shè)備配置,數(shù)據(jù)傳輸,同時(shí)還探討了USBD內(nèi)部設(shè)計(jì)的關(guān)鍵特性。這部分是VxWorks下USB驅(qū)動(dòng)的核心。
4.1 初始化USBD:分為兩步
(1)必須至少調(diào)用一次函數(shù)usbdInitialize()。在一個(gè)給定的系統(tǒng)中,usbdlnifialize()初始化內(nèi)部USBD數(shù)據(jù)結(jié)構(gòu),并依次調(diào)用其它USB驅(qū)動(dòng)棧模塊的入口。usbdinitialize()可以在啟動(dòng)時(shí)調(diào)用一次,也可以對(duì)每一個(gè)設(shè)備各調(diào)用一次。USBD 自己記錄了調(diào)用usbdInitialize()(‘+’)和usbdShutDown()(‘-’)的次數(shù)。只有大于等于1時(shí)才是真正初始化了,而等于 0是關(guān)閉了。
(2)用USBD 的lisbdHedAttaeh()函數(shù)來(lái)把至少一個(gè)HCD連接到USBD上。這一過(guò)程既可以在VxWorks啟動(dòng)時(shí),也可以在運(yùn)行時(shí)把HCD 連接到USBD 上去。后一種機(jī)制可以支持“熱插拔”,而不用象前一種那樣需要重新啟動(dòng)。
4.2 HCD的連接(attaching)與斷開(kāi)(detaching)
當(dāng)HCD連接到USBD 時(shí),調(diào)用者為usbdHedattaeh函數(shù)傳遞HCD執(zhí)行入口(表HCD_EXEC_FUNC)和HCD連接參數(shù)(HCD attach parameter)。USBD用HCD FNC ATYACH 服務(wù)請(qǐng)求依次激活HCD的執(zhí)行入口,傳遞同樣的HCD attach參數(shù)。
需要強(qiáng)調(diào)雖然可以改變用HCD定義的參數(shù),但是最好不應(yīng)該有所改變。對(duì)于WindRiver提供的UHCI和OHCI的HCD,HCI attach參數(shù)是一個(gè)指向結(jié)構(gòu)PCI_CFG_HEADER (定義在pciConstants.h) 的指針。
該結(jié)構(gòu)用UHCI和OHCI主控制器的PCI配置頭來(lái)初始化,而HCD用這個(gè)結(jié)構(gòu)中的信息來(lái)定位,管理特定的主控制器。典型的,調(diào)用者用 usbPeiClassFind ()和usbPciConfigHeaderGet()來(lái)得到想要的主控制器的PCI配置頭- 這兩個(gè)函數(shù)定義在usbPciLib 中(stubUsbarchPciLib.h中)。如果有UHCI或OHCI要連接到USBD,就要調(diào)用這些函數(shù)來(lái)獲得每一個(gè)主控制器的 PCI_CFG_HEADER。然后利用usbdHedAttaeh來(lái)激活已鑒別出的每一個(gè)主控制器。
注意:底層BSP可能不支持USB的HCD斷開(kāi),因?yàn)楫?dāng)中斷向量表重新使能時(shí),如果還應(yīng)用的是過(guò)期的向量表,會(huì)導(dǎo)致錯(cuò)誤。
4.3 啟動(dòng)順序
必須在所有USBD函數(shù)前執(zhí)行函數(shù)usbdInitialize()。存在以下兩種調(diào)用方式:
(1)傳統(tǒng)的“啟動(dòng)”初始化。執(zhí)行順序與其意義如下:
a.usbdInitialize();
b.usbdPciClassFind():定位一個(gè)USB主控制器;
c.usbdPeiConfigHeaderGet():讀USB主控制器配置頭;
d.usbdHedAttaeh():連接HCD,將其作為特定的主控制器:
e.調(diào)用USB class driver初始化入口點(diǎn);
f.USB class driver調(diào)用usbdlnitialize()。
(2)“熱插拔”調(diào)用。執(zhí)行順序與其意義如下:
Boot Code里調(diào)用:
a.USB class driver初始化入口點(diǎn);
b.USB class driver調(diào)用usbdlnitialize();
Hot-Swap code調(diào)用:
c.Hot-Swap 鑒別USB主控制器的連接或斷開(kāi);
d.Usbdlnitialize();
e.UsbdPciConfigHeaderGet():讀USB主控制器配置頭;
f.UsbdHedAttaeh():連接HCD,將其作為特定的主控制器。
因?yàn)闊岵灏慰梢栽谌魏螘r(shí)刻發(fā)生,所以USBD和其Client都必須被寫(xiě)成可以動(dòng)態(tài)識(shí)別USB設(shè)備被插入還是被拔出。當(dāng)主控制器連接到系統(tǒng)時(shí),USBD 自動(dòng)地鑒別與其相連的設(shè)備,并通知相關(guān)的client;同樣,拔出設(shè)備時(shí),也要通知相關(guān)設(shè)備。重要的是,USBD 的client,比如USB class driver,在client初始化時(shí),從不設(shè)想特定的設(shè)備已經(jīng)出現(xiàn);而在其他時(shí)候,這些驅(qū)動(dòng)隨時(shí)檢查設(shè)備是否已經(jīng)連接到系統(tǒng)上。
4.4 總線任務(wù)
對(duì)每一個(gè)連接到USBD 的主控制器,例如插入或拔出設(shè)備,USBD都會(huì)產(chǎn)生一個(gè)總線任務(wù),來(lái)監(jiān)控總線事件。一般情況下,這些任務(wù)是休眠的(不消耗CPU),只有當(dāng)USB hub報(bào)告它的一個(gè)端口有變化時(shí),它們才被喚醒。每一個(gè)USBD總線任務(wù)有VxWorks任務(wù)名:UsbdBus。
雖然 HCD委托USBD來(lái)管理,但有可能HCD 親自監(jiān)視主控制器事件。例如WindRiver提供了UHCI和OHCI的HCD來(lái)創(chuàng)造這樣的任務(wù)。對(duì)于WindRiver的UHCI模塊 (usbHcdUheiLib),后臺(tái)任務(wù)只是被周期地喚醒,目的是為了檢查超時(shí)IRP(用一個(gè)中斷來(lái)通知OHCI根hub發(fā)生改變)。
用以在USBD和USB之問(wèn)進(jìn)行通信的client模塊,除了調(diào)用usbdlnitialize()外,必須調(diào)用usbClientRegister ()使其在USBD注冊(cè)。當(dāng)一個(gè)client注冊(cè)到USBD時(shí),USBD把每一個(gè)以后將要用到的client的數(shù)據(jù)結(jié)構(gòu)定位,并跟蹤那個(gè)client的請(qǐng)求。
對(duì)于每一個(gè)client,在client注冊(cè)過(guò)程中,USBD還創(chuàng)建了一個(gè)callback任務(wù)。在成功注冊(cè) client后,USBD返回一個(gè)句柄USBD_CLIENT_HANDLE。以下對(duì)USBD的調(diào)用,將會(huì)用到這個(gè)句柄。當(dāng)所有句柄都不需要時(shí),可以調(diào)用 usbdClientUnregister()來(lái)釋放每一個(gè)client的數(shù)據(jù)結(jié)構(gòu)和callback任務(wù)。注意:此時(shí)所有client要求的任務(wù)都會(huì)被取消。
例如:注冊(cè)一個(gè)叫USBD_TEST的client,再注銷(xiāo)。
注冊(cè):usbdClientRegister("USBD_TEST',usbdClientHandle);
注銷(xiāo):usbdClientUnregister(usbdClientHandle);
評(píng)論