在USB设备的逻辑组织中,包含设备、配置、接口和端点4个层次。
配置:每个USB设备都提供了不同级别的配置信息,可以包含一个或多个配置,不同的配置使设备表现出不同的功能组合(在探测/连接期间需从其中选定一个),配置由多个接口组成。
接口:在USB协议中,接口由多个端点组成,代表一个基本的功能,是USB设备驱动程序控制的对象,一个功能复杂的USB设备可以具有多个接口。每个配置中可以有多个接口,而设备接口是端点的汇集(collection)。例如USB扬声器可以包含一个音频接口以及对旋钮和按钮的接口。一个配置中的所有接口可以同时有效,并可被不同的驱动程序连接。每个接口可以有备用接口,以提供不同质量的服务参数。在逻辑上,一个USB设备的功能划分是通过接口来完成的。比如说一个USB扬声器,可能会包括有两个接口:一个用于键盘控制,另外一个用于音频流传输。而事实上,这种设备需要用到不同的两个驱动程序来操作,一个控制键盘,一个控制音频流。
端点:是USB通信的最基本形式,每一个USB设备接口在主机看来就是一个端点的集合。主机只能通过端点与设备进行通信,以使用设备的功能。在USB系统中每一个端点都有惟一的地址,这是由设备地址和端点号给出的。每个端点都有一定的属性,其中包括传输方式、总线访问频率、带宽、端点号和数据包的最大容量等。一个USB端点只能在一个方向承载数据,或者从主机到设备(称为输出端点),或者从设备到主机(称为输入端点),因此端点可看作一个单向的管道。
端点可分为如下4种类型:
总体而言,USB设备非常复杂,由许多不同的逻辑单元组成,如图20.2所示,这些单元之间的关系如下:
图20.2 USB设备、配置、接口和端点 |
设备描述符:关于设备的通用信息,如供应商ID、产品ID和修订ID,支持的设备类、子类和适用的协议以及默认端点的最大包大小等。在Linux内核中,USB设备用usb_device结构体来描述,USB设备描述符定义为usb_device_descriptor结构体,如代码清单20.1所示。
代码清单20.1 usb_device_descriptor结构体
1 struct usb_device_descriptor 2 { 3 _ _u8 bLength; //描述符长度 4 _ _u8 bDescriptorType; //描述符类型编号 5 6 _ _le16 bcdUSB; //USB版本号 7 _ _u8 bDeviceClass; //USB分配的设备类code 8 _ _u8 bDeviceSubClass;// USB分配的子类code 9 _ _u8 bDeviceProtocol; //USB分配的协议code 10 _ _u8 bMaxPacketSize0; //endpoint0最大包大小 11 _ _le16 idVendor; //厂商编号 12 _ _le16 idProduct; //产品编号 13 _ _le16 bcdDevice; //设备出厂编号 14 _ _u8 iManufacturer; //描述厂商字符串的索引 15 _ _u8 iProduct; //描述产品字符串的索引 16 _ _u8 iSerialNumber; //描述设备序列号字符串的索引 17 _ _u8 bNumConfigurations; //可能的配置数量 18 } _ _attribute_ _ ((packed)); |
配置描述符:此配置中的接口数、支持的挂起和恢复能力以及功率要求。USB配置在内核中使用usb_host_config结构体描述,USB配置描述符定义为结构体usb_config_descriptor,如代码清单20.2所示。
代码清单20.2 usb_config_descriptor结构体
1 struct usb_config_descriptor 2 { 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 2 { 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 2 { 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 2 { 3 _ _u8 bLength; //描述符长度 4 _ _u8 bDescriptorType; //描述符类型 5 6 _ _le16 wData[1];/* 以UTF-16LE编码 */ 7 } _ _attribute_ _ ((packed)); |