当前位置:嗨网首页>书籍在线阅读

04-设备、配置、接口、端点

  
选择背景色: 黄橙 洋红 淡粉 水蓝 草绿 白色 选择字体: 宋体 黑体 微软雅黑 楷体 选择字体大小: 恢复默认

20.1.2 设备、配置、接口、端点

在USB设备的逻辑组织中,包含设备、配置、接口和端点4个层次。

每个USB设备都提供了不同级别的配置信息,可以包含一个或多个配置,不同的配置使设备表现出不同的功能组合(在探测/连接期间需从其中选定一个),配置由多个接口组成。

在USB协议中,接口由多个端点组成,代表一个基本的功能,是USB设备驱动程序控制的对象,一个功能复杂的USB设备可以具有多个接口。每个配置中可以有多个接口,而设备接口是端点的汇集(collection)。例如,USB扬声器可以包含一个音频接口以及对旋钮和按钮的接口。一个配置中的所有接口可以同时有效,并可被不同的驱动程序连接。每个接口可以有备用接口,以提供不同质量的服务参数。

端点是USB通信的最基本形式,每一个USB设备接口在主机看来就是一个端点的集合。主机只能通过端点与设备进行通信,以使用设备的功能。在USB系统中每一个端点都有惟一的地址,这是由设备地址和端点号给出的。每个端点都有一定的属性,其中包括传输方式、总线访问频率、带宽、端点号和数据包的最大容量等。一个USB端点只能在一个方向承载数据,或者从主机到设备(称为输出端点),或者从设备到主机(称为输入端点),因此端点可看作一个单向的管道。端点0通常为控制端点,用于设备初始化参数等。只要设备连接到USB上并且上电端点0就可以被访问。端点1、2等一般用作数据端点,存放主机与设备间往来的数据。

总体而言,USB设备非常复杂,由许多不同的逻辑单元组成,如图20.2所示,这些单元之间的关系如下:

P528_51346.jpg ● 设备通常有一个或多个配置;

● 配置通常有一个或多个接口;

● 接口通常有一个或多个设置;

● 接口有零或多个端点。

这种层次化配置信息在设备中通过一组标准的描述符来描述,如下所示。

● 设备描述符:关于设备的通用信息,如供应商ID、产品ID和修订ID,支持的设备类、子类和适用的协议以及默认端点的最大包大小等。在Linux内核中,USB设备用usb_device结构体来描述,USB设备描述符定义为usb_device_descriptor结构体,如代码清单20.1所示。

代码清单20.1 usb_device_descriptor结构体

1 struct usb_device_descriptor {

2 _ _u8 bLength; / 描述符长度 /

3 _ _u8 bDescriptorType; / 描述符类型编号 /

4

5 _ _le16 bcdUSB; / USB版本号 /

6 _ _u8 bDeviceClass; / USB分配的设备类code /

7 _ _u8 bDeviceSubClass;/ USB分配的子类code /

8 _ _u8 bDeviceProtocol; / USB分配的协议code /

9 _ _u8 bMaxPacketSize0; / endpoint0最大包大小 /

10 _ _le16 idVendor; / 厂商编号 /

11 _ _le16 idProduct; / 产品编号 /

12 _ _le16 bcdDevice; / 设备出厂编号 /

13 _ _u8 iManufacturer; / 描述厂商字符串的索引 /

14 _ _u8 iProduct; / 描述产品字符串的索引 /

15 _ _u8 iSerialNumber; / 描述设备序列号字符串的索引 /

16 _ _u8 bNumConfigurations; / 可能的配置数量 /

17 } _ attribute _ ((packed));

● 配置描述符:此配置中的接口数、支持的挂起和恢复能力以及功率要求。USB配置在内核中使用usb_host_config结构体描述,USB配置描述符定义为结构体usb_config_descriptor,如代码清单20.2所示。

代码清单20.2 usb_config_descriptor结构体

1 struct usb_config_descriptor {

3 _ _u8 bLength; / 描述符长度 /

4 _ _u8 bDescriptorType; / 描述符类型编号 /

5

6 _ _le16 wTotalLength; / 配置所返回的所有数据的大小 /

7 _ _u8 bNumInterfaces; / 配置所支持的接口数 /

8 _ _u8 bConfigurationValue; / Set_Configuration命令需要的参数值 /

9 _ _u8 iConfiguration; / 描述该配置的字符串的索引值 /

10 _ _u8 bmAttributes; / 供电模式的选择 /

11 _ _u8 bMaxPower; / 设备从总线提取的最大电流 /

12 } attribute _ ((packed));

● 接口描述符:接口类、子类和适用的协议,接口备用配置的数目和端点数目。USB接口在内核中使用usb_interface结构体描述,USB接口描述符定义为结构体usb_interface_descriptor,如代码清单20.3所示。

