USB协议详解第10讲(USB描述符-报告描述符)

目录

1.报告描述符的概念和作用

2.报告描述符中的通用项(Item)

3.通用项结构详解

4.Main、Global、Local标签功能

5.核心标签功能

6.STM32某鼠标设备报告描述符详细说明

7.报告描述符用途查询

8.报告描述符生成工具介绍


1.报告描述符的概念和作用

开门见山,报告描述符就是描述报告(HID接口上传输事务中的数据)的一组数据结构

首先大家可能会问,报告又是什么?我们前面讲过,USB主机一般是以中断的方式向HID设备发送或者索取数据,也就是说USB主机发送一个请求,设备要根据硬件操作,向主机提交自己的状态变化,例如鼠标,当主机给鼠标设备发送请求后,鼠标需要把自己当前位置信息发送给主机。其实大概应该清楚了,报告就是我们说的传输事务中的原生数据,如果是鼠标的话,这份报告则为鼠标左移、鼠标右移、鼠标滑轮滚动、鼠标左键、鼠标右键的当前状态数据的集合。

但是我们知道,我们看到的鼠标各各样,有很多鼠标是只有两个按键,有很多七八个按键,如下图,那么问题来了,这么多类型不一样的鼠标,主机是怎么知道鼠标设备发上来的一堆数据(报告)中哪个数据代表哪个按键,报告描述符来解决这个问题,这就是报告描述符的作用,就是描述HID设备数据的用途及属性。

图片图片

主机在对HID设备进行枚举的时候,会拿到报告描述符,之后给设备发送数据或者接收到设备数据会按照报告描述符对数据的每一位进行解析。

报告描述符如果全面讲解可以单独成书(还真是两本书,书名为hid1_11.pdf和hut1_21_0.pdf,大家可以关注一个早起的程序员,然后点击下载获取),本篇文章会抓重点讲解,教会大家报告描述符的数据结构并举例,大家可以在学习过程中举一反三。

2.报告描述符中的通用项(Item)

我们知道了报告描述符的概念及作用之后,那么大家肯定想知道了报告描述符是怎么描述数据的,其实报告描述符是由一个一个通用项组成,每一个通用项可以描述一个或者多个相同功能的数据(比如同时描述8个按钮),包括数据的用途和各种属性。

3.通用项结构详解

Item有两种类型,Short Item和Long Item,结构组成如下,一般我们接触到的都是Short Item,本篇博文也主要讲解Short Item。

(1)Short Item

(2)Long Item

在 Short Item 中,起始的 1 Byte 代表了这个 Item 的大小和类型,如下:

bSize  :代表后面的数据,最大 4 bytes。

bType  :代表了这个 Item 是什么类型的 Item,这里有三种大的类型:Main(0x00)、Global(0x01)、Local(0x02)。

bTag    :代表在对应的 Item 下的更加细的分类,也可以称之为标签。

    | —— Main      分为:Input、Output、Feature、Collection、End Collection。

    | —— Global    分为:Usage Page、 Logical Minimum、Logical Maximum、Physical Minimum、Physical Maximum、Report Size、Report ID等等。

    | —— Local     分为 :Usage 、Usage Minimum、Usage Maximum、String 等等。

对于 bSize + bType + bTag 的 1 Byte 的组合详见下表:

4.Main、Global、Local标签功能

非常核心的核心: 在阅读 Report 的时候,都是先用Global和Local标签说明数据项的各种相关属性,然后最后用一个Main标签进行输入输出说明,Main标签的出现就是一个数据项的结束。

前面提到通用项主要分为Main、Global、Local,大家肯定会问为什么要分类,他们三个到底是用来做什么的,下面一一讲解。

Mian类标签:主要用来对数据项的描述(Input、Output、Feature)和管理(Collection、End Collection)。

描述数据项:比如一个这个数据项到底是主机输入还是输出;

管理数据项:比如把多个数据项组织到一起。

Global类标签:对数据项的描述,这个主要用来描述数据项的用途数据逻辑范围数据物理范围数据大小数据个数,它的用法和我们平时看到的全局变量特别像,当定义之后,之后的所有数据项都是使用定义的属性,除非再此定义,如下详细说明。

Local类标签:对数据项的描述,和Global标签的作用类似,它的用法和我们平时看到的局部变量特别像,就是作用于不一样,只在本数据内项生效。

5.核心标签功能

1.Usage Page和Usage

用于描述数据项的用途,这个标签非常复杂,当UsagePage变化时Usage子功能也会跟着变化,下面会详细讲解。​也可以阅读文档“HID Usage Tables”获得更多使用细节,大家可以关注我,然后点击下载获取本书电子版,也可以去USB IF官网下载。

2.Logical Maximum

描述数据逻辑单元中的范围值。这是变量或数组项将要报告的最大值。例如设备报告的一个电压值读数是1V,而一个单位是 2mV ,则 Logical Maximum 值等于 500 。

3.Logical Minimum

描述数据逻辑单元中的范围值。这是变量或数组项将要报告的最小值。

4.Physical Maximum

描述数据真实的最大值,比如电压最大为2V,一个单位是 2mV ,则该值最大仍是2000mV,表示物理真实值。

5.Physical Minimum

描述数据真实的最小值。

6.Report Size

以位(bit)为单位指定数据的大小,无符号整数。 

7.Report Count

无符号整数,指定总共需要多少个同类型的数据。

8.Input

此项数据为设备到主机的。

9.Output

此项数据为主机到设备的。

10.Report ID

报告的ID,一个报告描述符可以描述多个HID功能,主机通过报告的ID就知道是哪个HID功能硬件发送的数据。举个例子,如果一个USB HID设备为键鼠一体的设备,就可以通过Report  ID把键盘和鼠标的数据描述分开,这样可以实现面向对象管理,其实就是相当于实现了多个报告,设备在发送数据的时候,第一个字节永远是Report ID,如果是鼠标发送事务第一个字节(Report ID)为1,如果是键盘发送事务第一个字节(Report ID)则为2,这样分开描述数据其实也就是分开了事务功能本身,如果一个很复杂的HID设备在设计的思路上可以分治而行,可以大大减小设计难度。

11.Collection

用于对数据描述项进行组织和管理,和End Collection一起使用

6.STM32某鼠标设备报告描述符详细说明

上面的报告描述符实现一个鼠标设备,主要有3个数据描述项组成。描述符总共描述了4个字节的报告(数据),数据描述项1描述了数据的第1个字节前3bit,代表鼠标的3个按钮,数据描述项2为了描述数据对齐操作,数据描述项3描述了数据的第2-4字节,代表鼠标的左右移动、上下移动、滚轮滚动的数据,这4个字节都是设备到主机的数据。其他的细节在注释里面讲解的很清楚。

假设设备给主机发送了当前最新4个字节的数据(报告),先低字节后高字节:0x01 0x00 0x00 0x00,主机会解析到鼠标左键被按下,鼠标没有移动,滑轮也没滚动,就是这样子理解的。

7.报告描述符用途查询

报告描述符Usage page和Usage可以通过hut1_21_0.pdf(HID Usage Tables )文档查询,首先每一个Usage Pages下面都包含多种Usage,下面会说明所有的Usage Pages,每一个Usage Pase下面的Usage大家可以直接点击查看。

我们可以点击Generic Desktop Page (0x01)去查询下面更具体的Usage,如下:

8.报告描述符生成工具介绍

我们可以用HID Descriptor Tool生成报告描述符,也可以直接打开(open file)一些标准设备的报告描述符进行学习,工具如下图,大家可以根据自己的硬件定义自己的报告描述符。

图片

图片


1.本文部分素材来源网络,版权归原作者所有,如涉及作品版权问题,请与我联系删除。

2.未经原作者允许不得转载本文内容,否则将视为侵权;

3.转载或者引用本文内容请注明来源及原作者;

4.对于不遵守此声明或者其他违法使用本文内容者,本人依法保留追究权等。

下面是我的个人微信公众号,关注【一个早起的程序员】精彩系列文章每天不断。

  • 35
    点赞
  • 86
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 3
    评论
gu8ConfigDescriptor是一个USB配置描述符数组,用于描述USB设备的配置信息。它通常由USB设备的制造商在设备固件中定义,由USB主机在插入设备时读取。 USB配置描述符包含了USB设备的配置信息,包括配置的唯一标识符、配置的描述信息、配置所需的最大功率、以及该配置所包含的所有接口和其他描述符的信息。 以下是一个典型的USB配置描述符的格式: ``` typedef struct { uint8_t bLength; // 描述符长度 uint8_t bDescriptorType; // 描述符类型 uint16_t wTotalLength; // 描述符总长度 uint8_t bNumInterfaces; // 接口数目 uint8_t bConfigurationValue; // 配置唯一标识符 uint8_t iConfiguration; // 配置描述信息 uint8_t bmAttributes; // 配置属性 uint8_t bMaxPower; // 配置所需的最大功率 } USB_ConfigurationDescriptorTypeDef; ``` 其中,bLength字段表示该描述符的长度;bDescriptorType字段表示该描述符的类型,这里应该为0x02,表示该描述符是一个配置描述符;wTotalLength字段表示该配置描述符及其后续的所有描述符的总长度;bNumInterfaces字段表示该配置所包含的接口数目;bConfigurationValue字段表示该配置的唯一标识符;iConfiguration字段表示该配置的描述信息的字符串索引;bmAttributes字段表示该配置的属性,包括是否需要供电、是否支持远程唤醒等;bMaxPower字段表示该配置所需的最大功率,单位为2mA。 在gu8ConfigDescriptor描述符数组中,每个USB配置描述符之后通常还会包括该配置所包含的所有接口的描述符和其他描述符,如字符串描述符、端点描述符等。这些描述符的总长度应该等于wTotalLength字段指定的长度。通过读取gu8ConfigDescriptor数组中的各个描述符USB主机可以了解到USB设备支持的配置信息、接口信息和传输方式等,从而选择最合适的驱动程序进行通信。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

一个早起的程序员

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值