USB的描述符详解总结

上传人:su****e 文档编号:201977374 上传时间:2023-04-21 格式:DOC 页数:24 大小:297KB
收藏 版权申诉 举报 下载
USB的描述符详解总结_第1页
第1页 / 共24页
USB的描述符详解总结_第2页
第2页 / 共24页
USB的描述符详解总结_第3页
第3页 / 共24页
资源描述:

《USB的描述符详解总结》由会员分享,可在线阅读,更多相关《USB的描述符详解总结(24页珍藏版)》请在装配图网上搜索。

1、USB的描述符与命令请求详解一、 描述符1. 什么是描述符所谓描述符,就是用于描述设备特性的具有特定格式排列的一种数据组织结构。2. 描述符的作用描述符的作用在于设备向主机汇报自己的信息、特征,主机根据这些信息从而加载相应的驱动程序。3. 描述符的分类描述符分为三大类:标准描述符、设备类描述符、厂商描述符。除字符串描述符可选外,任何设备都必须包含剩下的几种标准描述符。在USB1.0中规定了5种标准的描述符:设备描述符配置描述符接口描述符端点描述符字符串描述符规定的设备类描述符有:集线器类描述符、人机接口类描述符。下表是三种描述符的类型值:表1 . USB描述符的类型值类型描述符描述符值标准描述

2、符设备描述符(Device Descriptor)0x01配置描述符(Configuration Descriptor)0x02字符串描述符(String Descriptor)0x03接口描述符(Interface Descriptor)0x04端点描述符(EndPontEndPoint Descriptor)0x05(Device Qualifier descriptor)0x06(BOS descriptor)0x0F(Device Capability descriptor)0x10端点伴随描述符(Endpoint companion descriptor)0x30类描述符集线器类描述

3、符(Hub Descriptor)0x29人机接口类描述符(HID)0x21厂商定义的描述符0xFF4. 使用的几种类设备类DeviceClass下表是设备类值的含义。表2. 设备的类别(bDeviceClass)值(十进制)值(十六进制)说明00x00使用接口描述符中提供的类20x02通信类(CDC)90x09集线器类2200xDC用于诊断用途的设备类2240xFE0xEF混杂类型设备类2550xFF厂商定义的设备类接口类InterfaceClass下表是接口类值的含义。表3. USB协议定义的接口类别(bInterfaceClass)值(十六进制)类别0x01音频类0x02通信类(CDC)

4、0x03人机接口类(HID)0x05物理类0x06图像类0x07打印机类0x08大数据存储类0x09集线器类0x0ACDC数据类0x0B智能卡类0x0D安全类0x0EVideo 视频设备(摄像头,Class_0e&SubClass_03&Prot_00)0xDC诊断设备类0xE0无线控制器类0xEF混杂设备类0xFE特定应用类(包括红外的桥接器等)0xFF厂商定义的设备 类的交叉与独享在描述符中,只有设备描述符和接口描述符中会有类别之分,即只有设备和接口会分类使用,不过有些类别的使用只需经过设备或接口的区分就可彻底清楚明白,这说明在设备类别和接口类别的定义上会有共同的类别名称。而有些类别则是设

5、备或接口独享的,下表是与使用设备相关的类别划分交叉或共享情况:Base Class Usage Description 00h Device Use class information in the Interface Descriptors 01h Interface Audio 02h Both Communications and CDC Control 03h Interface HID (Human Interface Device) 05h Interface Physical 06h Interface Image 07h Interface Printer 08h Interf

6、ace Mass Storage 09h Device Hub 0Ah Interface CDC-Data 0Bh Interface Smart Card 0Dh Interface Content Security 0Eh Interface Video 0Fh Interface Personal Healthcare 10h Interface Audio/Video Devices DCh Both Diagnostic Device E0h Interface Wireless Controller EFh Both Miscellaneous FEh Interface App

7、lication Specific FFh Both Vendor Specific (此表也适用于标准命令Get_Descriptor中wValue域高字节的取值含义) 【说明:】在设备或接口分类上均可彻底分清使用的(Usage = Both),即在任一处描述符中定义即可的分清楚使用的类(Usage = Both)的基本类有:02h - 通信及CDC控制类;DCh - 诊断设备类;EFh - 混杂设备类;FFh - 厂商定义的设备类。5. 标准描述符设备描述符表4、USB设备描述符的结构偏移域Bytes值描述0bLength1数字此描述符的字节数1bDecriptorType1常量描述符的类

8、型(此处应为0x01,即设备描述符)2bcdUSB2BCD码USB版本号(BCD 码)4bDeviceClass1设备类设备类码:bDeviceClass = 0 ,表明设备类型使用接口描述符中定义的类型,且各个接口独立工作。bDeviceClass = FFh,表明设备类是由厂商自定义的。bDeviceClass = 1FEh,查表可得对应设备类值,该设备在不同的接口上支持不同的类。且这些接口可能不能独立工作。此值指出了这些接口集体的类定义。5bDeviceSubClass1设备子类设备子类码: 这些码值的具体含义根据bDeviceClass 域来看。 如bDeviceClass 域为零,此

9、域也须为零 如bDeviceClass 域为FFH,此域的所有值保留。6bDevicePortocol1设备协议协议码 这些码的值视bDeviceClass 和 bDeviceSubClass 的值而定。 如果设备支持设备类相关的协议,此码标志了设备类的值。如果此域的值为零,则此设备不支持设备类相关的协议,然而,可能它的接口支持设备类相关的协议。如果此域的值为FFH,此设备使用厂商定义的协议。7bMaxPacketSize01数字端点0的最大包大小(仅8,16,32,64为合法值)8idVendor2ID厂商标志(由USB-IF组织赋值)10idProduct2ID产品标志(由厂商赋值)12b

10、cdDevice2BCD 码设备版本号(BCD 码)14iManufacturer1索引描述厂商信息的字符串描述符的索引值。15iProduct1索引描述产品信息的字串描述符的索引值。16iSerialNumber1索引描述设备序列号信息的字串描述符的索引值。17bNumConfigurations1数字可能的配置描述符数目【说明1:】当设备类型bDeviceClass = 0时,说明类型将由接口描述符中定义的为准。【说明2:】从设备描述符表格中可知,有3个索引值:厂商信息索引、产品信息索引、设备序列号索引,这意味着,将有3个字符串描述符为其准备。配置描述符配置描述符中包含了配置描述符本身的长

11、度、所有配置信息的总长度、供电方式及远程唤醒、供电量。 如果主机发出标准命令Get_Descriptor要求获得设备的某个配置描述符时,该配置应用的所有信息都将发给主机,它包括:该标准配置符本身、该配置所包含的所有接口、端点描述符及设备类描述符和厂商描述符。 下表为配置描述符结构:表8、USB配置描述符的结构 偏移量 域大小 值 描述 0bLength1 数字此描述表的字节数长度。 1bDescriptorType1 常量配置描述表类型(此处为0x02) 2wTotalLength2 数字此配置信息的总长(包括配置,接口,端点和设备类及厂商定义的描述符),即:将要返回的配置信息总长度。 4bN

12、umInterfaces1 数字此配置所支持的接口个数 5bCongfigurationValue1 数字在SetConfiguration()请求中用作参数来选定此配置。 6iConfiguration1 索引描述此配置的字串描述符的索引 7bmAttributes1 位图配置特性: D7: 保留(设为1) D6: 自给电源 D5: 远程唤醒 D4.0:保留(设为1) 一个既用总线电源又有自给电源的设备会在MaxPower域指出需要从总线取的电量。并设置D6为1。运行时期的实际电源模式可由GetStatus(DEVICE) 请求得到。 8MaxPower1 mA在此配置下的总线电源耗费量。以

13、 2mA 为一个单位。 【说明1:】配置描述符也包含了个用于描述符该配置的字符串描述符索引iConfiguration,这说明将有个字符串描述符为其准备。 【说明2:】枚举的过程可分为4个状态阶段:接入状态阶段、缺省状态阶段、地址状态阶段、设置状态阶段,各状态阶段任务如下:接入状态阶段-主机检测到新设备接入后,将复位总线(释放总线于空闲状态)。缺省状态阶段-主机利用0x00地址访问新接入的设备,读取部分描述符后,会分配个设备地址。地址状态阶段-主机再次复位总线,然后用新分配的地址获取设备所有的描述符。设置状态阶段-主机根据设备的描述符,会对设备作些相关的配置。 【说明3:】bCongfigur

14、ationValue-USB设备的配置值。用于存放主机执行SetConfiguration命令的设置值。当主机发送GetConfiguration命令时,设备将向主机返回1个字节的配置值。然而,USB设备处于不同的状态时,对GetConfigration的请求也有不同的响应:1. 在枚举阶段,若设备处于地址状态时,对GetConfigration的请求返回为0;2. 在枚举阶段,若设备处于默认状态(缺省状态)时,对GetConfigration的请求视为无效;3. 在枚举阶段,若设备处于配置状态时,对GetConfigration的请求将返回bConfigurationValue字段的值(该值

15、可能是配置描述符的默认值,也可能是USB主机的设置值,这要看在执行GetConfigration命令前是否执行了SetConfigration命令)。因为主机要执行SetConfigration命令,所以bCongfigurationValue的默认值没什么用。实际上主机给bCongfigurationValue赋值后,bCongfigurationValue值就充当配置描述符的编号,用以区分不同的配置,因为一个设备可能有多个配置。接口描述符USB设备的接口,并不指物理接口,更确切的说应该是“功能接口“,是个赋予特定功能逻辑概念,是由一组物理端点为实现这一特定功能而凝聚的集合。/定义标准的接口

