1、Linux usb gadget 驱动利用 Linux USB gadget 设备驱动可以实现一些比较有意思的功能,举两个例子: 1、一个嵌入式产品中的某个存储设备,或是一个存储设备的某个分区,可以作为一个 U 盘被 PC;设别,从而非常方便的完成文件交互,这个功能被广泛的应用于手机、数码相机等产品中。2、一个嵌入式设备通过 USB 连接到你的 PC 后,在你的 PC 端会出现一个新的网络连接,在嵌入式设备上也会有一个网卡设备,你可以配置它们的 IP 地址,并进行网络通讯,俗称USBNET。所有 USB 通讯的设备端都有 usb device 程序,通常称它们为 usb 固件。在一些功能简单的
2、设备里,用一些专用的可编程 USB 控制器就可以了。而在一些运行了类似 linux 操作系统的复杂的嵌入式系统中,要完成 usb device 程序,就会要求你不仅熟悉 usb device 控制器的操作,还要熟悉操作系统的驱动架构。我想通过 “功能体验”、“驱动调试”、“gadget 驱动结构分析”、“编写一个自己的 gadget 驱动”这 4 个方面解析 linux usb gadget 设备驱动的编写方法。一、linux 模拟 U 盘功能的实现 在硬件环境为华清远见的 fs2410 平台,软件环境为 linux-2.6.26 的 linux 系统上,实现模拟 U 盘的功能。向内核添加代码
3、#include #include #include 修改 arch/arm/mach-s3c2410/mach-smdk2410.c/*USB device 上拉电阻处理 */static void smdk2410_udc_pullup(enum s3c2410_udc_cmd_e cmd)u8 *s3c2410_pullup_info = “ “,“Pull-up enable“,“Pull-up disable“,“UDC reset, in case of“;printk(“smdk2410_udc: %sn“,s3c2410_pullup_infocmd);s3c2410_gpio
4、_cfgpin(S3C2410_GPG9, S3C2410_GPG9_OUTP);switch (cmd)case S3C2410_UDC_P_ENABLE :s3c2410_gpio_setpin(S3C2410_GPG9, 1); /set gpg9 output HIGHbreak;case S3C2410_UDC_P_DISABLE :s3c2410_gpio_setpin(S3C2410_GPG9, 0); /set gpg9 output LOWbreak;case S3C2410_UDC_P_RESET :/FIXME!break;default:break;static str
5、uct s3c2410_udc_mach_info smdk2410_udc_cfg _initdata = .udc_command = smdk2410_udc_pullup,;static struct platform_device *smdk2410_devices _initdata = ,static void _init sdmk2410_init(void) u32 upll_value;set_s3c2410fb_info(s3c24xx_udc_set_platdata( /* 初始化*/s3c_device_sdi.dev.platform_data = /* Turn
6、 off suspend on both USB ports, and switch the* selectable USB port to USB device mode. */s3c2410_modify_misccr(S3C2410_MISCCR_USBHOST |S3C2410_MISCCR_USBSUSPND0 |S3C2410_MISCCR_USBSUSPND1, 0x0);/* 设置 USB 时钟 */upll_value = (0x78 USB Gadget Support -USB Peripheral Controller (S3C2410 USB Device Contr
7、oller) -S3C2410 USB Device Controller* S3C2410 udc debug messagesUSB Gadget Drivers File-backed Storage Gadget3、编译内核#make zImage#make modules在目录 drivers/usb/gadget 下生成 g_file_storage.ko加载驱动,测试功能利用前面的生成的内核,启动系统后,加载 g_file_storage.ko#insmod g_file_storage.ko# insmod g_file_storage.ko file=/dev/mtdbloc
8、k2 stall=0 removable=10.03 USB: usb_gadget_register_driver() g_file_storage0.04 USB: binding gadget driver g_file_storage0.05 USB: s3c2410_set_selfpowered()g_file_storage gadget: File-backed Storage Gadget, version: 20 October 2004g_file_storage gadget: Number of LUNs=1g_file_storage gadget-lun0: ro
9、=0, file: /dev/mtdblock30.06 USB: udc_enable calledsmdk2410_udc: Pull-up enable连接设备到 windows,windows 系统会自动设备到一个新的 U 盘加入。格式化 U 盘,存入文件。卸载 U 盘后,在目标板上执行如下操作:# mkdir /mnt/gadget# mount -t vfat /dev/mtdblock2 /mnt/gadget/ #ls可以看到 windows 存入 U 盘的文件。 二、usbnet 功能的实现 配置内核支持 usbnet USB Gadget Support -USB Peri
10、pheral Controller (S3C2410 USB Device Controller) -S3C2410 USB Device Controller* S3C2410 udc debug messagesUSB Gadget DriversEthernet Gadget (with CDC Ethernet support)* RNDIS support2、编译内核#make zImage#make modules在目录 drivers/usb/gadget 下生成 g_ether.ko3、加载驱动,测试功能利用前面的生成的内核,启动系统后,加载 g_ether.ko#insmod
11、 g_ether.ko#ifconfig usb0 192.168.1.120usb0 Link encap:Ethernet HWaddr 5E:C5:F6:D4:2B:91inet addr:192.168.1.120 Bcast:192.168.1.255 Mask:255.255.255.0UP BROADCAST RUNNING MULTICAST MTU:1500 Metric:1RX packets:253 errors:0 dropped:0 overruns:0 frame:0TX packets:43 errors:0 dropped:0 overruns:0 carrie
12、r:0collisions:0 txqueuelen:1000RX bytes:35277 (34.4 KiB) TX bytes:10152 (9.9 KiB) 连接设备到 windows,windows 系统会提示安装驱动,根据提示安装上 RNDIS 驱动。这个驱动可以在网络上找到。此时 windows 会新生成一个网络连接,配置它的 ip 地址等信息。然后就可以和目标系统通过 USB 实现网络通讯了这一节主要把在实现“linux 模拟 U 盘功能”过程中的一些调试过程记录下来,并加以解析。一、背景知识 1、USB Mass Storage 类规范概述 USB 组织在 universal
13、Serial Bus Mass Storage Class Spaceification 1.1 版本中定义了海量存储设备类(Mass Storage Class)的规范,这个类规范包括四个 独立的子类规范,即: 1. USB Mass Storage Class Control/Bulk/Interrupt (CBI) Transport2.USB Mass Storage Class Bulk-Only Transport3.USB Mass Storage Class ATA Command Block 4.USB Mass Storage Class UFI Command Speci
14、fication 前两个子规范定义了数据/命令/状态在 USB 上的传输方法。Bulk- Only 传输规范仅仅使用 Bulk 端点传送数据/命令/状态,CBI 传输规范则使用 Control/Bulk/Interrupt 三种类型的端点进行数据/命令/状态传送。后两个子规范则定义了存储介质的操作命令。ATA 命令规范用于硬盘,UFI 命令规范是针对 USB 移动存储。 Microsoft Windows 中提供对 Mass Storage 协议的支持,因此 USB 移动设备只需要遵循 Mass Storage 协议来组织数据和处理命令,即可实现与 PC 机交换数据。而 Flash 的存储单元
15、组织形式采用 FAT16 文件系统,这样,就可以直接在 Windows 的浏览器中通过可移动磁盘来交换数据了,Windows 负责对 FAT16 文件系统的管理,USB 设备不需要干预FAT16 文件系统操作的具体细节。 USB(Host)唯一通过描述符了解设备的有关信息,根据这些信息,建立起通信,在这 些描述符中,规定了设备所使用的协议、端点情况等。因此,正确地提供描述符,是USB 设备正常工作的先决条件。 Linux-2.6.26 内核中在利用 USB gadget 驱动实现模拟 U 盘时主要涉及到 file_storage.c、s3c2410_udc.c 等驱动文件(这些文件的具体结构,
16、将在下一篇文章中描述)。此时我们想先从这些代码中找到 USB 描述描述符,从中确定使用的存储类规范,从而确定协议。确定通讯协议是我们调试的基础。 存储类规范是由接口描述符决定的。接口描述符各项的定义义如下:其中,bInterfaceClass、bInterfaceSubClass、bInterfaceProtocol 可以判断出设备是否是存储类,以及属于哪种存储子类和存储介质的操作命令。在 file_storage.c 文件中,/* USB protocol value = the transport method */#define USB_PR_CBI 0x00 / Control/Bul
17、k/Interrupt#define USB_PR_CB 0x01 / Control/Bulk w/o interrupt#define USB_PR_BULK 0x50 / Bulk-only/* USB subclass value = the protocol encapsulation */#define USB_SC_RBC 0x01 / Reduced Block Commands (flash)#define USB_SC_8020 0x02 / SFF-8020i, MMC-2, ATAPI (CD-ROM)#define USB_SC_QIC 0x03 / QIC-157
18、(tape)#define USB_SC_UFI 0x04 / UFI (floppy)#define USB_SC_8070 0x05 / SFF-8070i (removable)#define USB_SC_SCSI 0x06 / Transparent SCSI默认的情况是:mod_data = / Default values.transport_parm = “BBB“,.protocol_parm = “SCSI“,默认的赋值如下:bInterfaceClass=08 表示:存储类bInterfaceSubClass=0x06 表示:透明的 SCSI 指令bInterfacePr
19、otocol=0x50 表示:bulk-only 传输2、BulkOnly 传输协议 下面看看 BulkOnly 传输协议:(详细的规范请阅读Universal Serial BusMass Storage ClassBulk-Only Transport) 设备插入到 USB 后,USB 即对设备进行搜索,并要求设备提供相应的描述符。在USBHost 得到上述描述符后,即完成了设备的配置,识别出为 BulkOnly 的 Mass Storage 设备, 然后即进入 BulkOnly 传输方式。在此方式下,USB 与设备间的所有数据均通过 BulkIn 和 BulkOut 来进行传输,不再通过
20、控制端点传输任何数据。 在这种传输方式下,有三种类型的数据在 USB 和设备之间传送,CBW、CSW 和普通数据。CBW(Command Block Wrapper,即命令块包)是从 USB Host 发送到设备的命令, 命令格式遵从接口中的 bInterfaceSubClass 所指定的命令块,这里为 SCSI 传输命令集。USB 设备需要将 SCSI 命令从 CBW 中提取出来,执行相应的命令,完成以后,向 Host 发出反映 当前命令执行状态的 CSW(Command Status Wrapper),Host 根据 CSW 来决定是否继续发 送下一个 CBW 或是数据。Host 要求 USB 设备执行的命令可能为发送数据,则此时需要将 特定数据传送出去,完毕后发出 CSW,以使 Host 进行下一步的操作。USB 设备所执行的操作可用下图描述:CBW 的格式如下: