ESP32蓝牙系列二:协议和规范

本文详细介绍了ESP32的GAP和GATT接口,包括广播流程、不同类型的广播、GATT中的数据表示、服务和服务特性、以及服务器和客户端的交互操作,重点展示了如何使用API和事件模式进行设备管理和数据传输。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

本文以ESP32的API接口对GAP GATT ATT再做分析

一、GAP

ESP32 BLE 通⽤用访问规范 (GAP) 接口 API 的实现和使⽤用流程, GAP 协议层定义了了 BLE 设备的发现流程,设备管理理和设备连接的建立。

BLE GAP 协议层采⽤用 API 调⽤和事件 (Event) 返回的设计模式,通过事件返回来获取 API在协议栈的处理理结果。当对端设备主动发起请求时,也是通过事件返回获取对端设备的状态。 BLE 设备定义了了四类 GAP ⻆色:

广播者 (Broadcaster):处于这种⻆色的设备通过发送⼴广播 (Advertising) 让接收者发

现⾃己。这种⻆角⾊色只能发⼴播,不能被连接。

观察者 (Observer):处于这种⻆角⾊色的设备通过接收⼴广播事件并发送扫描 (Scan) 请

求。这种⻆角⾊色只能发送扫描请求,不能被连接。

外围设备 (Peripheral):当广播者接受了了观察者发来的连接请求后就会进⼊入这种⻆角

⾊色。当设备进⼊入了了这种⻆色之后,将会作为从设备 (Slave) 在链路路中进⾏行行通信。

中央设备 (Central):当观察者主动进⾏行行初始化,并建⽴立⼀一个物理理链路路时就会进⼊入这角色。这种角色在链路路中同样被称为主设备 (Master)。

1、BLE广播流程

1.1使⽤用 public 地址进行广播
使⽤用 public 地址进⾏行广播时,需要将 esp_ble_adv_params_t 成员 own_addr_type 设置为BLE_ADDR_TYPE_PUBLIC,广播流程图如下:

1.2使用可解析地址进⾏广播

使用可解析地址进⾏广播时,底层协议栈会 15 分钟更更新一次广播地址,需要将esp_ble_adv_params_t 成员 own_addr_type 设置为 BLE_ADDR_TYPE_RANDOM,⼴广播流程图如下:

1.3使用静态随机地址进⾏广播

与使⽤可解析地址进⾏广播一样,使用静态随机地址进行广播也需要将esp_ble_adv_params_t 成员 own_addr_type 设置为 BLE_ADDR_TYPE_RANDOM,⼴广播流程图如下:

2、BLE 广播类型介绍

BLE⼴广播主要有 5 种类型,分别为:可连接可扫描非定向广播 (Connectable scannable undirected event type)、⾼高占空⽐比定向⼴广播 (High duty cycle directed event type)、可扫描⾮非定向⼴广播 (Scannable undirected event type)、不不可连接⾮非定向⼴广播 (Non-connectable undirected event type)、可连接低占空⽐比定向⼴广播 (Connectable low duty cycle directed event type)。

二、GATT

ATT 属性协议规定了了在 BLE 中的最小数据存储单位,而 GATT 规范则定义了如何⽤用特性值和描述符表示⼀一个数据,如何把相似的数据聚合成服务 (Service),以及如何发现对端设备拥有哪些服务和数据。

GATT 规范引进了了特性值的概念。这是由于在某些时候,一个数据可能并不只是单纯的数值,还会带有⼀些额外的信息:

•比如这个数据的单位是什什么?是重量量单位千克 kg、温度单位摄⽒氏度℃,还是其他单位;

•比如希望具体告知对⽅方这个数值的名称,例例如同为温度属性 UUID 下,希望告知对方该数据表示“主卧温度”,另⼀个数据表示“客厅温度”;

• 比如在表示 230000、 460000 等大数据时,可以增加指数信息,告知对⽅方该数据的指数是 10^4,这样仅需在空中传递 23、 46 即可。