16、描述符结构typedef struct _INTERFACE_DESCRIPTOR_STRUCT BYTE bLength; /接口描述符的字节数大小 BYTE bDescriptorType; /接口描述符的类型编号 BYTE bInterfaceNumber; /接口的编号 BYTE bAlternateSetting; /可替换的接口描述符编号。实际就是接口的描述符的编号。 BYTE bNumEndpoints; /该接口使用的端点数,不包括端点0 BYTE bInterfaceClass; /接口类 BYTE bInterfaceSubClass; /接口子类 BYTE bInterf

17、aceProtocol; /接口遵循的协议 BYTE iInterface; /描述该接口的字符串索引值 INTERFACE_DESCRIPTOR_STRUCT, * pINTERFACE_DESCRIPTOR_STRUCT;【说明1:】接口描述符中用到接口编号bInterfaceNumber,以区分在同一配置下的不同的接口。同时还有该接口描述符的索引iInterface,这意味着将为其准备准备一个字符串描述符。 【说明2:】接口描述符中有一项:可替换的接口描述符编号bAlternateSetting,表示对某一接口进行描述的描述符编号。虽然,USB设备的配置与配置描述符是一一对应的,即一个配

18、置只能由一个配置描述来描述它,但一个接口却允许有多种描述符来描述它,尽管接口描述符的编号还是唯一一个。说白了就是:一个接口有唯一的一个接口编号,但一个接口却可以有多个不同的描述符编号,而这些不同的接口描述符的编号值就是bAlternateSetting。所以,通过bInterfaceNumber可以选定一个唯一的接口,然后再通过bAlternateSetting选择想要的对该接口的描述。主机通过GetInterface可以获取当前正在使用的接口及接口描述,通过SetInerface可以选定某接口及其使用的描述符。端点描述符端点是设备与主机之间进行数据传输的逻辑接口,除配置使用的端点0(控制端点

19、,一般一个设备只有一个控制端点)为双向端口外,其它均为单向。端点描述符描述了数据的传输类型、传输方向、数据包大小和端点号(也可称为端点地址)等。每个设备必须要有一个默认的控制型端点,地址为0,它的数据传输为双向,而且没有专门的描述符,只是在设备描述符中定义了它的最大包长度。主机通过此端点向设备发送命令,获得设备的各种描述符的信息,并通过它来配置设备。/定义标准的端点描述符结构typedef struct _ENDPOINT_DESCRIPTOR_STRUCT BYTE bLegth; /端点描述符字节数大小 BYTE bDescriptorType; /端点描述符类型编号 BYTE bEndp

20、ointAddress; /端点地址及输入输出属性 BYTE bmAttributes; /端点的传输类型属性 WORD wMaxPacketSize; /端点收、发的最大包大小 BYTE bInterval; /对周期性端点的访问间隔 ENDPOINT_DESCRIPTOR_STRUCT, * pENDPOINT_DESCRIPTOR_STRUCT;【说明1:】端点的传输类型字节bmAttributes,描述了该端点的传输特性:01bit定义了传输类型-00=控制传输、01=同步传输、10=批量传输、11=中断传输。【说明2:】周期端点的访问周期字节bInterval,定义了该端点被主机的访

21、问周期,此域值对于批量传输和控制传输毫无意义。对于同步传输,其值必须为1,即1ms为标准的同步帧周期。对于中断传输,该值为1255,即1ms255ms。字符串描述符字符串描述符是一种可选的USB标准描述符,描述了如制商、设备名称或序列号等信息。如果一个设备无字符串描述符,则其它描述符中与字符串有关的索引值都必须为0。字符串使用的是Unicode编码。字符串描述符是用字符的形式描述设备、配置、接口、端点等信息。字符串描述符以一种格式2类符值的方式存在:1. 显示语言的字符串描述符-该字符串描述符表明了设备支持哪几种语言。2. 显示信息的字符串描述符-用于描述具体的信息。标准的字符串描述符的格式为

22、:表9. 字符串描述符偏移量 域 大小 值 描述 0bLength1数字此描述表的字节数(bString域的数值N2)1bDescriptorType1常量描述符类型(此处应为0x03)2NStringsN数字字符串显示语言的字符串描述符与显示信息的字符串描述符的区别在于Strings项的不同,对于显示语言的字符串描述符来说Strings项由多个wLANGIDn数组元素组成,每个wLANGIDn是一个双字节的代表语言的ID值。而对于显示信息的字符串描述符而言,Strings则是描述信息后的一组UNICODE编码。为什么会出现这两种情况,原因在于访问字符串描述符的过程,主机请求访问某个字符串描述

