日韩av黄I国产麻豆传媒I国产91av视频在线观看I日韩一区二区三区在线看I美女国产在线I麻豆视频国产在线观看I成人黄色短片

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 综合教程 >内容正文

综合教程

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

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

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

作者:?zjujoe?轉載請注明出處

Email?:?zjujoe@yahoo.com

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

前言

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

?

g_zero?驅動簡介

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

?

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

?

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

模塊初始化

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?變量存儲的是設備序列號,我們是一個非正式設備,隨便填充一下。模塊初始化函數調用?usb_gadget_register_driver?來向?udc driver?注冊一個?gadget?驅動, 我們這里是?zero_driver?。而退出函數則會做相反的操作:調用?usb_gadget_unregister_driver?取消原來的注冊。像所有的模塊初始化、退出函數一樣,現在還看不出什么花頭。我們再看一下?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?};

?

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

?

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

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

函數?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?函數的參數,根據?Gadget API?,一個?gadget?代表一個?usb slave?設備。這個數據結構是在底層控制器驅動中靜態分配的。?Udc?驅動在調用?gadget驅動各接口函數時都會提供這個數據結構。

?

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?平臺的?udc,?如果是,?直接出錯返回:?g_zero?驅動目前還不支持該平臺。

?

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);

?

注意函數?usb_ep_autoconfig_reset?不是由底層?udc?驅動實現,而是我們以前提過的?helper?函數的一部分。該函數功能很簡單:用于清空?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 */

?

函數?usb_ep_autoconfig?根據第二個參數所描述的限制條件,自動尋找適合條件的端點,并插入?gadget?的端點列表。這里?ep?是普通的數據端點,它的?driver_data?不需要存放特殊數據,那就保存一下自己的地址吧。(后面我們將看到?ep0?的?driver_data?放的是?zero_driver?的特殊數據)。我們看一下?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?的端點。下面繼續看?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?驅動被分配了一個編號,用作該設備描述符里的?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?驅動共性的內容,個性的數據則由各?gadget?驅動各自定義,對于?zero,?它定義了?zero_dev.?分配后存放在?gadget?結構的某個角落里: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?表示我們當前使用第幾個?configuration.?其它含義自明。下面繼續看?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

?

這幾行代碼分配用于控制傳輸的請求?/?數據緩沖以及結束函數。控制傳輸是每個?gadget?驅動要使用的傳輸方式,這里及早分配。結束函數?zero_setup_complete?只是打印一下狀態,我們就不貼出了。

?

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

?

這里根據底層的數據初始化設備描述符里端點?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

?

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

?

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?設備,則需要在描述符里設置相關特性。

?

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

?

能運行?Linux Gadget?驅動的設備一般電池供電,也就是?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?有關,?不深究。

?

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?驅動找到了合適的端點,并且初始化了設備相關結構?: zero_dev.?從而把?gadget?驅動和???udc?驅動僅僅地綁定在一起。?看到現在,我們還沒有感受到?gadget?驅動的真正意義,?前面的函數就像一座座橋梁,走過這些橋梁,我們終于來到美麗的湖心小島:zero_setup?。

?

函數?zero_setup

zero_setup?完成控制傳輸的大部分功能。比如獲取各種描述符、設置配置等。?Host?首先通過控制傳輸和設備進行通信,告訴設備它底下要干什么。?Zero gadget?驅動比較簡單,在主機進行?set configuration?后,就會在?IN/OUT?端點上準備好數據,供主機去用。并且通過?call?函數,在主機使用完前面準備好的數據后,繼續插入請求,這樣,主機就可以源源不斷的對我們這個設備進行讀寫操作。以下開始看代碼。

?

917?static int

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

919?{

?

照例,我們得到?usb_gadget?結構,同時,我們的第二個參數是?usb_ctrlrequest?結構:

?
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?。 這里結構名字有點誤導,?usb_ctrlrequest?代表的?是主機傳過來的控制請求。和后面的?usb_request?有較大區別。?usb_request?代表放到端點的隊列里等待主機過來讀寫的一個個數據包。下面我們繼續看?zero_setup?函數代碼。

?

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?函數分配的?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?;

?

請求各種描述符,當然需要是?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;

?

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

?

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;

?

配置描述符比較復雜,會返回該配置里的接口,端點等信息。配置描述符由:struct?usb_descriptor_header?[]?表達, 而且高速/?全速設備的配置描述符是不一樣。比如,高速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?};

?

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

