HID类的JoyStick描述符
应用程序改自沁恒官网的CH583EVT包中的CompoundDev工程,配合下方的描述符能够实现①直接接电脑,在设备管理器中能够查到被电脑识别为HID- compliant game contorller;②在CH582m单片机中自定义了回包内容,通过USB抓包工具可以抓到单片机模拟的JoyStick与电脑主机正常通信。笔者将把主要函数贴在另一篇随笔中。
设备、接口、类、端点描述符都是照猫画虎改的,每行的注释摘自《USB设计及应用设计》一书。HID报表描述符是根据HID1.12规范写的,JoyStick的报表描述符搬用了HID1.12手册的例子,报表格式如下。
在电脑设备管理器中识别的类型↓
USB抓包↓(数值是随便写的,一般按键松开后,需要把对应位置改成0x00再上传一包)
JoyStick设备长这样↓
/*前缀:b:8位字节 * w:16位字 * bm:按位寻址 * i:索引 * id:标识码 * bcd:采用BCD码编码*/ const uint8_t MyDevDescr[] = { // 设备描述符↓ 0x12, //bLength 描述符的字节数长度 标准为0x12 0x01, //bDescriptorType 描述符的类型 0x01为设备描述符 0x10, 0x01, //bcdUSB 设备支持的协议版本号,0x0110表示为1.1 0x00, //bDeviceClass 设备类代码 0:由每个接口指出它自己的类且各自独立工作 FF:设备类由厂商定义 //其他值表示设备在不同的接口上支持不同的类,接口之间可能不能独立工作 0x00, //bDeviceSubClass 设备子类代码 根据手册以及bDeviceClass的值而定,bDeviceClass为0,此值也要为0;若bDeviceClass为0xFF,此值保留 0x00, //bDevicePortocol 协议码 根据手册以及bDeviceClass和bDeviceSubClass的值而定,上两者为0,该值也要为0 0x08, //bMaxPacketSize0 端点0的数据包的最大净长度 0x00, 0x00, //idVendor 厂商ID 0x00, 0x00, //idProduct 产品ID 0x00, 0x00, //bcdDevice BCD设备发行号 0x00, //iManufacturer 厂商信息字符串的索引 0x00, //iProduct 产品信息字符串的索引 0x00, //iSerialNumber 设备序列号字符串的索引 0x01 //bNumConfigurations 可能的配置描述符数目 只有一种配置时此值为1 }; const uint8_t MyCfgDescr[] = { // 配置描述符↓ 0x09, //bLength 描述符的字节数长度 标准为0x09 0x02, //bDescriptorType 描述符的类型 0x02为配置描述符 0x22, 0x00, //wTotalLength 配置信息的总长(包括配置、接口、端点、类和厂商描述符) 0x01, //bNumInterfaces 该配置所支持的接口数目 0x01, //bConfigurationValue 被SET_CONFIGURATION请求用作参数来选定 0x00, //iConfiguration 该配置的字符串描述符索引值,在SET_CONFIGURATION中用作选定配置的参数 0xA0, //bmAttributes D6:自供电 D5:远程唤醒 其他位保留。 //若一个设备既能自供电又能使用总线供电,D6也要置1并在MaxPower指出需要从总线获取的电量 0x32, //MaxPower 该配置下的总线电源耗费量,为两倍此值(如0x32*2 = 100)mA //接口描述符↓ 0x09, //bLength 描述符的字节数长度 标准为0x09 0x04, //bDescriptorType 描述符的类型 0x04为接口描述符 0x00, //bInterfaceNumber 接口号,当前配置支持的接口数组索引(从0开始)。 //若该配置有俩接口,接下来的接口描述符此值就为1 0x00, //bAlternateSetting 可选设置的索引值。一个接口可以有多个接口描述符,靠此字段区分 0x01, //bNumEndpoints 除了USB设备必须支持的端点0外,此接口所包括的端点的个数。此值为0表示该接口只使用端点0。 0x03, //bInterfaceClass 接口所属的类值。为0表示为将来的标准保留。为FF表示此接口类由厂商说明。其他值查手册。0x03为HID类 0x00, //bInterfaceSublass 接口所属子类的值。根据手册以及bInterfaceClass的值来定,若上面的值为0,此值也要为0。 0x00, //bInterfaceProtocol 协议码,视手册以及上两者而定。上两者为0,此值也要为0。 0x00, //iInterface 此接口的字符串描述符的索引值。 //HID类描述符↓ 0x09, //bLength 描述符的字节数长度 标准为0x09 0x21, //bDescriptorType 描述符的类型 0x21为人机接口类描述符 0x12, 0x01, //bcdHID HID规范版本号的BCD码,此描述符所用版本为1.12 0x00, //bCountryCode 硬件目的国家的识别号码。不启用此功能则为0 0x01, //bNumDescriptors 支持的附属描述符数目。最小值为1:HID类至少有个报表描述符 0x22, //bDescriptorType 类别描述符的类型。只有报表描述符为0x22;还有个实体描述符则为0x23 0x4e, 0x00, //wDescriptorLength 报表描述符总长度 //端点描述符↓ 0x07, //bLength 描述符的字节数长度 标准为0x07 0x05, //bDescriptorType 描述符的类型 0x05为端点描述符 0x81, //bmEndpointAddress 低四位为端点号,最高位为0:OUT方向,为1:IN方向。其他位保留。 0x03, //bmAttributes 低两位为 00:控制传输;01:实时传输;10:批量传输;11:中断传输。其他位保留 0x08, 0x00, //bMaxPacketSize 端点收发的数据包最大净长度 0x0a, //bInterval 周期数据传输端点的时间间隙。对于批量和控制传输来说无意义 //若该端点配置实时传输,此值必须为1(ms);若该端点配置中断传输,此值为1~255(ms) }; //JoyStick报表描述符↓,大多数是查手册得到的数值。每行为一个short item。每个item的第一个字节的低两位表示这个字节后面还有多少字节的数据 const uint8_t JoyStickRepDesc[] = { //模拟操纵飞机的一种摇杆手柄 0x05, 0x01, //Usage Page (Generic Desktop) UsagePage定义数据的用法或功能 0x09, 0x04, //Usage (Joystick) Usage描述项目或collection的使用的索引 0xA1, 0x01, //Collection (Application) 包含有共同用途或者执行单一功能的项目。该项目可能指代节流阀、X方向和Y方向的指针,一起用来控制飞机的移动 0x05, 0x02, //Usage Page (Simulation Controls) 模拟输出的控制器 0x09, 0xBB, //Usage (Throttle) 节流阀,类似于飞机的油门 0x15, 0x81, //Logical Minimum (-127) 报表项目的最小数值 0x25, 0x7F, //Logical Maximum (127) 报表项目的最大数值 0x75, 0x08, //Report Size (8) 项目数据段的大小 每个数据段为8位 0x95, 0x01, //Report Count (1) 项目的数据段的数目 1个数据段 0x81, 0x02, //Input (Data, Variable, Absolute) 0x02的各位含义详见卷标 0x05, 0x01, //Usage Page (Generic Desktop) 通用桌面设备 0x09, 0x01, //Usage (Pointer) 指针,控制飞机的上下左右方向(鼠标也是一种Pointer) 0xA1, 0x00, //Collection (Physical) 包含代表数据在一个单一几何上的项目 0x09, 0x30, //Usage (X) X方向 0x09, 0x31, //Usage (Y) Y方向 0x95, 0x02, //Report Count (2) (X和Y方向)回传2个数据。协议手册的例子中并没有列ReportSize,该值沿用上一个ReportSize的值即8位 0x81, 0x02, //Input (Data, Variable, Absolute) 0x02的各位含义详见卷标 0xC0, //End Collection 关闭集合 0x09, 0x39, //Usage (Hat switch) 苦力帽,在游戏中用于控制飞机的视野。根据下面的数值,可能这是个4向苦力帽。 0x15, 0x00, //Logical Minimum (0) 报表项目的最小数值 0x25, 0x03, //Logical Maximum (3) 报表项目的最大数值 0x35, 0x00, //Physical Minimum (0) 以实际单位表示的逻辑小数值 0x46, 0x0E, 0x01, //Physical Maximum (270) 以实际单位表示的逻辑大数值 //将0~270这些数值分到0~3这些数据中,相当于每一个报表数值代表了67.75个实际单位 0x65, 0x04, //Unit (English Rotation: Angular Position) 表示单位。详见表格,本设备单位为英制、角度 //本设备只用到第0个半字节,故只需要1字节数据即可表示单位。0x65后面最多可达4个字节的数据,即8个半字节来表示单位 0x55, 0x00, //Unit Exponent (0) 10的(0)次幂。详见数值表,00h~到07h符号为正,08h~0Fh符号为负 0x75, 0x04, //Report Size (4) 每个数据段占4位 0x95, 0x01, //Report Count (1) 1个数据段 0x81, 0x42, //Input (Data, Variable, Absolute, Null State) 0x42的各位含义详见卷标 0x05, 0x09, //Usage Page (Buttons) 按键 0x19, 0x01, //Usage Minimum 定义1~4号四个按键 0x29, 0x04, //Usage Maximum 0x15, 0x00, //Logical Minimum (0) 报表项目的最小数值 0x25, 0x01, //Logical Maximum (1) 报表项目的最大数值 0x35, 0x00, //Physical Minimum (0) 以实际单位表示的逻辑小数值 0x45, 0x01, //Physical Maximum (1) 以实际单位表示的逻辑大数值 0x95, 0x04, //Report Count (4) 4个数据段 0x75, 0x01, //Report Size (1) 每个数据段占1位 0x64, //Unit (None) 无单位 0x81, 0x02, //Input (Data, Variable, Absolute) 0xC0 //End Collection };