因此GATT 规范引进了描述符的概念,每种描述符可以表达一种意思,⽤户可使⽤用描述符,描述数据的额外信息。必需说明的是,每个数据和描述符并⾮非一一对应,即一个复杂的数据可以拥有多个描述符,⽽而一个简单的数据可以没有任何描述符。
数据本身的属性值及其可能携带的描述符,构成了了特性 (Characteristic)* 的概念。数据特性包含以下几个部分:

特性声明 (Characteristic Declaration):主要告诉对⽅方此声明后⾯面跟的内容为特性数值。从当前特性声明开始到下⼀一个特性声明之间的所有句句柄 (Handle) 将构成一个完整的特性。此外,特性声明还包括紧跟其后的特性数值的可写可读属性信息。

特性数值 (Characteristic Value):特性的核⼼心部分,一般紧跟在特性声明后⾯面,承载特性的真正内容。

描述符 (Descriptor)描述符可以对特性进⾏行行进⼀一步描述,每个特性可以有多个描述

符,也可以没有描述符。

BLE 协议中会把⼀些常⽤用的功能定义成一个个的服务 (Service)*,例如把电池相关的特性和行为定义成电池服务 (Battery Service);把⼼心率测试相关的特性和⾏行行为定义成⼼心跳服务(Heart Rate Service);把体重测试相关的特性和⾏行行为定义成体重服务 (Weight Scale Service)。

可以看到,每个服务包含若干个特性,每个特性包含若干个描述符。用户可以根据自己的应用需求选择需要的服务,并组成最终的产品应⽤用。

一个完整服务的特性定义参考如下:

基于 ESP32 IDF 建立 GATT 服务

属性表的结构体规定了了户为了描述⼀个属性⽽需要初始化的元素,通过esp_gatts_attr_db_t 进⾏定义,总结如下:

基于 ESP32 IDF 发现对⽅方设备的服务信息(GATT 客户端)
GATT 客户端需要具有发现对⽅方设备的服务和特性的功能 (Service Discovery)。不同的设备可能会使⽤不同的发现流程,下面以查找对方设备的 GATT 服务为例介绍一下 ESP32 IDF使用的服务发现过程:

首先发现对方所有的 Service 信息,包括 Service 的 UUID 和 Handle 范围

- GATT Service, UUID 0x1801, Handles 0x0001~0x0005

- GAP Service, UUID 0x1800, Handles 0x0014~0x001C

然后在 GATT 的 Handle 范围内 (0x0001~0x0005),继续查找所有的特性 (0x2803)

- 找到特性 "Service Change Characteristic”, Handles 0x0002~0x0003

- 其中 0x0002 对应的是这个特性的特性声明

- 其中 0x0003 对应的是这个特性的特性值

- 所以每个特性⾄至少需要占据 2 个 Handle 的属性

既然 GATT Service 的 Handles 范围是 0x0001~0x0005,所以在 0x0003 后⾯面可能跟有相应的描述符,因此继续从 0x0004 开始查找所有的描述符

- 其中 0x0004 对应的是 “Client Characteristic Configuration” 描述符

- 其中 0x0005 暂时没有任何信息,可能是为这个 Service 预留留的 Handle

三、ATT

BLE 里面的数据以属性 (Attribute)方式存在,每条属性由四个元素组成:

属性句柄 (Attribute Handle):正如我们可以使⽤用内存地址查找内存中的内容⼀一样,

ATT 属性的句句柄也可以协助我们找到相应的属性,例例如第⼀一个属性的句句柄是

0x0001,第二个属性的句句柄是 0x0002,以此类推,最⼤大可以到 0xFFFF。

属性类型 (Attribute UUID):每个数据有⾃自⼰己需要代表的意思,例如表示温度、发射

功率、电池等各种各样的信息。蓝牙组织 (Bluetooth SIG) 对常用的⼀一些数据类型

进⾏了归类,赋予不同的数据类型不同的标识码 (UUID)。例如 0x2A09 表示电池信

息, 0x2A6E 表示温度信息。 UUID 可以是 16 ⽐比特的 (16-bit UUID),也可以是 128 比