23、符的步骤分成两步:第一步:获取语言信息-首先主机向设备发送标准请求命令Get_Descriptor,其参数为:描述符类型=字符串描述符,字符串的索引值=0,语言=0,这样设备将返回显示语言的字符串描述符,从而主机知道了设备能支持哪些语言。第二步:主机根据自已需要的语言,再次向设备发出标准请求命令Get_Descriptor,其参数为:描述符类型=字符串描述符,字符串索引值=目标字符串索引值,语言=目标语言。这次设备将返回目标已经明确的显示信息的字符串描述符。【说明1:】只有字符串描述符的长度不是固定的,其长度为N+2,其中N代表Strings项的字节数,2代表字符串描述符的bLength、bD

24、escritorType所占的两个字节。设备类描述符之HID描述符在USB协议中,HID设备的描述符没有划作为标准的描述符,而是作为一类设备单独划分出来进行描述,以设备类的方式来描述它。所以,描述它的格式用设备类描述符。 HID设备的信息在设备描述符和配置描述符中都不包含,而是包含在接口描述符中,所以在使用HID设备时,其设备描述符中的相关项应定义如下:bDeviceClass=0;bDeviceSubClass=0;bDeviceProtocol=0;其接口描述符应该:bInterfaceClass=0x03另外,对无引导的HID设备,其接口描述符中子类代码bInterfaceSubClas

25、s应置0,此时bInterfaceProtocol无效,置零即可。即为:bInterfaceClass=0x03 bInterfaceSubClass=0 bInterfaceProtocol=0对支持引导的USB设备,其接口描述符中子类代码bInterfaceSubClass应置1,此时bInterfaceProtocol可以为1或2,1表示键盘接口,3表示鼠标接口。其参考设置如下:bInterfaceClass=0x03 bInterfaceSubClass=1 bInterfaceProtocol=1或2下面是HID设备类描述符:【说明1:】HID设备类描述符并不是说仅用这一个描述符就可

26、描述清楚这类设备,而是指HID设备除包含所有的标准描述符外,还需这个HID设备来补充描述。也就是说,在使用一般的设备时,只需使用标准的描述符就可描述清楚,而若使用HID设备时,除了要使用全部的标准的描述符外还需HID描述符来补充描述。同时,从HID描述符中看出,它还将引出HID的报告描述符,在此不讲述。可以这么说,设备类描述符是作为一个对标准描述进行补充描述的描述符。6. 描述符的编号及索引1. 一个USB设备只能拥有一个设备描述符,故设备描述符不需要编号。但设备描述符通常会提供设备最基本的文字描述信息,通常包含厂商、设备、产品的信息,故它拥有3个字符串描述符的索引,这3个索引将指向3个字符串

27、描述,分别描述厂商信息、产品信息、设备序列号信息。简言之,设备描述符指示了设备有几种配置,及厂商、产品、设备序列号的字符串描述符索引。2. 配置描述符提供了相应的配置参数和查找参数:配置描述符编号bCongfigurationValue、配置描述符的字符串描述符的索引。3. 接口描述提供了该接口的应用参数和查找参数:接口编号bInterfaceNumber、接口描述符编号bAlternateSetting、该接口描述符对应的字符串描述符的索引。4. 字符串描述符是对各描述符所需的字符信息描述的实现,每个描述符所需的字符信息描述的索引都将对应一个字符串描述符。但通常都不那么做,而是把所有的字符描

28、述的实现都写在一个总的字符串描述符中,即字符串描述符的bStrings项,它们之间用索引来区分。7. 描述符的获取获取描述符的命令格式命令码CmdCode = GetDescriptor , 格式如下:bmRequestTypebRequestwValuewIndexwLength0x800x60类型和索引0或语言ID描述符长度wValue-其高字节wValue_H指明要获取的描述符类型(实际只有3种类型:设备描述符类型、配置描述符类型、字符串描述符类型),低字节wValue_L指明目标描述符的索引,然而wValue_L的值只对配置描述符和字符串描述符有效,而对设备描述符无效。wIndex-只

29、对字符串描述符有意义,对其它描述符时该值为0.。当然对于字符串描述符时,其值也可为0,表示要获取“显示语言的字符串描述符” ,若为其它值则代表了确定的语言ID,即表明要获取指定了语言的“显示信息的字符串描述符” 。wLength-主机要求的返回的描述符长度。如果wLength大于实际的描述符长度,则以实际描述符长度为准;如果wLength小于实际描述符长度,则以wLength值为准。获取描述符的过程获取描述符属于枚举的过程,其整个过程当然必经Setup传输的3大过程:Setup过程、数据过程、状态信息过程。首先,在Setup过程中,主机发送GetDescriptor命令。若成功,设备就开始准备