?

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?函數根據當前是否是高速設備,以及是否是?otg?設備,選擇合適的?configuration( souce sink config or loopback config)?,?調用usb_gadget_config_buf?生成最終的配置描述符。可以想象?usb_gadget_config_buf?的實現非常簡單?:?根據傳過來的?描述符列表?(?以?NULL?指針結束?)?,使用?memcpy??之類見每個描述符的內容拷貝到?buf?里。?下面我們繼續看???zero_setup?函數。

?

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

?

根據?host?傳遞過來的索引,響應相應的字符串。Zero?驅動的字符串描述符則只支持一種語言(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?};

?

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

?

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;

?

設置設備的當前配置,到這里,才凌空一腳,將設備帶入數據傳輸狀態,我們先把zero_setup?看完,再仔細看函數?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;

?

獲取設備的當前配置

?

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;

?

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

由于?zero_set_config?函數會調用?zero_reset_config,?所以這里應該可以不調用?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

?

獲取設備的當前配置的當前接口。

?

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

?

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

?

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??????????}

?

如果有數據需要傳給?Host,?則將其放到端點0?的傳送隊列。底層?udc?驅動會負責將其發給?host.

?

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

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

1084?}

?

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

?

函數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

?

函數?zero_reset_config?把所有的?端點置于?disable?狀態。

?

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??????????}

?

根據當前的配置,設置兩種不同的傳送方式。我們假定?host?設置的是?loopback?方式。另一種方式是類似的?(?數據內容不同?)?。

?

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?}

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

函數?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??????????}

?

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

?

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?)?,?等待主機向我們發送數據。等主機真正對我們進行?OUT?數據傳輸并且數據傳完后,會調用?loopback_complete?回調函數。

?

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?}

?

下面看?函數?loopback_complete

?

函數?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?傳輸正常結束,則會將其放到?IN?端點的傳輸隊列。

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

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

總結

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

?

參考

USB 2.0 Spec

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

?

用?KFI?和?Graphviz?跟蹤?/?優化內核代碼

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

總結

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

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。

在线中文字幕电影 | 国产精品久久久精品 | www.黄色片网站 | 中文字幕第 | 日韩美精品视频 | 91九色综合 | 国产福利91精品 | 久久试看| 久久婷婷影视 | 中午字幕在线 | 欧美精彩视频在线观看 | 国产在线1区 | 久久免费精品国产 | 99久久久国产免费 | 在线99 | 九九精品视频在线观看 | 五月天激情电影 | 黄色福利| 日韩欧美国产视频 | 欧美国产高清 | 中文伊人 | 欧美天天综合网 | 在线观看av免费 | 色橹橹欧美在线观看视频高清 | 国产精品一二三 | 黄色在线看网站 | 狠狠干夜夜操 | 免费看的黄色片 | 国产91欧美| aaa亚洲精品一二三区 | 亚洲在线网址 | 天天射天天操天天色 | 深爱开心激情 | 在线观看国产 | 精品成人国产 | 亚洲伊人天堂 | 中文字幕网站视频在线 | 天天天综合网 | 激情久久久 | 午夜精品久久久久久久久久久 | 天天射射天天 | 黄色一区二区在线观看 | 久久久精品欧美一区二区免费 | 青青河边草手机免费 | 中文国产在线观看 | 亚洲国产久 | 天天操天天干天天摸 | 国产日本在线播放 | 日韩精品久久久久久中文字幕8 | 欧美激情亚洲综合 | 日韩欧美不卡 | 亚洲专区欧美 | 久久免费精品 | 国产黄影院色大全免费 | 久久精品福利 | 亚洲欧洲一区二区在线观看 | 最近的中文字幕大全免费版 | 久久精品4 | 91免费观看视频网站 | 国产又粗又长的视频 | 在线免费观看羞羞视频 | 亚洲精品乱码久久久久久高潮 | 久久精品久久精品 | 西西人体4444www高清视频 | 视频一区二区免费 | 日本久久久久久久久 | 美女av免费看 | 久久国产精品色婷婷 | 色婷婷成人网 | 亚洲精品videossex少妇 | 2024av| 国产成人91| 久久这里 | 亚洲一区欧美精品 | 久久久精品影视 | 一区二区三区四区五区在线 | 国产97在线播放 | 久久高清免费观看 | 在线看成人| www.夜夜操.com | 国内一区二区视频 | 久久免费国产精品 | 中文字幕在线视频一区 | 亚洲第一av在线播放 | 伊人中文网 | 久久久亚洲精品 | 婷婷在线免费观看 | 日韩精品一区二区电影 | 国产伦理剧| 国产一卡在线 | 国产精品初高中精品久久 | 久草综合在线 | 亚洲成人av一区 | 99热.com| 国产成人l区| 亚洲视频axxx | 在线国产精品视频 | 在线观看av不卡 | 日韩草比 | 欧美日韩高清一区二区 国产亚洲免费看 | 欧美国产一区在线 | 国产精品亚州 | 久久免费国产精品 | 草久久影院 | 国产精品久久久精品 | 国内精品中文字幕 | 日韩大片免费在线观看 | 国产精品网站 | 午夜精品久久久久久久99 | 天天天天射| 激情在线五月天 | 五月天亚洲激情 | 日日夜夜天天久久 | 九九影视理伦片 | 国产涩涩网站 | 亚洲激情视频在线观看 | 99这里只有精品99 | 日日射av | 激情导航 | 黄色片网站 | 在线高清 | 中文理论片 | 国产免费专区 | 欧美成人中文字幕 | 激情五月婷婷综合网 | 久久呀 | 91九色最新 | 亚洲精品欧美专区 | 97综合网 | 香蕉视频在线看 | 在线国产小视频 | 综合网久久 | 伊人激情综合 | 日日夜夜爱 | 91精品视频网站 | 国产视频2区 | 超碰在线97观看 | 亚洲成人黄色av | 亚洲高清91 | 黄色最新网址 | 久久久久国产一区二区 | 国产99久久精品一区二区永久免费 | 亚洲高清av | 色精品视频 | 欧美日韩另类在线 | 免费男女羞羞的视频网站中文字幕 | 99久久精品日本一区二区免费 | 久草久草视频 | 午夜国产成人 | 伊人网站 | 91综合久久一区二区 | 亚洲日本va中文字幕 | 免费高清国产 | 一级片黄色片网站 | 我要色综合天天 | 最近免费在线观看 | 色婷婷丁香 | 国产又粗又猛又爽又黄的视频先 | 久久成人黄色 | 久久久久夜色 | 99久久精品国产一区二区成人 | 国内丰满少妇猛烈精品播放 | 亚洲精品乱码 | 国产黄免费在线观看 | 91精彩视频| 美女精品久久久 | 亚洲人成人天堂h久久 | 国产日韩欧美在线播放 | 精品黄色片| 日韩电影一区二区在线观看 | 日韩一级电影在线观看 | 亚洲成人一区 | 九九精品久久久 | 天天干.com | 国色天香在线 | 尤物一区二区三区 | 免费看污污视频的网站 | 亚洲精品美女久久 | 伊人资源站| 久久久久久久久久久国产精品 | 69久久99精品久久久久婷婷 | a级国产乱理伦片在线播放 久久久久国产精品一区 | 亚洲国产人午在线一二区 | 啪啪精品 | 91在线色 | 亚洲伊人色 | 国产在线高清视频 | 日本高清xxxx| 亚洲国产色一区 | 91视频观看免费 | www.夜夜操| 福利视频导航网址 | 18久久久 | 久草精品视频在线看网站免费 | 黄色av一级片 | 日本性生活一级片 | 日日干夜夜草 | 日韩美女黄色片 | 亚洲精品视频在线免费播放 | 亚洲一区在线看 | 91精品导航 | 麻豆va一区二区三区久久浪 | 超碰在线9| 久久久午夜视频 | 国产日韩在线播放 | 久久伊人精品一区二区三区 | aaa毛片视频 | 最近更新好看的中文字幕 | 久久免费观看少妇a级毛片 久久久久成人免费 | 丁香激情五月 | 天天干视频在线 | 婷婷香蕉 | 亚洲激情p | 欧美一级久久久久 | 欧洲视频一区 | 超碰97网站 | 国产精品网在线观看 | 久久久久国产精品一区二区 | 天天干天天上 | 日韩理论在线 | 日韩大片在线看 | 国产麻豆精品传媒av国产下载 | 五月天中文字幕mv在线 | 天天操天天谢 | 激情婷婷色 | 在线观看视频你懂 | 国产夫妻性生活自拍 | 久久av伊人 | 国产69熟| 欧美日韩高清一区二区 国产亚洲免费看 | 欧美一二区视频 | 高清国产午夜精品久久久久久 | 五月婷婷六月丁香 | 六月激情丁香 | 国产成人精品午夜在线播放 | 黄网站大全 | 成人免费网视频 | 看片一区二区三区 | 国产中文自拍 | 日韩伦理片一区二区三区 | 免费高清国产 | 亚洲成av人片在线观看 | 日韩色在线观看 | 欧美少妇xxx | 一级欧美一级日韩 | 在线视频一区二区 | 中文免费观看 | 91亚色视频在线观看 | 91九色蝌蚪视频网站 | 人人干在线观看 | 免费日韩一区二区三区 | 视频国产在线 | 一级c片| 99久久久久久久久 | www免费视频com| 久久艹艹 | 2024av | 国产欧美精品xxxx另类 | 久久一区国产 | 国产精品久久久影视 | 日韩在线视频不卡 | 亚洲激情视频在线观看 | 九九精品久久 | 国产成人av一区二区三区在线观看 | 日本系列中文字幕 | 91精品在线视频观看 | 亚洲精品国产综合99久久夜夜嗨 | 国产高清在线视频 | 中文字幕乱偷在线 | 五月天网页| 最近日本mv字幕免费观看 | 色a4yy| 99久久夜色精品国产亚洲 | 黄色亚洲 | 久久手机精品视频 | av一级在线 | 美女视频黄在线 | 国产老太婆免费交性大片 | 九色视频网址 | 丁香六月婷婷开心婷婷网 | 麻豆91视频| 欧美久久精品 | 黄色资源在线观看 | 色偷偷88888欧美精品久久 | 欧美福利视频一区 | 中文字幕日韩电影 | 香蕉免费在线 | 日韩av综合网站 | 亚洲精品一区二区在线观看 | 久操中文字幕在线观看 | 久久精品久久99 | 天天天综合网 | 成年人免费看片网站 | 亚洲最快最全在线视频 | 久草精品视频在线播放 | 久久免费黄色网址 | 日韩精品一区二区在线视频 | 国产亚洲成人精品 | 欧美激情精品久久久 | 西西444www大胆高清图片 | av免费观看网站 | 亚洲高清视频一区二区三区 | 亚洲精品网站在线 | 青青草国产成人99久久 | 韩国一区二区av | 国产亚洲精品久久久久久移动网络 | 日韩视频1区 | 在线高清av| 又爽又黄又刺激的视频 | 999久久久免费视频 午夜国产在线观看 | 免费av看片 | 成人在线黄色 | 人人澡人人爽欧一区 | 国产黄色精品在线 | 国产午夜影院 | 久久久免费电影 | 中文在线字幕观看电影 | 国内精品久久久久久久影视麻豆 | 久久a v电影 | 九九九热精品免费视频观看网站 | 五月婷婷激情综合 | 国产精品成人自产拍在线观看 | 视频国产精品 | 国产夫妻自拍av | av电影在线免费 | 成人a级黄色片 | 欧美激情精品久久久 | 黄色毛片视频免费 | 亚洲精品伦理在线 | 一区二区三区久久精品 | 国产资源在线观看 | 亚洲黄色app | 欧美在线观看视频一区二区三区 | 青青河边草免费直播 | 欧美极度另类 | 99精品国产在热久久 | 中文字幕一区二区三 | 亚洲精品乱码 | 97香蕉超级碰碰久久免费软件 | 国产五十路毛片 | 91av片| 日本大片免费观看在线 | 黄色日本片 | 日韩av电影免费观看 | 人人狠狠综合久久亚洲 | 亚洲国产成人精品在线观看 | 欧美a性 | 免费无遮挡动漫网站 | 欧美日韩一区三区 | 免费欧美精品 | 色综合婷婷 | 99久久精品免费看国产四区 | 精品一区二区综合 | 色搞搞| 成人国产网站 | 亚洲国产精品成人女人久久 | 国产精品久久久久久久久久东京 | 久久精品欧美一区二区三区麻豆 | 精品一区二区亚洲 | 色婷婷综合久久久中文字幕 | 欧美日韩久久不卡 | av免费网| 欧美男同视频网站 | 91资源在线播放 | 天天操天天色天天射 | 日日干美女 | 日韩免费久久 | 婷婷色亚洲| 免费黄a | 亚洲人成精品久久久久 | 日韩免费av片 | 丰满少妇一级片 | 五月天婷婷视频 | 久久99亚洲精品久久 | 久久国产精品99久久久久久丝袜 | 色老板在线视频 | 欧美最猛性xxxxx(亚洲精品) | av中文字幕在线播放 | 国产无遮挡又黄又爽在线观看 | av无限看 | 亚洲国产精品视频在线观看 | 91精品一区国产高清在线gif | 亚洲精品小区久久久久久 | 久久精品一区二区三 | 国产成人一区二区啪在线观看 | 91片网| 日本色小说视频 | 色综合久 | 九色自拍视频 | 一本一本久久a久久精品牛牛影视 | 欧美午夜性 | 在线观看黄网站 | 精品婷婷| 国产生活一级片 | 久久午夜鲁丝片 | 精品99久久久久久 | 亚洲天堂香蕉 | 婷婷国产一区二区三区 | 日韩激情小视频 | 欧美日韩在线免费视频 | 91女子私密保健养生少妇 | 欧美最猛性xxx | 国内精品久久久久国产 | 成人久久综合 | 日韩欧美在线视频一区二区三区 | 日韩专区在线观看 | 大片网站久久 | 九九热1| 日日夜夜天天射 | 91免费视频国产 | 九月婷婷人人澡人人添人人爽 | 成人99免费视频 | 日韩亚洲在线 | 91精选在线| 91人人澡人人爽 | 激情婷婷在线观看 | 国产精品久久久久久久久久久久午夜 | 精品国产精品国产偷麻豆 | 精品国产乱码久久久久久1区二区 | 日韩av在线免费看 | 亚洲国产精品一区二区久久hs | 综合色综合色 | 亚洲国产字幕 | 色综合久久久久综合体桃花网 | 午夜视频在线观看一区二区 | 91成人精品国产刺激国语对白 | 亚洲成av人片在线观看无 | 国产一级91 | 又黄又爽的免费高潮视频 | 午夜免费福利视频 | 久亚洲精品 | 成人免费观看在线视频 | 最新国产精品拍自在线播放 | 91视频电影| 日韩成人免费在线观看 | 久久久 精品| 人人天天夜夜 | 97av视频在线观看 | 久久综合色婷婷 | 欧美日韩中文国产一区发布 | 亚洲精品乱码久久久久久蜜桃91 | 精品久久久久亚洲 | 亚洲免费av一区二区 | 韩国一区二区在线观看 | 亚洲人成在线观看 | 国产精品96久久久久久吹潮 | 国产丝袜网站 | 日本一区二区三区免费看 | 国产精品一区二区精品视频免费看 | 亚洲人成精品久久久久 | 免费久久网站 | 99免费看片 | 午夜色婷婷 | 最新成人av| 亚洲精品国产精品国自产观看 | 午夜久久久久久久 | 天天插天天爱 | 日韩成人不卡 | 国产小视频在线看 | 亚洲国产电影在线观看 | .精品久久久麻豆国产精品 亚洲va欧美 | 美女国产 | 国产日本三级 | 超碰97国产在线 | 日韩av资源在线观看 | 亚洲一区久久久 | 欧美精品被 | 偷拍福利视频一区二区三区 | av中文字幕网址 | 丁香婷婷激情 | 婷婷综合亚洲 | 国产亚洲aⅴaaaaaa毛片 | 久久最新网址 | 日韩av影视在线观看 | 精品国产一区二区三区蜜臀 | 毛片a级片 | 国产精品久久久久久久午夜片 | 正在播放国产一区二区 | 一区二区三区日韩在线观看 | 国产免费午夜 | 一区免费视频 | 最新中文字幕在线观看视频 | 成人免费视频播放 | av电影在线不卡 | 精品在线观| 久久69精品久久久久久久电影好 | 五月婷婷六月丁香 | www.一区二区三区 | 九九在线视频 | 国产视频网站在线观看 | 国产永久免费高清在线观看视频 | 色综合久久中文综合久久牛 | 国产精品毛片一区视频播 | 91在线观看欧美日韩 | 日韩网站免费观看 | 国产亚洲精品免费 | 久久国产美女视频 | 国产视频一区在线免费观看 | 国产va精品免费观看 | 国产成人一区二区精品非洲 | 日本丰满少妇免费一区 | 久久久高清视频 | 开心激情网五月天 | 91人人澡 | 日韩在线观看视频在线 | 免费av黄色 | 日韩深夜在线观看 | 日韩欧美在线免费观看 | 国产在线精品一区二区三区 | 久久视奸 | 久久久国产一区二区三区四区小说 | 操操操影院 | 日韩欧美高清一区二区 | 91精品国自产在线 | 日韩免费在线观看 | 国内精品久久久久影院优 | 免费下载高清毛片 | 国产在线观看地址 | 日韩电影在线观看一区二区 | 免费观看一级特黄欧美大片 | 久久99国产精品久久 | 婷婷视频在线播放 | 欧美了一区在线观看 | 亚洲精品欧洲精品 | 黄色毛片一级 | 国产破处精品 | 一区二区亚洲精品 | 日本精品视频一区二区 | 最近中文字幕视频完整版 | 亚洲高清av在线 | 六月丁香激情综合 | 婷婷午夜激情 | 久久免费99 | 国产精品第7页 | 中文字幕日本特黄aa毛片 | 欧洲亚洲女同hd | 天天综合导航 | 97高清免费视频 | 国内视频在线 | 91丨九色丨国产女 | 欧美一区二区三区免费观看 | 日韩在线视频免费观看 | 免费观看的黄色片 | 黄影院| 国产精品毛片一区二区 | 在线观看色网站 | 国内精品99| 精品久久国产精品 | 激情av综合 | 国产精品aⅴ | 久久久五月天 | 97超碰人人澡 | 国产精品理论在线观看 | 99久久精品午夜一区二区小说 | 久久久久影视 | 久久69精品 | 这里只有精彩视频 | 香蕉影视app | 视色网站 | 少妇性bbb搡bbb爽爽爽欧美 | 特级西西人体444是什么意思 | 97碰在线| 欧美一级看片 | 日日摸日日添夜夜爽97 | 成人精品视频久久久久 | 欧美日韩国产亚洲乱码字幕 | 久草在线看片 | www.国产在线 | 欧美色图88 | 手机在线观看国产精品 | 欧美激情视频在线观看免费 | 日韩欧美在线综合网 | 九九综合在线 | 韩日视频在线 | 中文字幕精品www乱入免费视频 | 久久久国产精品视频 | 日韩一二三 | 国产精品欧美一区二区 | 国产精品九九九九九 | 亚洲日本韩国一区二区 | 国产精品ⅴa有声小说 | 五月黄色 | 免费视频一二三区 | 亚洲视频第一页 | 久草在线资源观看 | 最新免费中文字幕 | 日韩久久精品一区二区三区 | 亚洲精品福利在线 | 夜夜天天干 | 不卡的av电影 | 在线电影 你懂得 | 久久免费视频7 | 97超视频免费观看 | 精品免费观看视频 | 又粗又长又大又爽又黄少妇毛片 | 亚洲综合成人av | 日本最新高清不卡中文字幕 | 中文字幕日韩电影 | 99视频免费看 | 99精品久久精品一区二区 | 亚洲v精品 | 91麻豆精品国产91久久久无限制版 | 黄色app网站在线观看 | 日批视频在线观看免费 | 亚洲黄在线观看 | 欧美成人久久 | 国产高清中文字幕 | av网站手机在线观看 | 亚洲播放一区 | 精品在线观看一区二区 | 免费看黄在线观看 | 亚洲日本黄色 | 天天干夜夜干 | 亚洲精品看片 | 99精品一级欧美片免费播放 | 国产在线播放一区二区 | 天天se天天cao天天干 | 高清色免费| 久久人人爽视频 | 国产精品二区在线观看 | 国产成人福利在线 | 美女又爽又黄 | 日韩久久影院 | 狠狠操狠狠 | 丁香婷婷色综合亚洲电影 | 欧美色黄 | 中文字幕在线一二 | 99精品视频精品精品视频 | 91精品国产乱码久久桃 | 中文字幕 国产精品 | 一区二区三区视频在线 | 久久精品男人的天堂 | www.久草.com| 亚洲精品网站在线 | 97在线视频免费 | 久久视频热| 久久国产精彩视频 | 国产一级大片在线观看 | 日韩网| 久久综合狠狠综合 | a久久久久 | 久久精品小视频 | 最新在线你懂的 | 在线岛国av | 国产对白av | 久久99精品久久久久久秒播蜜臀 | 中文字幕文字幕一区二区 | 在线视频一二三 | 99视频在线免费播放 | www.黄色 | 久久久久黄 | 97操操 | 色鬼综合网 | 国产精品原创 | 欧美日韩一区二区三区在线免费观看 | 亚洲乱码中文字幕综合 | 最新免费av在线 | 免费观看福利视频 | 国产黄av | 亚洲最新视频在线播放 | 91在线精品观看 | 免费国产在线观看 | 国产成人三级三级三级97 | 欧美日韩一级在线 | 亚洲精品综合在线观看 | 精品免费视频123区 午夜久久成人 | 最新中文字幕视频 | 天天色天天干天天色 | 中文字幕免费观看 | 伊人www22综合色| 在线观看视频h | 日韩电影在线观看一区二区三区 | 久久免费的视频 | 国产成人区| 九九视频这里只有精品 | 亚洲国产精品久久久久 | 国产99久久99热这里精品5 | aaa日本高清在线播放免费观看 | 91在线最新 | 久久久久久久免费观看 | 九九精品视频在线观看 | av电影免费在线播放 | 日韩午夜一级片 | 最近2019中文免费高清视频观看www99 | 天天射天天拍 | 日韩a在线看 | 久久综合色影院 | 美女福利视频在线 | 欧美性精品 | 69av在线播放 | 国产精品自产拍在线观看 | 色婷婷视频在线观看 | 欧美激情xxxx性bbbb | 国产一区二区久久精品 | 国产午夜剧场 | 91xav| 久久免费精品国产 | 综合视频在线 | 国产精品久久 | 中文字幕高清有码 | 日韩电影在线观看一区二区三区 | 久久久久女人精品毛片 | 日本免费久久高清视频 | 国产在线视频在线观看 | 日韩精品在线免费观看 | 99久久精品免费看国产 | 成年人网站免费观看 | 99精品福利| 日韩av影视在线观看 | 日韩精品不卡在线观看 | 噜噜色官网 | 怡红院久久 | 日韩免费电影网站 | av中文字幕亚洲 | 国产在线观看污片 | 91精品影视 | 中文字幕二区三区 | 在线观看视频中文字幕 | 国产免费观看久久黄 | 亚洲精品国产拍在线 | 久久国产精品一区二区三区 | 午夜10000| 在线中文视频 | 99热在线观看免费 | 国产乱码精品一区二区蜜臀 | 欧美 日韩 成人 | 狠狠夜夜 | 综合久久久 | 亚洲久在线 | 夜色资源站国产www在线视频 | 91精品一区二区三区久久久久久 | 精品福利视频在线 | 欧美一级性生活 | 久草在线最新 | 中文字幕久久网 | 国产成人在线一区 | 欧美嫩草影院 | 视频在线亚洲 | www日| 国产午夜精品一区二区三区欧美 | 99精品国产aⅴ | 日韩在线国产精品 | 麻豆国产精品va在线观看不卡 | 最新精品国产 | 午夜精品久久一牛影视 | 国产一级大片在线观看 | 亚洲精品美女在线观看 | 久久久国产精品视频 | 黄色一级在线免费观看 | 日韩av片无码一区二区不卡电影 | 午夜久久视频 | 亚洲狠狠干 | 正在播放国产91 | 国产福利在线免费观看 | 亚洲japanese制服美女 | 五月色婷| 天天狠狠 | 激情婷婷丁香 | www.xxxx变态.com| 视频在线一区 | 亚洲日本韩国一区二区 | 午夜性生活 | 香蕉久草 | 91亚色视频在线观看 | 激情深爱五月 | 亚洲国产精久久久久久久 | 亚洲精品9 | 免费在线观看国产精品 | 国产成人久 | av片一区 | 激情视频国产 | 亚洲欧美日韩国产一区二区三区 | 国产视频亚洲精品 | 欧美日韩高清在线 | 欧美在线一二区 | 免费看三级黄色片 | 福利视频导航网址 | 91免费在线看片 | 亚洲黑丝少妇 | 亚洲国产精品影院 | 干 操 插 | av千婊在线免费观看 | 99久久国产免费看 | 狠狠干夜夜| 久久久亚洲国产精品麻豆综合天堂 | 亚洲午夜久久久久久久久电影网 | 日韩网站在线看片你懂的 | 处女av在线 | 激情婷婷丁香 | 久久久天堂 | 成人免费在线播放 | 国产电影一区二区三区四区 | 国产一区二区在线免费观看 | 欧美有色 | 亚洲欧美少妇 | 一区二区男女 | 胖bbbb搡bbbb擦bbbb| 色婷婷久久久综合中文字幕 | 午夜国产一区 | 久久精品视频观看 | 国产99久久精品 | 999国内精品永久免费视频 | 久久经典国产视频 | 天天操夜夜干 | 亚洲日本一区二区在线 | 亚洲三级av| 亚洲综合日韩在线 | 免费成人短视频 | 五月激情综合婷婷 | 高潮毛片无遮挡高清免费 | 五月婷婷六月丁香在线观看 | 久久久久久影视 | 在线观看一区 | 午夜视频一区二区 | 国产精品一区二区三区在线播放 | 最近中文字幕mv | 免费日韩高清 | 欧美伦理一区二区三区 | 天天爽天天射 | 操处女逼 | 色av婷婷 | 99热这里是精品 | 狠狠精品 | 美女免费黄网站 | 国产黄 | 国产成人精品一区在线 | 欧美成人69av| 福利二区视频 | 精品1区二区| 日韩精品一区二 | 免费观看久久久 | 免费激情在线电影 | 欧美热久久 | 亚洲另类xxxx | 999成人 | 国内外成人在线 | 国产精品一区二区三区在线看 | 日本在线精品视频 | 欧美大片www | 韩国视频一区二区三区 | 1024在线看片 | 99久久这里有精品 | 久久在线精品视频 | 国产亚洲精品久久网站 | 手机在线黄色网址 | 成人免费共享视频 | 欧美一区二区视频97 | av片一区二区 | 国产成人精品免费在线观看 | 97av视频在线观看 | 日日草视频 | 久草精品免费 | 午夜久久美女 | 人人爽人人澡 | 天天撸夜夜操 | 亚洲h在线播放在线观看h | 在线之家官网 | 亚洲成人av在线电影 | 伊人天堂久久 | www亚洲精品 | 久久久久久免费 | 久草电影免费在线观看 | 久久8| 亚洲第一av在线播放 | 97精品国产97久久久久久 | 97影视| 欧美日韩国产亚洲乱码字幕 | japanese黑人亚洲人4k | 成人免费xxxxxx视频 | 91免费高清| 久久久影视 | 日韩欧美一区视频 | 国产精品热视频 | 亚州视频在线 | 91禁在线观看 | 久久久久久久久久久成人 | 午夜国产一区二区三区四区 | 又紧又大又爽精品一区二区 | 九九九热视频 | 国产一区在线免费观看视频 | 精品国产伦一区二区三区 | 欧美日韩一级视频 | 超级碰碰免费视频 | 国产一区电影在线观看 | 欧美性一级观看 | 日黄网站 | 91精品国产成人观看 | 在线www色 | 久久久久久久久久久免费视频 | www.香蕉视频在线观看 | 久久久久久免费毛片精品 | 久久久久久麻豆 | 草久视频在线 | 色姑娘综合网 | 国产小视频网站 | 97综合网| 久久精品2 | 手机看片久久 | 久久久久久久毛片 | 久久理论视频 | 中文字幕在线观看一区 | 日韩免费福利 | 成人av动漫在线 | 特及黄色片 | 日韩av成人在线 | 中文字幕视频一区 | 久久国产精品免费看 | av大全免费在线观看 | 国产黄大片在线观看 | 97香蕉久久国产在线观看 | 国产一区精品在线 | 久久伦理 | 天天操福利视频 | 欧美激情h| 亚洲精品国产视频 | 成人蜜桃| 久久手机精品视频 | 亚洲春色综合另类校园电影 | 日本久久免费视频 | 九九免费精品视频在线观看 | 一级黄色片在线免费观看 | 亚洲精品在线免费播放 | 天天射天天操天天色 | 亚洲天堂在线观看完整版 | www久久精品 | 毛片永久新网址首页 | 国产护士hd高朝护士1 | 狠狠色伊人亚洲综合网站野外 | 国产在线一线 | 婷婷丁香狠狠爱 | 欧美一级片在线 | 中文字幕免费久久 | 国产亚洲高清视频 | 深夜免费福利网站 | 四虎免费在线观看视频 | www.狠狠插.com| 91久久国产露脸精品国产闺蜜 | 久久精品网站免费观看 | 日韩在线观看中文 | 高清av中文字幕 | 国产 日韩 在线 亚洲 字幕 中文 | 涩涩伊人| 国产精品国产三级国产aⅴ9色 | 国产美女被啪进深处喷白浆视频 | 99久久精品日本一区二区免费 | 国产网红在线观看 | 91精品夜夜| 国产精品国产精品 | 国产精品资源在线观看 | 国产91大片 | 日韩3区| 成片人卡1卡2卡3手机免费看 | 91pony九色丨交换 | 久久区二区 | 久久专区| av一区二区三区在线观看 | 免费黄色网止 | 日本在线视频一区二区三区 | 91精品国产自产在线观看永久 | 久久色在线观看 | 亚洲午夜精品久久久久久久久久久久 | 青青草在久久免费久久免费 | 国产一区二区三区在线免费观看 | 国产午夜三级一二三区 | 久久久久久久久久久网站 | 久久国产经典视频 | 国产又粗又猛又黄 | 欧美日韩国产精品一区 | 亚洲成熟女人毛片在线 | 亚洲在线a | 亚洲精品视频在线观看免费 | 精品国产乱码久久久久久天美 | 91麻豆精品国产自产 | 国产精品v欧美精品 | 国产精品久久久久久久久久 | 久久久观看 | 91在线播放国产 | 狠狠干干 | 狠狠操天天干 | 国产麻豆剧果冻传媒视频播放量 | 顶级欧美色妇4khd | 91亚洲免费 | 中文字幕日韩伦理 | 国产精品久久久久一区二区三区共 | 中文字幕精品视频 | 91tv国产成人福利 | 探花系列在线 | 好看av在线 | 免费精品国产va自在自线 | 美女网站在线观看 | 国产精品aⅴ| 婷婷激情综合 | 欧美日韩在线观看一区 | 欧美一二三视频 | 麻豆视屏 | 国产精品毛片一区二区在线看 | 日本久久久久久久久久久 | 91亚洲综合 |