搞定蓝牙-第六篇(HID

搞定蓝牙-第六篇(HID)

  • ble与HID
    • HOGP
    • GAPP与HID
  • ESP32程序分析

ble与HID

HOGP

我们发现,电脑连接了蓝牙键盘就可以直接使用了,不需要配置任何东西,那么,这两者是怎么通讯的呢。我们使用的电脑windows系统内置一段程序来自动识别鼠标、键盘的数据, windows系统的这个程序和鼠标、键盘内部的程序的需要使用同一套规则,两者才能正常通讯。除了鼠标、键盘还有游戏手柄、多媒体控制器等等都可以使用这套规则,这类设备叫做HID(Human Interface Device)。在ble中,我们称这套规则为HOGP(HID Over GATT Profile)。

GAPP与HID

HOGP的层次比GAP、GATT协议要高一层,也就是HOGP是基于这两个协议的,也就是说,要实现HOGP首先需要实现GAP和GATT。
之前的内容说到,ble设备的通讯是基于GATT的,通讯的内容最小单元是Attribute,所以HOGP也不例外,而GATT的核心是属性表,程序中要做的就是维护属性表。我们来看看HOGP的属性表是怎么样的。
在这里插入图片描述
我们用电脑连接有些键盘、鼠标会看到能显示电池电量,这个也是通过GATT实现的,也就是一个普通的键盘中的ble的GATT实现了HID服务和电池服务。从上图也能看出来确实如此。
首先了解电池服务,这个比较简单,就三个属性:特征值、客户端特性配置(CCC)和特征展示格式。前两个在GATT那一章说过了,特征值就是实际要传输的电池电量数据,而这里的特征展示格式就是表示这个电池电量数据以什么样的方式来传输(是百分百还是其他、是毫安还是安、是带小数点还是放大了十倍)。
HID服务就比较复杂了,包含了五个属性。HID信息,用于显示这个HID的一些基础信息,它的值就是由这些部分组成,版本号、硬件本地化(一个本地ID,如果有多个HID服务,需要设置不同ID)和是否支持远程唤醒。
在这里插入图片描述
协议模式,我们发现,windows在启动的时候可以输入键盘的键值来进入bios模式,但是在进入bios模式的时候还没初始化HID的程序呢,那么键盘是怎么和他正常通讯的呢。所以HID设备需要支持两种模式,一种用于在系统HID程序初始化之前,使用更加底层的协议来通讯,另外一种在系统初始化了HID程序之后,也就是配置启动模式或报告模式。
在这里插入图片描述

控制点,一般蓝牙键盘是需要低功耗设计的,HID的保持连接是需要一直传输数据的,而射频是很耗电的,所以,在系统注销、长时间未检测到用户使用这个设备的时候,键盘应该能自动休眠,这个时候可以通过控制点这个属性通知键盘进入低功耗模式了(主动关闭HID连接等)。
在这里插入图片描述

报告地图,这个东西在USB HID那边叫做报告描述符。我们使用的键盘可能是87键、104键、108键,还有一些宏键盘是几个按键的,那么HOGP是怎么支持这些所有这些不同类型的键盘的呢?所以,键盘需要告诉系统,我有多少个键、我有多少个LED、我的键值从哪些到哪些等等。这些内容通过报告地图固定在键盘的芯片里面了。举个栗子。

//表示用途页为通用桌面设备 
    0x05, 0x01,                    // USAGE_PAGE (Generic Desktop) 
 
    //表示用途为键盘 
    0x09, 0x06,                    // USAGE (Keyboard) 
     
    //表示应用集合,必须要以END_COLLECTION来结束它,见最后的END_COLLECTION 
    0xa1, 0x01,                    // COLLECTION (Application) 
     
    //表示用途页为按键 
    0x05, 0x07,                    //   USAGE_PAGE (Keyboard) 
 
    //用途最小值,这里为左ctrl键 
    0x19, 0xe0,                    //   USAGE_MINIMUM (Keyboard LeftControl) 
    //用途最大值,这里为右GUI键,即window键 
    0x29, 0xe7,                    //   USAGE_MAXIMUM (Keyboard Right GUI) 
    //逻辑最小值为0 
    0x15, 0x00,                    //   LOGICAL_MINIMUM (0) 
    //逻辑最大值为1 
    0x25, 0x01,                    //   LOGICAL_MAXIMUM (1) 
    //报告大小(即这个字段的宽度)为1bit,所以前面的逻辑最小值为0,逻辑最大值为1 
    0x75, 0x01,                    //   REPORT_SIZE (1) 
    //报告的个数为8,即总共有8个bits 
    0x95, 0x08,                    //   REPORT_COUNT (8) 
    //输入用,变量,值,绝对值。像键盘这类一般报告绝对值, 
    //而鼠标移动这样的则报告相对值,表示鼠标移动多少 
    0x81, 0x02,                    //   INPUT (Data,Var,Abs) 
    //上面这这几项描述了一个输入用的字段,总共为8个bits,每个bit表示一个按键 
    //分别从左ctrl键到右GUI键。这8个bits刚好构成一个字节,它位于报告的第一个字节。 
    //它的最低位,即bit-0对应着左ctrl键,如果返回的数据该位为1,则表示左ctrl键被按下, 
    //否则,左ctrl键没有按下。最高位,即bit-7表示右GUI键的按下情况。中间的几个位, 
    //需要根据HID协议中规定的用途页表(HID Usage Tables)来确定。这里通常用来表示 
    //特殊键,例如ctrl,shift,del键等 
 
  
 
    //这样的数据段个数为1 
    0x95, 0x01,                    //   REPORT_COUNT (1) 
    //每个段长度为8bits 
    0x75, 0x08,                    //   REPORT_SIZE (8) 
    //输入用,常量,值,绝对值 
    0x81, 0x03,                    //   INPUT (Cnst,Var,Abs) 
     
    //上面这8个bit是常量,设备必须返回0 
 
 
    //这样的数据段个数为5 
    0x95, 0x05,                    //   REPORT_COUNT (5) 
    //每个段大小为1bit 
    0x75, 0x01,                    //   REPORT_SIZE (1) 
    //用途是LED,即用来控制键盘上的LED用的,因此下面会说明它是输出用 
    0x05, 0x08,                    //   USAGE_PAGE (LEDs) 
    //用途最小值是Num Lock,即数字键锁定灯 
    0x19, 0x01,                    //   USAGE_MINIMUM (Num Lock) 
    //用途最大值是Kana,这个是什么灯我也不清楚^_^ 
    0x29, 0x05,                    //   USAGE_MAXIMUM (Kana) 
    //如前面所说,这个字段是输出用的,用来控制LED。变量,值,绝对值。 
    //1表示灯亮,0表示灯灭 
    0x91, 0x02,                    //   OUTPUT (Data,Var,Abs) 
 
    //这样的数据段个数为1 
    0x95, 0x01,                    //   REPORT_COUNT (1) 
    //每个段大小为3bits 
    0x75, 0x03,                    //   REPORT_SIZE (3) 
    //输出用,常量,值,绝对 
    0x91, 0x03,                    //   OUTPUT (Cnst,Var,Abs)     
    //由于要按字节对齐,而前面控制LED的只用了5个bit, 
    //所以后面需要附加3个不用bit,设置为常量。 
 
  
 
    //报告个数为6 
    0x95, 0x06,                    //   REPORT_COUNT (6) 
    //每个段大小为8bits 
    0x75, 0x08,                    //   REPORT_SIZE (8) 
    //逻辑最小值0 
    0x15, 0x00,                    //   LOGICAL_MINIMUM (0) 
    //逻辑最大值255 
    0x25, 0xFF,                    //   LOGICAL_MAXIMUM (255) 
    //用途页为按键 
    0x05, 0x07,                    //   USAGE_PAGE (Keyboard) 
    //使用最小值为0 
    0x19, 0x00,                    //   USAGE_MINIMUM (Reserved (no event indicated)) 
    //使用最大值为0x65 
    0x29, 0x65,                    //   USAGE_MAXIMUM (Keyboard Application) 
    //输入用,变量,数组,绝对值 
    0x81, 0x00,                    //   INPUT (Data,Ary,Abs) 
    //以上定义了6个8bit宽的数组,每个8bit(即一个字节)用来表示一个按键,所以可以同时 
    //有6个按键按下。没有按键按下时,全部返回0。如果按下的键太多,导致键盘扫描系统 
    //无法区分按键时,则全部返回0x01,即6个0x01。如果有一个键按下,则这6个字节中的第一 
    //个字节为相应的键值(具体的值参看HID Usage Tables),如果两个键按下,则第1、2两个 
    //字节分别为相应的键值,以次类推。 
 
 
    //关集合,跟上面的对应 
    0xc0                           // END_COLLECTION

这个报告总共有8字节输入(一个字节ctrl~win键、一个字节led、六个字节普通按键,使用普通键可以一次发送六个),1字节输出。如果只有左ctrl键按下,则返回01 00 00 00 00 00 00 00(十六进制),如果只有数字键1 按下,则返回00 00 59 00 00 00 00 00,如果数字键1 和2 同时按下,则返回00 00 59 5A 00 00 00 00,如果再按下左shift 键,则返回02 00 59 5A 00 00 00 00,然后再释放1 键,则返回02 00 5A 00 00 00 00 00,然后全部按键释放,则返回00 00 00 00 00 00 00 00。这些数据(即报告)都是通过中断端点返回的。当按下Num Lock键时,PC会发送输出报告,从报告描述符中我们知道,Num Lock的LED对应着输出报告的最低位,当数字小键盘打开时,输出xxxxxxx1(二进制,打x的由其它的LED状态决定);当数字小键盘关闭时,输出xxxxxxx0(同前)。取出最低位就可以控制数字键锁定LED了。

报告特征,有些键盘带有一个旋钮调节音量,所以这个键盘既是多媒体类型的HID又是键盘类型HID,但是只需要连接一个HID就能实现这个功能,这是因为报告地图实现了多种HID的配置,为了区分是那种HID以及这种HID的类型,使用报告特征来表示。
基础理论已经到此,下面是ESP32的程序分析。

ESP32程序分析

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/108778.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

离线语音通断器开发-稳定之后顺应新需求

使用云知声的US516p6方案开发了一系列的离线语音通断器,目前已经取得了不小的收获,有1路的,3路的,4路的,唛头和扬声器包括唛头线材也在不断的更新打磨中找到了效果特别好的供应商。 离线语音通断器,家用控…

【Java】HashSet集合用法

目录 HashSet 集合特点 示例代码 手写HashSet集合 HashSet 没有Get() HashSet 集合特点 HashSet 基于HashMap 来实现的,是一个不允许有重复元素的集合HashSet 允许有 null 值HashSet 是无序的,即不会记录插入的顺序HashSet集合实现了Set接口HashSet …

Spring IOC 和 AOP

核心概念 咱们这节就讲完了,在这节中我们讲了两个大概念,一个叫做IOC,一个叫做DI IOC是什么?是用对象的时候不要自己用new而是由外部提供,而spring在进行实现的时候是谁提供,就是IOC容器给你提供。 DI是什…

图神经网络论文笔记(一)——北邮:基于学习解纠缠因果子结构的图神经网络去偏

作者 :范少华 研究方向 :图神经网络 论文标题 :基于学习解纠缠因果子结构的图神经网络去偏 论文链接 :https://arxiv.org/pdf/2209.14107.pdf        https://doi.org/10.48550/arXiv.2209.14107 大多数图神经网络(GNNs)通…

一年一度表白代码(发射爱心)

代码有什么不懂可以私信我 动态画下面的效果图,发射爱心,可改名字 源代码 import turtle import time# 画心形圆弧 def hart_arc():for i in range(200):turtle.righ

帮你快速解锁忘记密码手机的十个工具

将手机解锁到任时候都会让人感觉呼吸新鲜空气。这就像摆脱无形的锁链一样,让您有权选择并避免那些讨厌的限制。但如何解锁手机呢?这就是解锁软件发挥作用的地方。这些方便的工具可以帮助您摆脱束缚并打开一个充满可能性的世界。 解锁手机的合法性 现在&…

Spark UI中Shuffle dataSize 和shuffle bytes written 指标区别

背景 本文基于Spark 3.1.1 目前在做一些知识回顾的时候,发现了一些很有意思的事情,就是Spark UI中ShuffleExchangeExec 的dataSize和shuffle bytes written指标是不一样的, 那么在AQE阶段的时候,是以哪个指标来作为每个Task分区大…

红队专题-Web渗透之资产情报信息收集能力(社工)总结

信息收集 思路框架知识整理 招募六边形战士队员主动信息收集-直接访问[工具]打点收集内容服务器系统版本、域名域名信息收集工具 dnsenumtheHarvesterLayer子域名收集 DiscoverSubdomain子域名信息搜集工具 wydomain目标域名、DNS收集 subDomainsBrute 端口同服旁站/服务/bann…

Unity ScrollView最底展示

Unity ScrollView最底展示 问题方案逻辑 问题 比如在做聊天界面的时候我们肯定会使用到ScrollView来进行展示我们的聊天内容,那么这个时候来新消息的时候就需要最底展示,我认为这里有两种方案; 一种是通过算法每一条预制体的高度*一共多少…

讲述为什么要学习Adobe XD以及 Adobe XD下载安装

首先 我们要了解 Adobe XD 是个什么东西 XD是Adobe公司专门开发出来面向交互、界面设计的矢量绘图工具。 然后是 他可以做什么? 最基本的 可以做UI界面设置 所有 手机 平板 电脑等设备的UI界面 我们都可以通过XD完成 还有就是原型设置 我们可以做各种界面图 还有…

对于构建自定义协议的思考(Java)

工作转眼也1年时间了,回顾历程,协议占了绝大多数 JSON(比较常见的通信文本了),protoBuf(小编有写过教程),自定义协议(字节拼接,在一些iot领域中的标准几乎都…

vue+Fullcalendar

vueFullcalendar: vueFullcalendar项目代码https://gitee.com/Oyxgen404/vue--fullcalendar.git

2.2 消元法的概念

一、消元法介绍 消元法(elimination)是一个求解线性方程组的系统性方法。下面是使用消元法求解一个 2 2 2\times2 22 线性方程组的例子。消元之前,两个方程都有 x x x 和 y y y,消元后,第一个未知数 x x x 将从第…

C#__简单了解XML文档

/* XML(可扩展标记语言):用于传输和存储数据 XML文档:树结构;包含根元素 XML元素:从开始标签到结束标签的部分 XML语法规则: 1、所有XML元素都必须有结束标签 …

大数据与人工智能的未来已来

大数据与人工智能的定义 大数据: 大数据指的是规模庞大、复杂性高、多样性丰富的数据集合。这些数据通常无法通过传统的数据库管理工具来捕获、存储、管理和处理。大数据的特点包括"3V": 大量(Volume):大数…

html2pdf

页面布局时将需要保存在同一页pdf的dom元素用div包裹,并为该div添加class类名,例如.convertPDF,如果有多页创建多个.convertPDF这个div,再循环保存pdf即可 用到了html2canvas和JsPdf这两个插件,自行站内搜索安装

【Java 进阶篇】Java Request 获取请求参数的通用方式详解

在Java Web开发中,获取HTTP请求的参数是一项基本任务。请求参数可以包含在URL中,也可以包含在请求体中,例如表单提交时的参数。在Java中,可以使用HttpServletRequest对象来获取HTTP请求的参数。本文将详细解释如何使用Java获取HTT…

Response Header中不暴露Server(IIS)版本、ASP.NET及相关版本等信息

ASP MVC开发的Web默认情况下会在请求的回应中暴露Server、X-AspNet-Version、X-AspNetMvc-Version、X-Powered-By等相关服务端信息,公开这些敏感信息会存在一定的安全风险。 X-SourceFiles标头用于被IIS / IIS Express中某些调试模块理解,它包含到磁盘上…

Python 接口自动化 —— requests框架

1.前言 Python内置的urllib模块,也可以用于访问网络资源。但是,它用起来比较麻烦,而且,缺少很多实用的高级功能。因此我们使用 requests 模块进行进行接口测试。 requests官方文档资料地址:http://cn.python-request…

不同网段的IP怎么互通

最近在整理工作的时候发现一个不同网段无法互通的问题,就是我们大家熟知的一级路由和二级路由无法互通的问题。由于需要记录整个过程的完整性,这里也需要详细记录下整个过程,明白的人不用看,可以直接跳过,到解决方法去…