30、数据,通信将继续向前推进,进入数据过程。然后,在数据过程中,主机启动IN事件,设备就把准备好的数据(描述符)发送出去。若成功,则通信继续向前推进,进入状态信息过程。最后,在状态信息过程,主机发送通信过程的信息状态,祝贺并告知通信完美结束。获取配置描述符对于主机来说,配置是广义的,包括狭义的配置、接口配置、端点配置等,而接口配置、端点配置等都隶属于标准配置描述符,故主机若要求获取配置描述符时,实际上是要求获取除设备描述符和字符串描述符以外的所有描述符。对于只有标准描述符的设备而言,当主机要求或者配置描述符时,需设备按照顺序把标准配置描述符、标准接口描述符、标准端点描述符一次性发给主机。所以,通常

31、在写程序时,会将广义上的“配置”打成一个包,在包中,由标准配置描述符引领,按照发送顺序依次实现标准接口描述符、标准端点描述符等。这样做的理由是,在标准配置描述符中有一项wTotalLength,它代表广义上的配置包描述符总长度,根据这个参数就可把广义的配置包描述符一起发给主机,以避免多个描述符时的多次传输。bmRequestTypebRequestwValuewIndexwLength0x800x60类型和索引0或语言ID描述符长度 wValue _H = 配置描述符类型。wValue _L = 配置描述符编号(索引),实际为bCongfigurationValue值。wIndex = 0 。

32、wLength,其值由主机自己规定。因为,是按确定的顺序发送的,故主机解析的结果也将一一对应。下面是一个广义配置包描述符的结构模板: uint8_t USB_ConfigDescriptor = 标准配置描述符的实现;标准接口描述符的实现;标准设备类描述符的实现;标准端点描述符的实现;获取字符串描述符从设备描述符到端点描述符,需要许多的信息描述,即需要许多字符串描述符来描述它们的信息。然而,标准字符串中没有总长度显示项wTotalLength,且每个字符串描述符的格式都一样,所以不可能向获取配置描述符那样,用广义的配置包描述符一起发给主机,况且有些字符串描述符不是必须的,所以很难做到统一的格式

33、。不过,为了方便管理,在编程时通常还是把所有的字符串描述符组织在一起,不过主机在访问它们时只能一个一个的访问,而不能打包访问,它们之间的选取是依赖各个字符串描述符的长度进行跳过操作来实现的。所以这种组织在一起,只是为了方便管理或好看,而没有其它任何作用,组织的形式通常以“显示语言的字符串描述符”领头,模板如下:uint8_t USB_StringDescriptor = 显示语言的标准字符串描述符;显示信息的标准字符串描述符1 ; 显示信息的标准字符串描述符n ;;在字符串描述符组织中,各个字符串是怎么区分的?是应用程序,因为这个组织是编写应用程序时自己规定内部秩序的,故组织中各个字符串描述符

34、对应的索引,程序员当然知道。bmRequestTypebRequestwValuewIndexwLength0x800x60类型和索引0或语言ID描述符长度 wValue _H = 字符串描述符类型。wValue _L = 字符串描述符对应的编号(索引)。wIndex = 0 或ID。wLength,其值由主机自己规定。在获取字符串描述符的第一步:获取设备所支持的语言中,wValue _L = 0,wIndex = 0 ,设备将把显示语言的标准字符串描述符发给主机,主机会从中挑选一种语言。在获取字符串描述符的第二步:获取显示信息的字符串描述符中,wValue _L = 目标字符串对应的编号,w

35、Index=语言ID,设备则把确定的字符串描述符发给主机。二、 标准命令usb协议分析-设备描述符配置包-描述符/* usb协议分析仅供大家参考-设备描述符配置包,设备描述符, 地址设置, 配置描述符, 字符串描述符 */* -1- usb设备描述符配置包 */typedef struct _USB_SETUP_PACKET REQUEST_TYPE bmRequestType; BYTE bRequest; WORD_BYTE wValue; WORD_BYTE wIndex; WORD wLength; USB_SETUP_PACKET;1.bmRequestType 是包含有下面几方面的

36、内容:D7 D6 D5 D4 D3 D2 D1 D0在这一个字节里,又按位分为:D7位是表示后面传送数据的方向位。当D7等于0时,表示后面的数据是从主控器发送到USB设备。在PC里,就是从PC机发送到USB的设备。当D7等于1时,表示后面的数据是从USB设备发送到主控器。在PC里,就是从USB设备发送到USB设备。在上次里,我收到并显示出来的数据是80,就表示从USB设备里发送数据给PC。在这里再次给出上一次的包数据:80 06 00 01 00 00 40 00这里的80,就是D7位为1。D6-D5位是请求主分类型0 是表示标准的请求。1 是表示类别的请求。2 是表示厂商的请求。3 是保留。

