日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當(dāng)前位置: 首頁 > 编程资源 > 综合教程 >内容正文

综合教程

Linux-USB Gadget : Part 4: 最简单的 gadget驱动:g_zero

發(fā)布時間:2023/12/3 综合教程 39 生活家
生活随笔 收集整理的這篇文章主要介紹了 Linux-USB Gadget : Part 4: 最简单的 gadget驱动:g_zero 小編覺得挺不錯的,現(xiàn)在分享給大家,幫大家做個參考.

Linux-USB Gadget : Part 4:?最簡單的?gadget驅(qū)動:g_zero

作者:?zjujoe?轉(zhuǎn)載請注明出處

Email?:?zjujoe@yahoo.com

BLOG?:?http://blog.csdn.net/zjujoe

前言

前面講過,?gadget api?提供了?usb device controller?驅(qū)動和上層?gadget?驅(qū)動交互的接口。?UDC?驅(qū)動是服務(wù)提供者,而各種?gadget?驅(qū)動則是服務(wù)的使用者。其實還有一些通用代碼,因為功能比較簡單,我們稱之為?helpe?函數(shù)。在閱讀了?Gadget API?文檔后,讓我們開始閱讀代碼,?udb?驅(qū)動代碼比較復(fù)雜,我們先從?gadget?驅(qū)動看起。各種?gadget?驅(qū)動中,?最簡單的要數(shù)?g_zero?驅(qū)動。

?

g_zero?驅(qū)動簡介

作為最簡單的?gadget?驅(qū)動,?g_zero?的功能基于兩個?BULK?端點實現(xiàn)了簡單的輸入輸出功能,?它可以用作寫新的?gadget?驅(qū)動的一個實例。?g_zero?驅(qū)動還有一個重要角色,?即配合?host?端的?usbtest?(內(nèi)核模塊及用戶層代碼),?用于測試底層?udc?驅(qū)動。當(dāng)然,也可以是測試主機的控制器驅(qū)動。

?

兩個?BULK?端點為一個?IN?端點?,??一個?OUT?端點。基于這兩個(由底層提供的)端點,?g_zero?驅(qū)動實現(xiàn)了兩個?configuration?。?第一個?configuration?提供了?sink/source?功能:兩個端點一個負責(zé)輸入,一個負責(zé)輸出,其中輸出的內(nèi)容根據(jù)設(shè)置可以是全?0?,也可以是按照某種算法生成的數(shù)據(jù)。另一個configuration?提供了?loopback?接口,?IN?端點負責(zé)把從?OUT?端點收到的數(shù)據(jù)反饋給?Host.

?

根據(jù)系統(tǒng)配置,?g_zero?驅(qū)動提供了全速及高速功能,從而稍微增加了代碼復(fù)雜度。另外,它還支持?otg?接口,從?usb2.0?協(xié)議我們知道,?otg?其實是?usb device?實現(xiàn)的一個補充功能。它增加了一套接口,使得同一設(shè)備可以在設(shè)備角色以及有限主機角色之切換。上層?gadget?驅(qū)動主要是在描述符方面提供配合支持。下面我們開始看代碼。

模塊初始化

1309?static int?__init?init?(void)

1310?{

1311??????????/* a real value would likely come through some id prom

1312???????????* or module option.??this one takes at least two packets.

1313???????????*/

1314??????????strlcpy?(serial?,?"0123456789.0123456789.0123456789"?, sizeof?serial?);

1315

1316??????????return?usb_gadget_register_driver?(&zero_driver?);

1317?}

1318?module_init?(init?);

1320?static void?__exit?cleanup?(void)

1321?{

1322??????????(&zero_driver?);

1323?}

1324?module_exit?(cleanup?);?usb_gadget_unregister_driver

?

Serial?變量存儲的是設(shè)備序列號,我們是一個非正式設(shè)備,隨便填充一下。模塊初始化函數(shù)調(diào)用?usb_gadget_register_driver?來向?udc driver?注冊一個?gadget?驅(qū)動, 我們這里是?zero_driver?。而退出函數(shù)則會做相反的操作:調(diào)用?usb_gadget_unregister_driver?取消原來的注冊。像所有的模塊初始化、退出函數(shù)一樣,現(xiàn)在還看不出什么花頭。我們再看一下?zero_driver?的定義:

?

1283?static struct?usb_gadget_driver?zero_driver?= {

1284?#ifdef CONFIG_USB_GADGET_DUALSPEED

1285??????????.speed???????????= USB_SPEED_HIGH,

1286?#else

1287??????????.speed???????????= USB_SPEED_FULL,

1288?#endif

1289??????????.function???????= (char *)?longname?,

1290??????????.bind????????????=?zero_bind?,

1291??????????.unbind?????????=?__exit_p?(zero_unbind?),

1293???????????.setup???????????=?zero_setup?,

1294??????????.disconnect?????=?zero_disconnect?,

1296???????????.suspend????????=?zero_suspend?,

1297??????????.resume?????????=?zero_resume?,

1299??????????.driver??????????= {

1300??????????????????.name????????????= (char *)?shortname?,

1301??????????????????.owner??????????=?THIS_MODULE?,

1302??????????},

1303?};

?

根據(jù)?CONFIG_USB_GADGET_DUALSPEED?,代碼選擇是支持高速還是全速,我們使用的?PXA?平臺支持高速傳輸,所以我們假定該配置變量為真。根據(jù)?Gadget API?文檔(以及下面的代碼調(diào)用圖),在初始化階段?usb_gadget_register_driver?函數(shù)會調(diào)用?bind?函數(shù),而?setup?函數(shù)是用于處理?udc?驅(qū)動沒有處理的控制傳輸部分。這兩個函數(shù)是整個?zero gadget驅(qū)動的精華部分。其它函數(shù)則只是為了完整性而提供,有興趣者可以對照?Gadget API?文檔及代碼,自行研究。至于?.driver?成員變量,那主要是為?LDM(linux device module)?服務(wù)的。現(xiàn)在關(guān)于?LDM?的文檔滿天飛,這里就不多說了。