代码清单20.3 usb_interface_descriptor结构体

1 struct usb_interface_descriptor {

3 _ _u8 bLength; / 描述符长度 /

4 _ _u8 bDescriptorType; / 描述符类型 /

5

6 _ _u8 bInterfaceNumber; / 接口的编号 /

7 _ _u8 bAlternateSetting; / 备用的接口描述符编号 /

8 _ _u8 bNumEndpoints; / 该接口使用的端点数,不包括端点0 /

9 _ _u8 bInterfaceClass; / 接口类型 /

10 _ _u8 bInterfaceSubClass; / 接口子类型 /

11 _ _u8 bInterfaceProtocol; / 接口所遵循的协议 /

12 _ _u8 iInterface; / 描述该接口的字符串索引值 /

13 } _ attribute _ ((packed));

● 端点描述符:端点地址、方向和类型,支持的最大包大小,如果是中断类型的端点则还包括轮询频率。在Linux内核中,USB端点使用usb_host_endpoint结构体来描述,USB端点描述符定义为usb_endpoint_descriptor结构体,如代码清单20.4所示。

代码清单20.4 usb_endpoint_descriptor结构体

1 struct usb_endpoint_descriptor {

3 _ _u8 bLength; / 描述符长度 /

4 _ _u8 bDescriptorType; / 描述符类型 /

5 _ _u8 bEndpointAddress; / 端点地址:0~3位是端点号,第7位是方向(0-OUT,1-IN) /

6 _ _u8 bmAttributes; / 端点属性:bit[0:1] 的值为00表示控制,为01表示同步,为02表示批量,为03表示中断 /

7 _ _le16 wMaxPacketSize; / 本端点接收或发送的最大信息包的大小 /

8 _ _u8 bInterval; / 轮询数据传送端点的时间间隔 /

9 / 对于批量传送的端点以及控制传送的端点,此域忽略 /

10 / 对于同步传送的端点,此域必须为1 /

11 / 对于中断传送的端点,此域值的范围为1~255 /

12 _ _u8 bRefresh;

13 _ _u8 bSynchAddress;

14 } _ attribute _ ((packed));

● 字符串描述符:在其他描述符中会为某些字段提供字符串索引,它们可被用来检索描述性字符串,可以以多种语言形式提供。字符串描述符是可选的,有的设备有,有的设备没有,字符串描述符对应于usb_string_descriptor结构体,如代码清单20.5所示。

代码清单20.5 usb_string_descriptor结构体

1 struct usb_string_descriptor {

3 _ _u8 bLength; / 描述符长度 /

4 _ _u8 bDescriptorType; / 描述符类型 /

5

6 _ _le16 wData[1]; / 以UTF-16LE编码 /

7 } _ attribute _ ((packed));

例如,笔者在PC上插入一个SanDisk U盘后,通过lsusb命令得到这个U盘相关的描述符,从中可以显示这个U盘包含了一个设备描述符、一个配置描述符、一个接口描述符以及批量输入和批量输出两个端点描述符。呈现出来的信息内容直接对应于usb_device_descriptor、usbconfig descriptor、usb_interface_descriptor、usb_endpoint_descriptor、usb_string_descriptor结构体,如下所示:

Bus 001 Device 004: ID 0781:5151 SanDisk Corp.

Device Descriptor:

bLength 18

bDescriptorType 1

bcdUSB 2.00

bDeviceClass 0 Interface

bDeviceSubClass 0

bDeviceProtocol 0

bMaxPacketSize0 64

idVendor 0x0781 SanDisk Corp.

idProduct 0x5151

bcdDevice 0.10

iManufacturer 1 SanDisk Corporation

iProduct 2 Cruzer Micro

iSerial 3 20060877500A1BE1FDE1

bNumConfigurations 1

Configuration Descriptor:

bLength 9

bDescriptorType 2

wTotalLength 32

bNumInterfaces 1

bConfigurationValue 1

iConfiguration 0

bmAttributes 0x80

MaxPower 200mA

Interface Descriptor:

bLength 9

bDescriptorType 4

bInterfaceNumber 0

bAlternateSetting 0

bNumEndpoints 2

bInterfaceClass 8 Mass Storage

bInterfaceSubClass 6 SCSI

bInterfaceProtocol 80 Bulk (Zip)

iInterface 0

Endpoint Descriptor:

bLength 7

bDescriptorType 5

bEndpointAddress 0x81 EP 1 IN

bmAttributes 2

Transfer Type Bulk

Synch Type none

wMaxPacketSize 512

bInterval 0

Endpoint Descriptor:

bLength 7

bDescriptorType 5

bEndpointAddress 0x01 EP 1 OUT

bmAttributes 2

Transfer Type Bulk

Synch Type none

wMaxPacketSize 512

bInterval 1

Language IDs: (length=4)

0409 English(US)