37、D4-D0位是表求接收这个包的接口。0 是表示USB设备接收。1 是表示接口接收。2 是表示端点接收。3 是表示其它接收,不知道的。4-31是保留。2.bRequest 是本描述符的请求类型,也就是后面发送的数据是什么样的东西。由于USB里有很多配置信息,比如获取设备描述符,又有设置USB地址等等,就是通过这个字节来区分的。从USB协议里查找表9-4,就可看到如下的编码:GET_STATUS 0CLEAR_FEATURE 1Reserved for future use 2SET_FEATURE 3Reserved for future use 4SET_ADDRESS 5GET_DESCRI

38、PTOR 6SET_DESCRIPTOR 7GET_CONFIGURATION 8SET_CONFIGURATION 9GET_INTERFACE 10SET_INTERFACE 11SYNCH_FRAME 12在上面的数据包里,看到它的内容是06,那么它就是GET_DESCRIPTOR类型。也就是主控器想读取USB设备的描述符,到这里就已经分析出来的意思,就是主控器想读取USB描述符,但还不知道是什么描述符的内容。3.wValue是根据不同的请求而设置不同的值。一般就是传送参数给设备标明这是什么请求。在上面GET_DESCRIPTOR获取设备描述符里,它的值是00 01。在GET_DESCR

39、IPTOR里这个字段的低字节表示描述符的索引,高字节表示描述符的类型。高字节的类型如下:DEVICE 1CONFIGURATION 2STRING 3INTERFACE 4ENDPOINT 5DEVICE_QUALIFIER 6OTHER_SPEED_CONFIGURATION 7INTERFACE_POWER1 8wValue值在这里的高字节是01,那么它就是设备描述符了。低字节是00,那么它就是表示从偏移地址0开始读取设备描述符。由于在配置描述符里有很多配置,所以低字节在那里就可以用来识别获取同样类型的描述符不同的配置。4.wIndex是根据不同的请求而设置不同的值。一般用来说明端点号或者

40、说明接口标识。在获取描述符里,设置为0,或者是语言ID。在这个发送的描述符里,它是设置为00 00。5.wLength是根据请求来决定下一阶段发送数据的长度。前面请求第一个字节里,已经说明下一阶段数据传送的方向,这里说明了传送数据的长度。不管是发送数据,还是接收数据,都不要超过这个数据长度,否则主机会出问题,或者设备有问题。在这个获取设备描述里,它的长度是40 00,按小端格式去解释,就是64个字节。到这里,就把主控器发下来的数据解释完成了,知道去做什么的事情和回应。下一次就去分析怎么样返回设备描述符。/* - */* -2- 回应设备描述符 */上一次已经介绍怎么样收到主控器的获取设备描述符

41、的数据,这里就解释怎么样发送回应数据给主控器。先从USB协议里找到标准设备的定义,我把它用C语言定义如下:typedef struct _USB_DEVICE_DESCRIPTOR BYTE bLength;BYTE bDescriptorType;WORD bcdUSB;BYTE bDeviceClass;BYTE bDeviceSubClass;BYTE bDeviceProtocol;BYTE bMaxPacketSize0;WORD idVendor;WORD idProduct;WORD bcdDevice;BYTE iManufacturer;BYTE iProduct;BYTE

42、iSerialNumber;BYTE bNumConfigurations; USB_DEVICE_DESCRIPTOR;返回给主控器的数据结构就是上面的内容,只要把上面的结构填写合适的内容,就可以发送回去给主控器。在我的USB设备里,我把它填写如下的数据:12 01 10 01 00 00 00 40 00 80 00 80 00 01 04 2C 4A 01看到这串数据是不明白是什么东西的,现在就来仔细地分析它的具体定义。下面就按着一个字段一个字段地分析它。bLength是本结构的数据长度,这样可以方便以后兼容不同的版本协议。因为不同的结构是不同的长度,这样就可以区分不同的协议了。比如有一

43、天想添加一个字段,那么它的长度就会改变,这时就可以根据不同的长度进行解释不同的协议了。这次返回的结构长度是0x12,也就是18个字节,它的长度是从bLength长度开始,也就是说是完全整个结构的长度。bDescriptorType是描述符的类型。它的定义跟主控器发下来描述符的类型是一样的,如下:DEVICE 1CONFIGURATION 2STRING 3INTERFACE 4ENDPOINT 5DEVICE_QUALIFIER 6OTHER_SPEED_CONFIGURATION 7INTERFACE_POWER1 8由于返回的是设备描述符,所以就选择了1,也就是包里显示的第二个字节01。用