?

簡單起見,我們目前不去深究?udc?驅(qū)動代碼(那比我們的?g_zero?驅(qū)動要復(fù)雜很多,?而且很多代碼硬件相關(guān),需要閱讀硬件?spec?才能理解),而是使用?kft?及?graphviz?(見參考,?colorant?大俠提供的文檔)工具得到函數(shù)調(diào)用關(guān)系圖:(我們這里使用?pxa udc?驅(qū)動,如果使用?dummy_hcd?會得到類似但更簡單的關(guān)系圖)

從上圖中,我們可以看到在初始化階段,?udc?驅(qū)動會調(diào)用?zero?驅(qū)動的?bind?函數(shù),也會調(diào)用?zero?驅(qū)動的?setup?函數(shù)?(?主要是得到一些描述符?)?,?setup?函數(shù)主要是在后面我們的?device?和主機連接后用于處理控制傳輸?shù)捻憫?yīng)(大部分)。在初始階段只是順便幫忙提供點信息,進行的是假傳輸,真提供信息給?udc?驅(qū)動。下面我們重點分析?bind?函數(shù)。

函數(shù)?zero_bind

1140?static int?__init

1141?zero_bind?(struct?usb_gadget?*gadget)

1142?{

1143??????????struct?zero_dev??????????*dev?;

1144??????????struct?usb_ep????????????*ep;

1145??????????int?????????????????????gcnum;

?

首先映入眼簾的是?zero_bind?函數(shù)的參數(shù),根據(jù)?Gadget API?,一個?gadget?代表一個?usb slave?設(shè)備。這個數(shù)據(jù)結(jié)構(gòu)是在底層控制器驅(qū)動中靜態(tài)分配的。?Udc?驅(qū)動在調(diào)用?gadget驅(qū)動各接口函數(shù)時都會提供這個數(shù)據(jù)結(jié)構(gòu)。

?

1147??????????/* FIXME this can't yet work right with SH ... it has only

1148???????????* one configuration, numbered one.

1149???????????*/

1150??????????if (gadget_is_sh?(gadget))

1151??????????????????return -ENODEV?;

?

注意我們以前說過?gadget_is_*?系列函數(shù)提供了查詢硬件能力的接口,這里用于判斷是否是?SH?平臺的?udc,?如果是,?直接出錯返回:?g_zero?驅(qū)動目前還不支持該平臺。

?

1153??????????/* Bulk-only drivers like this one SHOULD be able to

1154???????????* autoconfigure on any sane usb controller driver,

1155???????????* but there may also be important quirks to address.

1156???????????*/

1157??????????usb_ep_autoconfig_reset?(gadget);

?

注意函數(shù)?usb_ep_autoconfig_reset?不是由底層?udc?驅(qū)動實現(xiàn),而是我們以前提過的?helper?函數(shù)的一部分。該函數(shù)功能很簡單:用于清空?gadget?的?端點列表。

?

1158??????????ep =?usb_ep_autoconfig?(gadget, &fs_source_desc?);

1159??????????if (!ep) {

1160?autoconf_fail:

1161???????????????????printk?(KERN_ERR?"%s: can't autoconfigure on %s/n"?,

1162??????????????????????????shortname?, gadget->name?);

1163??????????????????return -ENODEV?;

1164??????????}

1165??????????EP_IN_NAME?= ep->name?;

1166??????????ep->driver_data?= ep;???/* claim */

1167?????????

1168??????????ep =?usb_ep_autoconfig?(gadget, &fs_sink_desc?);

1169??????????if (!ep)

1170??????????????????goto autoconf_fail;

1171??????????EP_OUT_NAME?= ep->name?;

1172??????????ep->driver_data?= ep;???/* claim */

?

函數(shù)?usb_ep_autoconfig?根據(jù)第二個參數(shù)所描述的限制條件,自動尋找適合條件的端點,并插入?gadget?的端點列表。這里?ep?是普通的數(shù)據(jù)端點,它的?driver_data?不需要存放特殊數(shù)據(jù),那就保存一下自己的地址吧。(后面我們將看到?ep0?的?driver_data?放的是?zero_driver?的特殊數(shù)據(jù))。我們看一下?fs_source_desc:

?

296?static struct?usb_endpoint_descriptor

297?fs_source_desc?= {

298??????????.bLength =??????????????USB_DT_ENDPOINT_SIZE?,

299??????????.bDescriptorType =??????USB_DT_ENDPOINT?,

300

301??????????.bEndpointAddress =?????USB_DIR_IN?,

302??????????.bmAttributes =?????????USB_ENDPOINT_XFER_BULK?,

303?};

?

可見該描述符描述的是一個類型為?BULK,?方向為?IN?的端點。?fs_sink_desc?的定義類似,描述一個類型為?BULK,?方向為?OUT?的端點。下面繼續(xù)看?zero_bind?的代碼。

?

1174??????????gcnum =?usb_gadget_controller_number?(gadget);

1175??????????if (gcnum >= 0)

1176??????????????????device_desc?.bcdDevice?=?cpu_to_le16?(0x0200 + gcnum);

1177??????????else {

1178??????????????????/* gadget zero is so simple (for now, no altsettings) that

1179???????????????????* it SHOULD NOT have problems with bulk-capable hardware.

1180???????????????????* so warn about unrcognized controllers, don't panic.

1181???????????????????*

1182???????????????????* things like configuration and altsetting numbering

1183???????????????????* can need hardware-specific attention though.

1184???????????????????*/

1185??????????????????printk?(KERN_WARNING?"%s: controller '%s' not recognized/n"?,

1186??????????????????????????shortname?, gadget->name?);

1187??????????????????device_desc?.bcdDevice?=?__constant_cpu_to_le16?(0x9999);

1188??????????}

?

每一個?udc?驅(qū)動被分配了一個編號,用作該設(shè)備描述符里的?bcd?碼。?如果沒有分配,沒辦法,就將就著用?0x9999?吧。

?

1191??????????/* ok, we made sense of the hardware ... */

1192??????????dev?=?kzalloc?(sizeof(*dev?),?GFP_KERNEL?);

1193??????????if (!dev?)

1194??????????????????return -ENOMEM?;

1195??????????spin_lock_init?(&dev?->lock?);

1196??????????dev?->gadget = gadget;

1197??????????set_gadget_data?(gadget,?dev?);

1198

?

stuct gadget?維護所有?gadget?驅(qū)動共性的內(nèi)容,個性的數(shù)據(jù)則由各?gadget?驅(qū)動各自定義,對于?zero,?它定義了?zero_dev.?分配后存放在?gadget?結(jié)構(gòu)的某個角落里:gadget.dev.driver_data?。?zero_dev?定義如下:

?

119?struct?zero_dev?{

120??????????spinlock_t???????????????lock?;

121??????????struct?usb_gadget????????*gadget;

122??????????struct?usb_request???????*req;???????????/* for control responses */

124??????????/* when configured, we have one of two configs:

125???????????* - source data (in to host) and sink it (out from host)

126???????????* - or loop it back (out from host back in to host)

127???????????*/

128??????????u8???????????????????????config?;

129??????????struct?usb_ep????????????*in_ep, *out_ep;

131??????????/* autoresume timer */

132??????????struct?timer_list????????resume;

133?};

?

這里?resume?是用于喚醒?host?的?timer?的列表,?config?表示我們當(dāng)前使用第幾個?configuration.?其它含義自明。下面繼續(xù)看?zero bind?代碼。

?

1199??????????/* preallocate control response and buffer */

1200??????????dev?->req =?usb_ep_alloc_request?(gadget->ep0,?GFP_KERNEL?);

1201??????????if (!dev?->req)

1202??????????????????goto enomem;

1203??????????dev?->req->buf?=?usb_ep_alloc_buffer?(gadget->ep0,?USB_BUFSIZ?,

1204??????????????????????????????????&dev?->req->dma?,?GFP_KERNEL?);

1205??????????if (!dev?->req->buf?)

1206??????????????????goto enomem;

1207

1208??????????dev?->req->complete?=?zero_setup_complete?;

1209

?

這幾行代碼分配用于控制傳輸?shù)恼埱?/?數(shù)據(jù)緩沖以及結(jié)束函數(shù)。控制傳輸是每個?gadget?驅(qū)動要使用的傳輸方式,這里及早分配。結(jié)束函數(shù)?zero_setup_complete?只是打印一下狀態(tài),我們就不貼出了。

