SEP6200平臺上Linux內(nèi)核的USB OTG驅(qū)動設(shè)計
摘要:為了滿足兩個USB設(shè)備之間直接進(jìn)行數(shù)據(jù)通信的需求,通過對現(xiàn)有USB OTG協(xié)議和相關(guān)應(yīng)用的研究,在國產(chǎn)SoC芯片SEP6200平臺上設(shè)計并實現(xiàn)了基于Linux內(nèi)核的USB OTG模塊驅(qū)動。首先介紹了OTG標(biāo)準(zhǔn)中的對話請求協(xié)議(SRP)和主機交換協(xié)議(HNP),然后制定并設(shè)計了基于USB控制芯片USB3343的硬件模塊方案,最后根據(jù)Linux內(nèi)核中已有的USB驅(qū)動架構(gòu)完成了USB OTG設(shè)備驅(qū)動的設(shè)計,并最終實現(xiàn)了SEP6200嵌入式平臺USB Host和Device角色的自由轉(zhuǎn)換功能。
關(guān)鍵詞:USB OTG;SEP6200;USB3343;Linux;設(shè)備驅(qū)動
引言
傳統(tǒng)的USB協(xié)議(USB 2.0)規(guī)定了兩個設(shè)備之間進(jìn)行數(shù)據(jù)通信時,必須有一個Host設(shè)備作為固定的主機,通過這個主機來控制設(shè)備問的數(shù)據(jù)傳輸。在日常應(yīng)用中,基本都是將USB設(shè)備連接到PC,并在PC的控制下進(jìn)行數(shù)據(jù)交換。這種交換方式,一旦離開了PC,各設(shè)備間就無法利用USB口進(jìn)行直接操作。隨著嵌入式設(shè)備的快速發(fā)展,兩個USB設(shè)備之間直接進(jìn)行數(shù)據(jù)交換的需求變得越來越迫切。USB On—The—Go(USB OTG)技術(shù)便應(yīng)運而生。OTG技術(shù)實現(xiàn)了在沒有Host的情況下,設(shè)備之間點對點的數(shù)據(jù)控制與傳輸。本文通過研究USB 2.0協(xié)議和USB OTG補充規(guī)范,結(jié)合具體應(yīng)用需求,建立了基于國產(chǎn)SoC的嵌入式USB OTG模塊硬件平臺,并在Linux操作系統(tǒng)平臺上設(shè)計并實現(xiàn)了基于國產(chǎn)內(nèi)核主控芯片SEP6200的USB OTG設(shè)備驅(qū)動。
1 USB OTG協(xié)議與實現(xiàn)方式概述
USB OTG標(biāo)準(zhǔn)在完全兼容USB2.0標(biāo)準(zhǔn)的基礎(chǔ)上,增添了電源管理(節(jié)省功耗)功能,并可提供一定的主機檢測能力,支持對話請求協(xié)議(SRP)和主機交換協(xié)議(HNP)。在OTG中,初始主機設(shè)備稱為A—device,外設(shè)稱為B—device。
對話請求協(xié)議(Session Request Protocol,SRP):為了節(jié)省功耗,A—device在總線不被使用時允許VBUS總線關(guān)斷,此時B設(shè)備就會檢測到該狀態(tài)并進(jìn)入功低耗模式。當(dāng)外部B—device需要向A—device發(fā)起對話時,它會通過SRP向A—device發(fā)出申請,請求A—device向VBUS供電并啟動一次對話。對話將隨VBUS的開啟而開始,隨VBUS的關(guān)閉而結(jié)束。
主機交換協(xié)議(Host Negotiation Protocol,HNP):該協(xié)議允許兩個直接相連的OTG設(shè)備進(jìn)行主機Host功能的切換,它能夠根據(jù)電纜的連接方式來決定初始化角色,從而允許設(shè)備在通信時能夠改變控制信號。當(dāng)用戶模式或應(yīng)用程序下B—device有輸入時,HNP將作初始響應(yīng)。
OTG兩用設(shè)備使用新型mini—AB插座,從而使mini—A插頭、mini—B插頭和mini—AB插座增添了第5個引腳(ID),以用于識別不同的電纜端點。mini—A插頭中的ID引腳接地,mini—B插頭中的ID引腳浮空。當(dāng)OTG設(shè)備檢測到接地的ID引腳時,表示默認(rèn)的是A—device(主機);而檢測到ID引腳浮著的設(shè)備,則認(rèn)為是B—device(外設(shè))。系統(tǒng)一旦連接后,OTG的角色還可以更換。A-Device作為默認(rèn)主機并提供VBUS電源,并在檢測到有設(shè)備連接時復(fù)位總線、枚舉并配置B-device。
2 USB OTG模塊硬件方案
為了后期USB OTG驅(qū)動設(shè)計需要,首先對OTG硬件模塊作一些簡單的介紹。本文設(shè)計的USB OTG模塊在硬件上由主控芯片SEP6200以及USB控制芯片USB3343構(gòu)成。SEP6200是東南大學(xué)國家ASIC工程中心自主研發(fā)的一款定位于手持視頻播放設(shè)備、衛(wèi)星導(dǎo)航產(chǎn)品的高性能應(yīng)用處理器,它內(nèi)部集成了ULPI總線接口,支持SRP和HNP協(xié)議,擁有USB2.0 OTG功能。USB控制芯片USB3343是SMSC公司生產(chǎn)的一款高速USB收發(fā)器,支持主機、設(shè)備和OTG三種功能,非常適合用于要求低功耗工作和待機工作的便攜式設(shè)備。SEP6200內(nèi)部的OTG Controller和USB3343之間通過otg_transceiver接口來完成交互。
在本模塊中,SEP6200處理器通過自帶的ULPI接口與USB3343進(jìn)行連接和通信,而USB3343收發(fā)控制器則是通過外接USB MiniAB插座與外部USB設(shè)備進(jìn)行連接。其硬件連接結(jié)構(gòu)圖如圖1所示。
此硬件平臺以SEP6200處理器為核心,處理器除了ULPI接口的12根總線與USB3343相連接外,另外分別通過兩根GPIO接口線來控制實現(xiàn)OTG的低功耗功能。其中一根(GPIO_RESETB)用來對USB3343進(jìn)行軟件復(fù)位,另一根(GPIO_EN)用來控制對VBUS總線供電。當(dāng)核心板作為device與PC進(jìn)行連接時,軟件檢測到ID信號狀態(tài)為浮空后將GPIO_EN引腳置0,切斷電源模塊,使5 V供電電壓與VBUS總線隔絕。相反,當(dāng)核心板作為Host與U盤等外設(shè)進(jìn)行通信時,軟件檢測到ID信號狀態(tài)為接地,然后將GPIO_EN引腳置位,通過電源使能模塊將5 V電壓加載到VBUS總線上,實現(xiàn)核心板作為主機對外設(shè)進(jìn)行供電,從而實現(xiàn)對外部設(shè)備的檢測和枚舉等后續(xù)操作。
3 USB OTG驅(qū)動架構(gòu)的設(shè)計與實現(xiàn)
3.1 OTG設(shè)備驅(qū)動程序架構(gòu)
本設(shè)計的軟件驅(qū)動開發(fā)基于Linux操作系統(tǒng)平臺。作為開源操作系統(tǒng),Linux有良好的兼容性和強大的USB技術(shù)支持。OTG設(shè)備驅(qū)動程序主要由USB目標(biāo)設(shè)備軟件包和USB主設(shè)備軟件包構(gòu)成。OTG驅(qū)動通過連接器中“ID”信號的不同連接,根據(jù)SRP和HNP協(xié)議的內(nèi)部機制,確定使用USB主設(shè)備軟件包還是USB目標(biāo)設(shè)備軟件包。圖2是OTG沒備驅(qū)動程序架構(gòu)框圖。
當(dāng)OTG設(shè)備以從機方式工作時,USB目標(biāo)設(shè)備功能模塊工作。目標(biāo)設(shè)備控制驅(qū)動完成USB目標(biāo)設(shè)備軟件包與OTG硬件問的數(shù)據(jù)交換。USB協(xié)議層完成USB協(xié)議規(guī)范,USB目標(biāo)設(shè)備類驅(qū)動的功能取決于OTG設(shè)備的功能。
當(dāng)OTG設(shè)備以主機方式工作時,USB主設(shè)備功能模塊丁作。主控制驅(qū)動完成USB主設(shè)備軟件包與OTG硬件間的數(shù)據(jù)交換,USB總線驅(qū)動保存設(shè)備的信息。OTG提供通用的主設(shè)備類驅(qū)動程序,驅(qū)動支持的這些目標(biāo)主設(shè)備包含在目標(biāo)設(shè)備的列表里。
基于以上分析,驅(qū)動軟件設(shè)計的總體任務(wù)主要分為:USB從機設(shè)備驅(qū)動設(shè)計、USB主機控制器驅(qū)動設(shè)計以及USB OTG驅(qū)動設(shè)計。其中從機設(shè)備驅(qū)動模塊和主機控制器驅(qū)動模塊互相獨立,OTG驅(qū)動依賴于從機設(shè)備驅(qū)動和主機控制器驅(qū)動。
3.2 USB從機設(shè)備驅(qū)動設(shè)計
基于SEP6200的核心板系統(tǒng)在作為USB從機設(shè)備時,要應(yīng)答PC主機的標(biāo)準(zhǔn)請求,處理USB總線事務(wù)和用戶功能。因此從機驅(qū)動包括3層:UD C驅(qū)動程序、Gadget驅(qū)動程序和Gadget API。UDC驅(qū)動程序負(fù)責(zé)直接訪問硬件,控制USB設(shè)備與主機間的底層通信,向上層提供與硬件相關(guān)操作的回調(diào)函數(shù)。Gadget驅(qū)動程序具體控制USB設(shè)備功能的實現(xiàn)。Gadget API則是提供給UDC驅(qū)動程序回調(diào)函數(shù)的簡單封裝。
Linux內(nèi)核首先通過musb_init_controller()函數(shù)初始化USB設(shè)備控制器,OTG驅(qū)動提供的狀態(tài)機根據(jù)硬件信息將USB OTG工作模式配置成從機模式。接著內(nèi)核會初始化Gadget驅(qū)動,完成Gadget驅(qū)動和UDC驅(qū)動的綁定,并配置控制器使其開始工作。然后調(diào)用底層GadgetAPI函數(shù)來激活端點進(jìn)行數(shù)據(jù)傳輸。USBOTG定義了三個端點;程序中定義端點0在控制傳輸中應(yīng)答設(shè)備枚舉,端點1的功能為向PC機發(fā)送數(shù)據(jù),端點2的功能為接收PC機發(fā)送的數(shù)據(jù)。最后,通過中斷響應(yīng)和中斷處理函數(shù)來完成核心板作為設(shè)備與PC機的通信操作。內(nèi)核使用struct musb結(jié)構(gòu)體描述UCD驅(qū)動實例的各種信息。
3.3 USB主機控制器驅(qū)動設(shè)計
核心板系統(tǒng)的主機驅(qū)動分為幾個部分:Host controller drive、USB core和Class driver。在本設(shè)計中,Hostcontroiler drive與所應(yīng)用的USB主控芯片USB3343相關(guān),而USB Core和Class driver在Linux內(nèi)核中提供了相應(yīng)的支持。
Linux內(nèi)核使用usb_hcd結(jié)構(gòu)描述USB主機控制器驅(qū)動Host controller drive。usb_hcd結(jié)構(gòu)描述了USB主機控制器的硬件信息、狀態(tài)和操作函數(shù),其定義如下:
軟件根據(jù)HNP協(xié)議執(zhí)行完所有的HNP狀態(tài)后,OTG驅(qū)動進(jìn)入USB Host模式。在對底層硬件初始化后,注冊Hub和USB存儲類設(shè)備,隨后加載hub_probe對根Hub進(jìn)行初始化和枚舉。當(dāng)外部USB設(shè)備插入后,系統(tǒng)將對USB類設(shè)備進(jìn)行枚舉,查找對應(yīng)的USB存儲類設(shè)備驅(qū)動,并加載stora ge_probe,然后讀取插入的USB存儲類設(shè)備的文件系統(tǒng)結(jié)構(gòu),將該存儲設(shè)備注冊為一個scsidisk。使用mount命令將該設(shè)備掛載到相應(yīng)目錄后,即可完成對USB存儲設(shè)備的讀寫操作。
3.4 USB OTG驅(qū)動設(shè)計
OTG驅(qū)動維護(hù)著一個OTG狀態(tài)機,從而支持HNP和SRP協(xié)議,轉(zhuǎn)換主機端和設(shè)備端功能,上文中硬件狀態(tài)的檢測和驅(qū)動模式轉(zhuǎn)化即由此實現(xiàn)。在模塊初始化的時候,首先對OTG驅(qū)動進(jìn)行注冊,OTG驅(qū)動將自身注冊為一個char設(shè)備。當(dāng)找到設(shè)備時,在注冊函數(shù)中執(zhí)行設(shè)備控制器驅(qū)動對應(yīng)的probe函數(shù),初始化OTG模塊,隨后調(diào)用中斷申請函數(shù)申請中斷,并設(shè)置相應(yīng)的中斷處理函數(shù)。在本設(shè)計中,當(dāng)有OTG中斷產(chǎn)生時,首先執(zhí)行硬件訪問層的中斷處理函數(shù)phy3343_hal_isr(),在這個函數(shù)中讀取中斷來源,若判斷是OTG中斷,則調(diào)用OTG驅(qū)動的中斷處理例程usb_ otg_isr(),啟動OTG狀態(tài)機。
OTG驅(qū)動還需要向文件系統(tǒng)提供相應(yīng)的file operations接口,供上層的application調(diào)用,這些接口函數(shù)包括:usb_otgdev_open、usb_ otgdev_close、usb_otgdev_ioctl、usb_otgdev_fasync等。usb_otgdev_open函數(shù)負(fù)責(zé)OTGapplication在打開OTG設(shè)備文件時,通過Linux文件系統(tǒng)接口調(diào)用該函數(shù),進(jìn)行OTG驅(qū)動自身參數(shù)的初始化;usb_otgdev_close函數(shù)負(fù)責(zé)OTG application在關(guān)閉OTG設(shè)備文件時,通過Linux文件系統(tǒng)接口調(diào)用該函數(shù),進(jìn)行OTG驅(qū)動自身參數(shù)重啟;usb_otgdev_ioctl函數(shù)是在OTGapplication調(diào)用OTG設(shè)備文件的ioctl函數(shù)時,Linux文件系統(tǒng)接口調(diào)用該函數(shù);usb_otgdev_fasync函數(shù)則是當(dāng)OTG的application設(shè)置或者重設(shè)異步通知時,對該函數(shù)進(jìn)行調(diào)用,當(dāng)異步通知模式設(shè)置成功返回值為0,失敗時返回負(fù)值。
Linux內(nèi)核中通過定義struet otg_transceiver結(jié)構(gòu)體,提供給開發(fā)者一個與USB硬件進(jìn)行直接交互的接口。在本設(shè)計中,即通過該結(jié)構(gòu)體實現(xiàn)了軟件與USB主控芯片USB3343的交互通信功能。
結(jié)語
USB OTG技術(shù)滿足了兩個設(shè)備之間直接進(jìn)行數(shù)據(jù)通信的要求。在對現(xiàn)有的OTG協(xié)議和相關(guān)應(yīng)用進(jìn)行深入研究的基礎(chǔ)上,本文在國產(chǎn)SoC芯片SEP6200平臺上設(shè)計并實現(xiàn)了基于Linux內(nèi)核的USB OTG模塊驅(qū)動。整個SEP6200核心板系統(tǒng)能夠支持USB主機模式、USB設(shè)備模式,并且能夠?qū)崿F(xiàn)兩個模式間的切換。
評論