44、这个类型来区分不同的描述符。bcdUSB是USB发布的协议版本。也就是本设备能适用于那种协议,目前USB主要有两个版本,一个是1.10,一个是2.10版本。在本设备里,采用了1.10的协议版本。由于这个字段是采用BCD编码,所以1.10的表示为0x0110的格式,按小端格式输出来,就变成10 01的显示了。bDeviceClass是设备分类。当它的值是0时,表示所有接口在配置描述符里,并且所有接口是独立的。当它的值是1到FEH时,表示不同的接口关联的。当它的值是FFH时,它是厂商自己定义的。在这个设备里,是定义为0。bDeviceSubClass是设备子分类码。当前面的bDeviceClass

45、值是0时,这里一定要设置为0。其它就跟据USB-IF组织定义的编码。bDeviceProtocol是设备使用的协议。如果使用USB-IF组织定义的协议,就需要设置这里的值。如果不使用,就直接设置为0。如果厂商自己定义的可以设置为FFH。以上三个值,在本设备里全部设置为0。bMaxPacketSize0是端点0收发最大的包大小。仅允许设置8,16,32,64中的任何一个大小。在本设备里是设置为64个字节大小。所以看到这个字段是40 的大小。12 01 10 01 00 00 00 40 00 80 00 80 00 01 04 2C 4A 01idVendor是厂商标识。由USB-IF分配的编码

46、。在这里使用0x8000。idProduct是厂商定义的产品标识。由厂家和产品标识,就可以让操作系统加载不同的驱动程序。如下:12 01 10 01 00 00 00 40 00 80 00 80 00 01 04 2C 4A 01bcdDevice是用BCD表示的设备发布的版本号。这里是1.00。12 01 10 01 00 00 00 40 00 80 00 80 00 01 04 2C 4A 01iManufacturer是厂商字符串的偏移值。这值主要说明了它在字符串描述符里的偏移位置。如果它设置为0,表示没有厂商字符串。在这里是0x04,就是从字符串描述符开始位置算起第4个字节位置读取

47、字符串。iProduct是产品字符串的偏移值。这值主要说明了它在字符串描述符里的偏移位置。如果它设置为0,表示没有产品字符串。在这里是0x2C,就是从字符串描述符开始位置算起第2C个字节位置读取字符串。iSerialNumber是序列号字符串的偏移值。这值主要说明了它在字符串描述符里的偏移位置。如果它设置为0,表示没有序列号字符串。在这里是0x4A,就是从字符串描述符开始位置算起第4A个字节位置读取字符串。所有字符串,都是采有UNICODE编码。bNumConfigurations是配置描述符的个数。在这里只使用了一个配置,所以设置为1。/* - */* -3- 设置USB地址 */前面已经解

48、释主控器怎么样发送设备描述符下来,然后设备返回相应的设备描述符。下一步主控器的动作是做什么呢?由于在USB总线上的设备有很多,为了区分不同的设备通讯,就需要给每个设备分配一个地址,这跟网络中的IP地址是一样的,或者跟MAC地址也是一样的。因而,接着下来就是主控器分配地址给设备,USB的设备地址是从1开始到127。下面就是接收到主控器发下来的数据包:00 05 01 00 00 00 00 00由USB_SETUP_PACKET定义具体地分析这个数据,就可以知道应做什么样的响应了。下面就来解释这个操作。先取得bmRequestType的类型,也就是第一个字节,它是00。从USB协议里查看,它的方

49、向位是主控器发送给设备,由D6D5位就知道它是USB协议里定义的标准请求,由D4-D0位知道它是USB设备接收这个包数据。bRequest是05,从前面已经介绍的类型,就知道它是设置地址,如下:SET_ADDRESS 5所以这个包需要按设置地址的格式去解释后面的数据。由于USB协议可以知道,USB的设备地址放在字段wValue里,因它的值是01 00,按小端格式解释就是0x0001了。其它相应的字段wIndex和wLength应都是0,如果是其它非0的数据,是没有定义的。USB的串行引擎通过这个地址来判断是否接收总线上的数据,如果发送的地址跟它一致,就会接收主控器发过来的数据,当然从这个设备发

50、送出去的数据也带有这个地址,因此就可以让主控器识别不同的USB设备数据了。/* - */* -4- 配置描述符 */前面已经介绍设置USB的设备地址,接着下来是做什么呢?其实有了设备地址后,主控器还会再次发送获取上面已经读取的设备描述符下来,如下:80 06 00 01 00 00 12 00然后USB设备也再次回应它,但这次发送的长度是0x0012了,不再是第一次64个字节长度了。接着USB设备就返回下面的描述符给主控器,也就是第一次已经发送的设备描述符,如下:12 01 10 01 00 00 00 40 00 80 00 80 00 01 04 2C 4A 01这样分配地址之后,再次获取