?

1210??????????device_desc?.bMaxPacketSize0 = gadget->ep0->maxpacket;

?

這里根據(jù)底層的數(shù)據(jù)初始化設(shè)備描述符里端點?0?(控制端點)的最大包大小。

?

1212?#ifdef CONFIG_USB_GADGET_DUALSPEED

1213??????????/* assume ep0 uses the same value for both speeds ... */

1214??????????dev_qualifier?.bMaxPacketSize0 =?device_desc?.bMaxPacketSize0;

1215

1216??????????/* and that all endpoints are dual-speed */

1217??????????hs_source_desc?.bEndpointAddress =?fs_source_desc?.bEndpointAddress;

1218??????????hs_sink_desc?.bEndpointAddress =?fs_sink_desc?.bEndpointAddress;

1219?#endif

?

高速設(shè)備需要的額外的描述符,我們對某些字段進行初始化。

?

1221??????????if (gadget->is_otg) {

1222??????????????????otg_descriptor?.bmAttributes |=?USB_OTG_HNP?,

1223??????????????????source_sink_config?.bmAttributes |=?USB_CONFIG_ATT_WAKEUP?;

1224???????????????????loopback_config?.bmAttributes |=?USB_CONFIG_ATT_WAKEUP?;

1225??????????}

?

如果是?otg?設(shè)備,則需要在描述符里設(shè)置相關(guān)特性。

?

1227??????????usb_gadget_set_selfpowered?(gadget);

?

能運行?Linux Gadget?驅(qū)動的設(shè)備一般電池供電,也就是?selfpowered?。

?

1229???????????init_timer?(&dev?->resume);

1230??????????dev?->resume.function =?zero_autoresume?;

1231??????????dev?->resume.data?= (unsigned long)?dev?;

1232??????????if (autoresume?) {

1233??????????????????source_sink_config?.bmAttributes |=?USB_CONFIG_ATT_WAKEUP?;

1234??????????????????loopback_config?.bmAttributes |=?USB_CONFIG_ATT_WAKEUP?;

1235??????????}

?

這段代碼跟自動喚醒?host?有關(guān),?不深究。

?

1237??????????gadget->ep0->driver_data?=?dev?;

?

多記一份?zero_dev?的地址,?方便使用。

?

1239??????????INFO?(dev?,?"%s, version: "?DRIVER_VERSION?"/n"?,?longname?);

1240??????????INFO?(dev?,?"using %s, OUT %s IN %s/n"?, gadget->name?,

1241???????????????????EP_OUT_NAME?,?EP_IN_NAME?);

1242

1243??????????snprintf?(manufacturer?, sizeof?manufacturer?,?"%s %s with %s"?,

1244??????????????????init_utsname?()->sysname,?init_utsname?()->release?,

1245??????????????????gadget->name?);

1246

1247??????????return 0;

1248

1249?enomem:

1250??????????zero_unbind?(gadget);

1251??????????return -ENOMEM?;

1252?}

?

自此???zero_bind?分析完畢。它主要是為?gadget?驅(qū)動找到了合適的端點,并且初始化了設(shè)備相關(guān)結(jié)構(gòu)?: zero_dev.?從而把?gadget?驅(qū)動和???udc?驅(qū)動僅僅地綁定在一起。?看到現(xiàn)在,我們還沒有感受到?gadget?驅(qū)動的真正意義,?前面的函數(shù)就像一座座橋梁,走過這些橋梁,我們終于來到美麗的湖心小島:zero_setup?。

?

函數(shù)?zero_setup

zero_setup?完成控制傳輸?shù)拇蟛糠止δ堋1热绔@取各種描述符、設(shè)置配置等。?Host?首先通過控制傳輸和設(shè)備進行通信,告訴設(shè)備它底下要干什么。?Zero gadget?驅(qū)動比較簡單,在主機進行?set configuration?后,就會在?IN/OUT?端點上準(zhǔn)備好數(shù)據(jù),供主機去用。并且通過?call?函數(shù),在主機使用完前面準(zhǔn)備好的數(shù)據(jù)后,繼續(xù)插入請求,這樣,主機就可以源源不斷的對我們這個設(shè)備進行讀寫操作。以下開始看代碼。

?

917?static int

918?zero_setup?(struct?usb_gadget?*gadget, const struct?usb_ctrlrequest?*ctrl?)