特的 (128-bit UUID)。

属性值 (Attribute Value):属性值是每个属性真正要承载的信息,其他 3 个元素都是

为了让对⽅方能够更更好地获取属性值。有些属性的⻓长度是固定的,例例如电池属性

(Battery Level) 的⻓长度只有 1 个字节,因为需要表示的数据仅有 0~100%,⽽而 1 个字

节足以表示 1~100 的范围;⽽而有些属性的⻓长度是可变的,例例如基于 BLE 实现的透

传模块。

属性许可 (Attribute Permissions):每个属性对各⾃自的属性值有相应的访问限制,⽐比

如有些属性是可读的、有些是可写的、有些是可读⼜又可写的等等。拥有数据的⼀一⽅方

可以通过属性许可,控制本地数据的可读写属性。

我们把存有数据(即属性)的设备叫做服务器器 (Server),⽽而将获取别人设备数据的设备叫做客户端 (Client)。下⾯面是服务器和客户端间的常⽤用操作:

客户端给服务端发数据通过对服务器器的数据进行写操作 (Write),来完成数据发送工作。写操作分两种,一种是写⼊入请求 (Write Request),一种是写⼊入命令 (WriteCommand),两者的主要区别是前者需要对⽅方回复响应 (Write Response),⽽而后者不需要对⽅方回复响应。

服务端给客户端发数据,主要通过服务端指示 (Indication) 或者通知 (Notification) 的

形式,实现将服务端更更新的数据发给客户端。与写操作类似,指示和通知的主要区

别是前者需要对⽅方设备在收到数据指示后,进⾏行行回复 (Confirmation)。

客户端也可以主动通过读操作读取服务端的数据

服务器器和客户端之间的交互操作都是通过上述的消息 ATT PDU 实现的。每个设备可以指定自己设备⽀支持的最⼤大 ATT 消息⻓长度,我们称之为 MTU。 ESP32 IDF ⾥里里⾯面规定 MTU 可以设置的范围是 23~517 字节,对属性值的总⻓长度没有做限制。

如果用户需要发送的数据包⻓长度⼤大于 (MTU-3)*,则需要调⽤用准⼊入写入请求 (Prepare Write Request) 来完成数据的写操作。同理理,在读取一个数据时候,如果数据的长度超过 (MTU- 1),则需要通过大对象读取请求 (Read Blob Request) 来继续读取剩余的值。

ESP32是一款基于Espressif Systems的高效、低功耗的物联网开发板,集成了Wi-Fi蓝牙功能。编写蓝牙协议(通常指BLE(Bluetooth Low Energy)或经典的蓝牙)涉及几个关键步骤: 1. **初始化蓝牙模块**: 首先,你需要在代码中初始化ESP32蓝牙模块,通过`esp_ble_init()`函数开启蓝牙功能。 ```cpp #include <esp_bt.h> esp_err_t err = esp_bluedroid_init(); ``` 2. **设置服务特性**: 创建蓝牙服务(Service)特性(Characteristic),这通常是自定义的,并按照蓝牙规范(如GATT)定义数据结构。 ```cpp esp_gatt_service_t service; esp_gatt_char_props_t char_props; esp_gatt_char_desc_t descr; // ...设置属性描述符... err = esp_gatt_create_characteristic(&service, &char_props, &descr); ``` 3. **添加注册**: 将服务服务下的特征添加到全局的服务列表,然后调用`esp_gatt_register_services()`注册它们。 4. **响应蓝牙连接请求**: 当设备连接时,需要处理连接事件并根据需求提供适当的配置服务。 5. **数据传输**: 实现发送(`esp_gatt_send_write_response()`)接收(`esp_gatt_read_cb()`)的数据交互,通常使用GATT协议中的写入、读取等命令。 6. **断开连接处理**: 当蓝牙连接断开时,记得删除已注册的服务,关闭相关的资源。 7. **错误处理**: 每个操作都应包含适当的错误检查处理机制。 ```cpp if (err != ESP_OK) { // 处理错误 } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值