51、设备描述符成功了,接着下来就是主控器获取配置描述符。下面就是收到的配置描述符数据:80 06 00 02 00 00 09 00分析上面的数据如下:bmRequestType是80,表示方向USB设备发送给主控器,接收设备是USB设备。bRequest是06,表示这是获取描述符。GET_DESCRIPTOR 6wValue是00 02。低字节表示偏移地址00,高字节表示描述符的类型。如下:CONFIGURATION 2所以这里的返回的设备描述符是配置描述符。wIndex是00 00。wLength是09 00。它表示返回描述符的长度。这里是9个字节。接着下来,就是设备返回配置描述符给主控器,发

52、送的数据如下:09 02 22 00 01 01 00 01 32发送的数据是按下面的结构来定义,这也是在USB协议里定义的格式。如下:typedef struct _USB_CONFIGURATION_DESCRIPTOR BYTE bLength;BYTE bDescriptorType;WORD wTotalLength;BYTE bNumInterfaces;BYTE bConfigurationValue;BYTE iConfiguration;BYTE bmAttributes;BYTE MaxPower; USB_CONFIGURATION_DESCRIPTOR;bLength是

53、配置的长度,也就是配置结构的整个长度。在这里9个字节。bDescriptorType是描述符的类型,这里配置描述符,所以设置为02。wTotalLength是所有配置设置的结构长度。包括配置描述符、接口描述符、HID或者其它描述符和端点描述符的长度。这里是22 00,也就是0x0022个字节。bNumInterfaces是接口个数,这里一个。bConfigurationValue是配置的个数,当设置配置时发送的值。这时设置为1个配置。iConfiguration是说明配置的字符的偏移值。这里是0。bmAttributes是配置特性,D7位保留。D6位是说明是否自供电。D5位是否支持远程唤醒。D

54、4D0是保留,设置为0。MaxPower是使用的功率,它采用电流来表示。每2mA为单位,比如它的值是50时就表示是100mA的电流消耗。通过这样说明,主控器就知道这个设备是什么样的设备,有多少功能。/* - */* -5- 字符串描述符 */上一次说到把配置描述符返回给主控器那里了,现在接着下来,就会收到主控器发来字符串描述符。如果在设备描述符那里指定没有字符串描述的话,在这里是不会收到字符串描述符的。由于我在设备描述符里指定有字符串描述符的偏移地址,因此,就收到主控器发出请求字符串描述符。收到的数据如下:80 06 00 03 00 00 FF 00bmRequestType是80,表示方向

55、USB设备发送给主控器,接收设备是USB设备。bRequest是06,表示这是获取描述符。GET_DESCRIPTOR 6wValue是00 03。低字节表示偏移地址00,高字节表示描述符的类型,如下:STRING 3wIndex是00 00。wLength是FF 00。它表示返回描述符的长度。这里是256个字节。因此,这个获取字符串描述符,就是从字符串描述内存里,0偏移地址开始的位置读取第一个字符串描述符返回给主控器。接着就返回下面的数据给主控器:04 03 09 04上面的数据是按字符串描述符来组织的,它的结构,我定义结构如下:typedef struct _USB_STRING_DESC

56、RIPTOR BYTE bLength;BYTE bDescriptorType;WORD bString/*/; USB_STRING_DESCRIPTOR;bLength是所有数据的长度。在这里是4。bDescriptorType是描述类型,这里字符串描述符,所以它是3。bString是可变的字符数组。不超过254个应都可以的,并且它是使用UNICODE编码的字符串。在这里是09 04,这是美国英语的标识,0x0409。如果想输入中文的标识,只要改为0x0804就可以了。通过这个字符串描述符,主控器就知道字符串描述符是使用什么语言说明的了,这样就可以支持全世界的语言标识。USB描述符【整理】USB描述符USB描述符信息存储在USB设备中,在枚举过程中,USB主机会向USB设备发送GetDescriptor请 求,USB设备在收到这个请求之后,会将USB描述符信息返回给USB主机,USB主机分析返回来的数据,判断出该设备是哪一种USB设备,建立相应的数 据链接通道。那么USB描述符信息到底是一个什么样的数据呢,USB协议中有详细描述。通用的USB描述符信息包括设备描述符、配置描述符、接口描述符和端点描述符,具体不同的USB设备还包括其它类型的描述符,例如,USB鼠标、键盘还包括HID描述符和报告描述符,还有可能包括字

展开阅读全文
温馨提示:
1: 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
2: 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
3.本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
5. 装配图网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。
关于我们 - 网站声明 - 网站地图 - 资源地图 - 友情链接 - 网站客服 - 联系我们

copyright@ 2023-2025  zhuangpeitu.com 装配图网版权所有   联系电话:18123376007

备案号:ICP2024067431-1 川公网安备51140202000466号


本站为文档C2C交易模式,即用户上传的文档直接被用户下载,本站只是中间服务平台,本站所有文档下载所得的收益归上传人(含作者)所有。装配图网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对上载内容本身不做任何修改或编辑。若文档所含内容侵犯了您的版权或隐私,请立即通知装配图网,我们立即给予删除!