919?{

?

照例,我們得到?usb_gadget?結(jié)構(gòu),同時,我們的第二個參數(shù)是?usb_ctrlrequest?結(jié)構(gòu):

?
140 struct usb_ctrlrequest {
141???????? __u8 bRequestType;
142???????? __u8 bRequest;
143???????? __le16 wValue;
144???????? __le16 wIndex;
145???????? __le16 wLength;
146 } __attribute__ ((packed));

?

具體含義請參考?Usb spec Ch9?。 這里結(jié)構(gòu)名字有點誤導(dǎo),?usb_ctrlrequest?代表的?是主機傳過來的控制請求。和后面的?usb_request?有較大區(qū)別。?usb_request?代表放到端點的隊列里等待主機過來讀寫的一個個數(shù)據(jù)包。下面我們繼續(xù)看?zero_setup?函數(shù)代碼。

?

920??????????struct?zero_dev??????????*dev?=?get_gadget_data?(gadget);

921??????????struct?usb_request???????*req =?dev?->req;

922??????????int?????????????????????value?= -EOPNOTSUPP?;

923??????????u16??????????????????????w_index =?le16_to_cpu?(ctrl?->wIndex);

924??????????u16??????????????????????w_value =?le16_to_cpu?(ctrl?->wValue);

925??????????u16??????????????????????w_length =?le16_to_cpu?(ctrl?->wLength);

?

獲得我們在?bind?函數(shù)分配的?zero_dev, usb_request?, 以及由主機傳過來的“請求”的各字段。

?

927??????????/* usually this stores reply data in the pre-allocated ep0 buffer,

928???????????* but config change events will reconfigure hardware.*/

930???????????req->zero?= 0;

931??????????switch (ctrl?->bRequest) {

932

933??????????case?USB_REQ_GET_DESCRIPTOR?:

934??????????????????if (ctrl?->bRequestType !=?USB_DIR_IN?)

935???????????????????????????goto?unknown?;

?

請求各種描述符,當(dāng)然需要是?IN?類型的請求。

?

936??????????????????switch (w_value >> 8) {

938??????????????????case?USB_DT_DEVICE?:

939??????????????????????????value?=?min?(w_length, (u16?) sizeof?device_desc?);

940??????????????????????????memcpy?(req->buf?, &device_desc?,?value?);

941??????????????????????????break;

942?#ifdef CONFIG_USB_GADGET_DUALSPEED

943??????????????????case?USB_DT_DEVICE_QUALIFIER?:

944??????????????????????????if (!gadget->is_dualspeed)

945???????????????????????????????????break;

946??????????????????????????value?=?min?(w_length, (u16?) sizeof?dev_qualifier?);

947??????????????????????????memcpy?(req->buf?, &dev_qualifier?,?value?);

948??????????????????????????break;

?

對應(yīng)?USB 2.0 Spec CH9,?以上代碼很容易理解。 每一個描述符使用?struct?usb_device_descriptor?描述,比如, 設(shè)備描述符:

?

222?static struct?usb_device_descriptor

223?device_desc?= {

224??????????.bLength =??????????????sizeof?device_desc?,

225??????????.bDescriptorType =??????USB_DT_DEVICE?,

226

227??????????.bcdUSB =???????????????__constant_cpu_to_le16?(0x0200),

228??????????.bDeviceClass =?????????USB_CLASS_VENDOR_SPEC?, 0xff

229

230??????????.?idVendor?=?????????????__constant_cpu_to_le16?(?DRIVER_VENDOR_NUM?),

231??????????.?idProduct?=????????????__constant_cpu_to_le16?(?DRIVER_PRODUCT_NUM?),

232??????????.?iManufacturer?=????????STRING_MANUFACTURER?, 25?, 廠商描述符

233??????????.?iProduct?=?????????????STRING_PRODUCT?,????42?,廠品描述符

234???????????.?iSerialNumber?=????????STRING_SERIAL?,?????101,?序列號

235??????????.bNumConfigurations =???2,

236?};

?

950??????????????????case?USB_DT_OTHER_SPEED_CONFIG?:

951??????????????????????????if (!gadget->is_dualspeed)

952??????????????????????????????????break;

953??????????????????????????// FALLTHROUGH

954?#endif?/* CONFIG_USB_GADGET_DUALSPEED */

955??????????????????case?USB_DT_CONFIG?:

956??????????????????????????value?=?config_buf?(gadget, req->buf?,

957??????????????????????????????????????????w_value >> 8,

958??????????????????????????????????????????w_value & 0xff);

959??????????????????????????if (value?>= 0)

960??????????????????????????????????value?=?min?(w_length, (u16?)?value?);

961??????????????????????????break;

?

配置描述符比較復(fù)雜,會返回該配置里的接口,端點等信息。配置描述符由:struct?usb_descriptor_header?[]?表達, 而且高速/?全速設(shè)備的配置描述符是不一樣。比如,高速loopback?配置的配置描述符為:

?

378?static const struct?usb_descriptor_header?*?hs_loopback_function?[] = {

379??????????(struct?usb_descriptor_header?*) &?otg_descriptor?,

380??????????(struct?usb_descriptor_header?*) &?loopback_intf?,

381??????????(struct?usb_descriptor_header?*) &?hs_source_desc?,

382??????????(struct?usb_descriptor_header?*) &?hs_sink_desc?,

383??????????NULL?,

384?};

?

可見,本質(zhì)上,配置描述符是返回一組描述符。下面看一下配置描述符是如何生成的。

?

432?static int

433?config_buf?(struct?usb_gadget?*gadget,

434???????????????????u8?*?buf?,?u8?type?, unsigned?index?)

435?{

436??????????int?????????????????????????????is_source_sink;

437??????????int??????????????????????????????len?;

438??????????const struct?usb_descriptor_header?**function;

439?#ifdef CONFIG_USB_GADGET_DUALSPEED

440??????????int?????????????????????????????hs = (gadget->?speed?== USB_SPEED_HIGH);

441?#endif

442

443??????????/* two configurations will always be index 0 and index 1 */

444??????????if (?index?> 1)

445??????????????????return -?EINVAL?;

446??????????is_source_sink =?loopdefault?? (?index?== 1) : (?index?== 0);

447

448?#ifdef CONFIG_USB_GADGET_DUALSPEED

449??????????if (?type?==?USB_DT_OTHER_SPEED_CONFIG?)

450??????????????????hs = !hs;

451??????????if (hs)

452??????????????????function = is_source_sink

453????????????????????????????hs_source_sink_function

454??????????????????????????:?hs_loopback_function?;

455??????????else

456?#endif

457??????????????????function = is_source_sink

458????????????????????????????fs_source_sink_function

459??????????????????????????:?fs_loopback_function?;

460

461??????????/* for now, don't advertise srp-only devices */

462??????????if (!gadget->is_otg)

463??????????????????function++;

464

465??????????len?=?usb_gadget_config_buf?(is_source_sink

466??????????????????????????????????????????? &?source_sink_config

467??????????????????????????????????????????: &?loopback_config?,

468??????????????????????????buf?,?USB_BUFSIZ?, function);

469??????????if (?len?< 0)

470???????????????????return?len?;

471??????????((struct?usb_config_descriptor?*)?buf?)->bDescriptorType =?type?;

472??????????return?len?;

473?}

?

代碼很簡單,?config_buf?函數(shù)根據(jù)當(dāng)前是否是高速設(shè)備,以及是否是?otg?設(shè)備,選擇合適的?configuration( souce sink config or loopback config)?,?調(diào)用usb_gadget_config_buf?生成最終的配置描述符。可以想象?usb_gadget_config_buf?的實現(xiàn)非常簡單?:?根據(jù)傳過來的?描述符列表?(?以?NULL?指針結(jié)束?)?,使用?memcpy??之類見每個描述符的內(nèi)容拷貝到?buf?里。?下面我們繼續(xù)看???zero_setup?函數(shù)。

?

963??????????????????case?USB_DT_STRING?:

964??????????????????????????/* wIndex == language code.

965???????????????????????????* this driver only handles one language, you can

966???????????????????????????* add string tables for other languages, using

967???????????????????????????* any UTF-8 characters

968???????????????????????????*/

969??????????????????????????value?=?usb_gadget_get_string?(&stringtab?,

970??????????????????????????????????????????w_value & 0xff, req->buf?);

971??????????????????????????if (value?>= 0)

972??????????????????????????????????value?=?min?(w_length, (u16?)?value?);

973??????????????????????????break;

974??????????????????}

975??????????????????break;

976

?

根據(jù)?host?傳遞過來的索引,響應(yīng)相應(yīng)的字符串。Zero?驅(qū)動的字符串描述符則只支持一種語言(0409, en-us?):

?

409?static struct?usb_gadget_strings?????????stringtab?= {

410??????????.language???????= 0x0409,???????/* en-us */

411??????????.strings?????????=?strings?,

412?};

?

399?/* static strings, in UTF-8 */

400?static struct?usb_string?????????????????strings?[] = {

401??????????{?STRING_MANUFACTURER?,?manufacturer?, },

402??????????{?STRING_PRODUCT?,?longname?, },

403??????????{?STRING_SERIAL?,?serial?, },

404??????????{?STRING_LOOPBACK?,?loopback?, },

405??????????{?STRING_SOURCE_SINK?,?source_sink?, },

406??????????{??}????????????????????/* end of list */

407?};

?

有點像應(yīng)用層(比如?vc?)為了支持多語言而獨立出來的字符串資源。事實上就是這樣!我們可以很容易再增加一種語言。下面我們繼續(xù)看???zero_setup?函數(shù)。

?

977??????????/* currently two configs, two speeds */

978??????????case?USB_REQ_SET_CONFIGURATION?:

979??????????????????if (ctrl?->bRequestType != 0)

980??????????????????????????goto?unknown?;

981??????????????????if (gadget->a_hnp_support)

982??????????????????????????DBG?(dev?,?"HNP available/n"?);

983??????????????????else if (gadget->a_alt_hnp_support)

984??????????????????????????DBG?(dev?,?"HNP needs a different root port/n"?);

985??????????????????else

986??????????????????????????VDBG?(dev?,?"HNP inactive/n"?);

987??????????????????spin_lock?(&dev?->lock?);

988??????????????????value?=?zero_set_config?(dev?, w_value,?GFP_ATOMIC?);

989??????????????????spin_unlock?(&dev?->lock?);

990??????????????????break;

?

設(shè)置設(shè)備的當(dāng)前配置,到這里,才凌空一腳,將設(shè)備帶入數(shù)據(jù)傳輸狀態(tài),我們先把zero_setup?看完,再仔細看函數(shù)?zero_set_config?。

?

991??????????case?USB_REQ_GET_CONFIGURATION?:

992??????????????????if (ctrl?->bRequestType !=?USB_DIR_IN?)

993??????????????????????????goto?unknown?;

994??????????????????*(u8?*)req->buf?=?dev?->config?;

995??????????????????value?=?min?(w_length, (u16?) 1);

996??????????????????break;

?

獲取設(shè)備的當(dāng)前配置

?

998??????????/* until we add altsetting support, or other interfaces,

999???????????* only 0/0 are possible.??pxa2xx only supports 0/0 (poorly)

1000???????????* and already killed pending endpoint I/O.

1001???????????*/

1002??????????case?USB_REQ_SET_INTERFACE?:

1003??????????????????if (ctrl?->bRequestType !=?USB_RECIP_INTERFACE?)

1004???????????????????????????goto?unknown?;

1005??????????????????spin_lock?(&dev?->lock?);

1006??????????????????if (dev?->config?&& w_index == 0 && w_value == 0) {

1007??????????????????????????u8???????????????config?=?dev?->config?;

1008

1009??????????????????????????/* resets interface configuration, forgets about

1010???????????????????????????* previous transaction state (queued bufs, etc)

1011???????????????????????????* and re-inits endpoint state (toggle etc)

1012???????????????????????????* no response queued, just zero status == success.

1013???????????????????????????* if we had more than one interface we couldn't

1014????????????????????????????* use this "reset the config" shortcut.

1015???????????????????????????*/

1016??????????????????????????zero_reset_config?(dev?);

1017??????????????????????????zero_set_config?(dev?,?config?,?GFP_ATOMIC?);

1018??????????????????????????value?= 0;

1019??????????????????}

1020??????????????????spin_unlock?(&dev?->lock?);

1021??????????????????break;

?

設(shè)置接口,由于我們每個configuration?只有一個接口,所以這里的效果跟前面設(shè)置配置類似。

由于?zero_set_config?函數(shù)會調(diào)用?zero_reset_config,?所以這里應(yīng)該可以不調(diào)用?zero_reset_config.

?

1022??????????case?USB_REQ_GET_INTERFACE?:

1023??????????????????if (ctrl?->bRequestType != (USB_DIR_IN?|USB_RECIP_INTERFACE?))

1024??????????????????????????goto?unknown?;

1025??????????????????if (!dev?->config?)

1026??????????????????????????break;

1027??????????????????if (w_index != 0) {

1028??????????????????????????value?= -EDOM?;

1029??????????????????????????break;

1030???????????????????}

1031??????????????????*(u8?*)req->buf?= 0;

1032??????????????????value?=?min?(w_length, (u16?) 1);

1033??????????????????break;

1034

?

獲取設(shè)備的當(dāng)前配置的當(dāng)前接口。

?

1035??????????/*

1036???????????* These are the same vendor-specific requests supported by

1037????????????* Intel's USB 2.0 compliance test devices.??We exceed that

1038???????????* device spec by allowing multiple-packet requests.

1039???????????*/

1040??????????case 0x5b:???????/* control WRITE test -- fill the buffer */

1041??????????????????if (ctrl?->bRequestType != (USB_DIR_OUT?|USB_TYPE_VENDOR?))

1042???????????????????????????goto?unknown?;

1043??????????????????if (w_value || w_index)

1044??????????????????????????break;

1045??????????????????/* just read that many bytes into the buffer */

1046??????????????????if (w_length >?USB_BUFSIZ?)

1047??????????????????????????break;

1048???????????????????value?= w_length;

1049??????????????????break;

1050??????????case 0x5c:???????/* control READ test -- return the buffer */

1051???????????????????if (ctrl?->bRequestType != (USB_DIR_IN?|USB_TYPE_VENDOR?))

1052??????????????????????????goto?unknown?;

1053??????????????????if (w_value || w_index)

1054??????????????????????????break;

1055??????????????????/* expect those bytes are still in the buffer; send back */

1056??????????????????if (w_length >?USB_BUFSIZ

1057???????????????????????????????????|| w_length != req->length?)

1058??????????????????????????break;

1059??????????????????value?= w_length;

1060??????????????????break;

1061

?

根據(jù)協(xié)議,我們可以定制私有的類型,這里是?Intel?定義的測試類型,用于測試端點0?的數(shù)據(jù)收發(fā)。端點0?通常用于控制傳輸, 用它進行數(shù)據(jù)傳輸完全是為了測試目的。

?

1062??????????default:

1063?unknown?:

1064??????????????????VDBG?(dev?,

1065??????????????????????????"unknown control req%02x.%02x v%04x i%04x l%d/n"?,

1066??????????????????????????ctrl?->bRequestType,?ctrl?->bRequest,

1067??????????????????????????w_value, w_index, w_length);

1068??????????}

1069

1070??????????/* respond with data transfer before status phase? */

1071??????????if (value?>= 0) {

1072??????????????????req->length?=?value?;

1073??????????????????req->zero?=?value?< w_length;

1074??????????????????value?=?usb_ep_queue?(gadget->ep0, req,?GFP_ATOMIC?);

1075??????????????????if (value?< 0) {

1076???????????????????????????DBG?(dev?,?"ep_queue --> %d/n"?,?value?);

1077??????????????????????????req->status?= 0;

1078??????????????????????????zero_setup_complete?(gadget->ep0, req);

1079??????????????????}

1080??????????}

?

如果有數(shù)據(jù)需要傳給?Host,?則將其放到端點0?的傳送隊列。底層?udc?驅(qū)動會負責(zé)將其發(fā)給?host.

?

1082??????????/* device either stalls (value < 0) or reports success */

1083??????????return?value?;

1084?}

?

函數(shù)?zero_setup?完成了?usb spec ch9?定義的很多功能。而我們前面介紹的?bulk-in/bulk-out?數(shù)據(jù)端點開始工作則是在?set configuration?(或者?set interface?)后,由?zero_set_config?函數(shù)觸發(fā)。下面開始分析該函數(shù)。

?

函數(shù)zero_set_config

848?static int

849?zero_set_config?(struct?zero_dev?*dev?, unsigned?number?,?gfp_t?gfp_flags)

850?{

851??????????int?????????????????????result?= 0;

852??????????struct?usb_gadget????????*gadget =?dev?->gadget;

853

854???????????if (number?==?dev?->config?)

855??????????????????return 0;

862??????????zero_reset_config?(dev?);

863

?

函數(shù)?zero_reset_config?把所有的?端點置于?disable?狀態(tài)。

?

864??????????switch (number?) {

865??????????case?CONFIG_SOURCE_SINK?:

866??????????????????result?=?set_source_sink_config?(dev?, gfp_flags);

867??????????????????break;

868??????????case?CONFIG_LOOPBACK?:

869???????????????????result?=?set_loopback_config?(dev?, gfp_flags);

870??????????????????break;

871???????????default:

872??????????????????result?= -EINVAL?;

873??????????????????/* FALL THROUGH */

874??????????case 0:

875??????????????????return?result?;

876??????????}

?

根據(jù)當(dāng)前的配置,設(shè)置兩種不同的傳送方式。我們假定?host?設(shè)置的是?loopback?方式。另一種方式是類似的?(?數(shù)據(jù)內(nèi)容不同?)?。

?

878??????????if (!result?&& (!dev?->in_ep || !dev?->out_ep))

879??????????????????result?= -ENODEV?;

880??????????if (result?)

881??????????????????zero_reset_config?(dev?);

882??????????else {

883??????????????????char *speed?;

884

885??????????????????switch (gadget->speed?) {

886??????????????????case USB_SPEED_LOW:?????speed?=?"low"?; break;

887??????????????????case USB_SPEED_FULL:????speed?=?"full"?; break;

888??????????????????case USB_SPEED_HIGH:????speed?=?"high"?; break;

889??????????????????default:????????????????speed?=?"?"?; break;

890??????????????????}

891

892??????????????????dev?->config?=?number?;

893??????????????????INFO?(dev?,?"%s speed config #%d: %s/n"?,?speed?,?number?,

894??????????????????????????????????(number?==?CONFIG_SOURCE_SINK?)

895????????????????????????????????????????????source_sink?:?loopback?);

896??????????}

897??????????return?result?;

898?}

一些善后處理。?下面我們看函數(shù)?set_loopback_config

函數(shù)?set_loopback_config

747?static int

748?set_loopback_config?(struct?zero_dev?*dev?,?gfp_t?gfp_flags)

749?{

750??????????int?????????????????????result?= 0;

751??????????struct?usb_ep????????????*ep;

752??????????struct?usb_gadget????????*gadget =?dev?->gadget;

753

754??????????gadget_for_each_ep?(ep, gadget) {

?

針對?gadget?端點列表的每一個端點進行操作。

?

755??????????????????const struct?usb_endpoint_descriptor?????*d;

756

757??????????????????/* one endpoint writes data back IN to the host */

758??????????????????if (strcmp?(ep->name?,?EP_IN_NAME?) == 0) {

759??????????????????????????d =?ep_desc?(gadget, &hs_source_desc?, &fs_source_desc?);

760??????????????????????????result?=?usb_ep_enable?(ep, d);

761??????????????????????????if (result?== 0) {

762???????????????????????????????????ep->driver_data?=?dev?;

763??????????????????????????????????dev?->in_ep = ep;

764???????????????????????????????????continue;

765??????????????????????????}

766

767??????????????????/* one endpoint just reads OUT packets */

768??????????????????} else if (strcmp?(ep->name?,?EP_OUT_NAME?) == 0) {

769??????????????????????????d =?ep_desc?(gadget, &hs_sink_desc?, &fs_sink_desc?);

770??????????????????????????result?=?usb_ep_enable?(ep, d);

771??????????????????????????if (result?== 0) {

772??????????????????????????????????ep->driver_data?=?dev?;

773??????????????????????????????????dev?->out_ep = ep;

774??????????????????????????????????continue;

775??????????????????????????}

776

777??????????????????/* ignore any other endpoints */

778??????????????????} else

779??????????????????????????continue;

780

781??????????????????/* stop on error */

782??????????????????ERROR?(dev?,?"can't enable %s, result %d/n"?, ep->name?,?result?);

783??????????????????break;

784??????????}

?

激活端點。并設(shè)置速度?(?高速或者全速?)?。

?

786??????????/* allocate a bunch of read buffers and queue them all at once.

787???????????* we buffer at most 'qlen' transfers; fewer if any need more

788???????????* than 'buflen' bytes each.

789???????????*/

790??????????if (result?== 0) {

791??????????????????struct?usb_request???????*req;

792??????????????????unsigned????????????????i?;

793

794??????????????????ep =?dev?->out_ep;

795??????????????????for (i?= 0;?i?<?qlen?&&?result?== 0;?i?++) {

796??????????????????????????req =?alloc_ep_req?(ep,?buflen?);

797??????????????????????????if (req) {

798??????????????????????????????????req->complete?=?loopback_complete?;

799??????????????????????????????????result?=?usb_ep_queue?(ep, req,?GFP_ATOMIC?);

800??????????????????????????????????if (result?)

801??????????????????????????????????????????DBG?(dev?,?"%s queue req --> %d/n"?,

802??????????????????????????????????????????????????????????ep->name?,?result?);

803??????????????????????????} else

804??????????????????????????????????result?= -ENOMEM?;

805??????????????????}

806??????????}

?

首先在?OUT?端點上掛一堆請求(?usb_request?)?,?等待主機向我們發(fā)送數(shù)據(jù)。等主機真正對我們進行?OUT?數(shù)據(jù)傳輸并且數(shù)據(jù)傳完后,會調(diào)用?loopback_complete?回調(diào)函數(shù)。

?

807??????????if (result?== 0)

808??????????????????DBG?(dev?,?"qlen %d, buflen %d/n"?,?qlen?,?buflen?);

809

810???????????/* caller is responsible for cleanup on error */

811??????????return?result?;

812?}

?

下面看?函數(shù)?loopback_complete

?

函數(shù)?loopback_complete

698?static void?loopback_complete?(struct?usb_ep?*ep, struct?usb_request?*req)

699?{

700??????????struct?zero_dev?*dev?= ep->driver_data?;

701???????????int?????????????status?= req->status?;

702

703??????????switch (status?) {

704

705??????????case 0:?????????????????????????/* normal completion? */

706??????????????????if (ep ==?dev?->out_ep) {

707??????????????????????????/* loop this OUT packet back IN to the host */

708??????????????????????????req->zero?= (req->actual < req->length?);

709??????????????????????????req->length?= req->actual;

710??????????????????????????status?=?usb_ep_queue?(dev?->in_ep?, req,?GFP_ATOMIC?);

711??????????????????????????if (status?== 0)

712??????????????????????????????????return;

713

714??????????????????????????/* "should never get here" */

715??????????????????????????ERROR?(dev?,?"can't loop %s to %s: %d/n"?,

716??????????????????????????????????ep->name?,?dev?->in_ep->name?,

717??????????????????????????????????status?);

718??????????????????}

719

720??????????????????/* queue the buffer for some later OUT packet */

721??????????????????req->length?=?buflen?;

722??????????????????status?=?usb_ep_queue?(dev?->out_ep?, req,?GFP_ATOMIC?);

723???????????????????if (status?== 0)

724??????????????????????????return;

725

726??????????????????/* "should never get here" */

727??????????????????/* FALLTHROUGH */

728

729??????????default:

730??????????????????ERROR?(dev?,?"%s loop complete --> %d, %d/%d/n"?, ep->name?,

731??????????????????????????????????status?, req->actual, req->length?);

732??????????????????/* FALLTHROUGH */

733

734??????????/* NOTE:??since this driver doesn't maintain an explicit record

735????????????* of requests it submitted (just maintains qlen count), we

736???????????* rely on the hardware driver to clean up on disconnect or

737???????????* endpoint disable.

738???????????*/

739??????????case -ECONNABORTED?:?????????????/* hardware forced ep reset */

740??????????case -ECONNRESET?:???????????????/* request dequeued */

741??????????case -ESHUTDOWN?:????????????????/* disconnect from host */

742??????????????????free_ep_req?(ep, req);

743???????????????????return;

744??????????}

745?}

?

如果?OUT?傳輸正常結(jié)束,則會將其放到?IN?端點的傳輸隊列。

如果?IN?傳輸正常結(jié)束,則會將其放到?OUT?端點的傳輸隊列。

這樣,通過回調(diào)函數(shù)不斷在兩個隊列?(IN/OUT)?之間切換這些請求?(usb_request),?就實現(xiàn)了在主機看來的?loopback?設(shè)備。

總結(jié)

Gadget?驅(qū)動的特殊性在于它是?host?端對等驅(qū)動的?slave,?而不是上層某個應(yīng)用的?slave.?響應(yīng)的,它是實現(xiàn)是很有意思的。我們沒有看到?read/write?函數(shù),也沒有看到我們最常實現(xiàn)的?ioctl?函數(shù),?而是把重點放在回調(diào)函數(shù)?zero_setup?上。?g_zero gadget?驅(qū)動實現(xiàn)了一個最簡單的?bulk-in/bulk-out?功能,向我們展示了?gadget?驅(qū)動如果利用?gadget API?來完成數(shù)據(jù)傳輸功能。對于復(fù)雜的?gadget?驅(qū)動,?setup?回調(diào)函數(shù)只是一個起點。

?

參考

USB 2.0 Spec

http://www.usb.org/developers/docs/

?

用?KFI?和?Graphviz?跟蹤?/?優(yōu)化內(nèi)核代碼

http://blog.csdn.net/colorant/archive/2008/07/09/2627493.aspx

總結(jié)

以上是生活随笔為你收集整理的Linux-USB Gadget : Part 4: 最简单的 gadget驱动:g_zero的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網(wǎng)站內(nèi)容還不錯,歡迎將生活随笔推薦給好友。