ESP-AT 系列: BLE client 的 scan、创建和连接-程序员宅基地

技术标签: ESP32  AT  AT 指令 (ESP-AT)  BLE  GATT  Profile  

AT 工程:https://github.com/espressif/esp-at
AT 文档:https://docs.espressif.com/projects/esp-at/zh_CN/latest/

一、基础知识

想要进行 BLE相关的开发,我们必须具备一定的基础知识,当然基础知识肯定是非常简单的,仅罗列些和本文介绍指令相关的知识。

ESP32 里的蓝牙协议栈是符合蓝牙4.2协议规范的, 本文的描述也仅针对蓝牙4.2规范。

1. BLE 协议架构

BLE 协议栈中,大致分为这几个层级:PHYLLHCIL2CAPSMATTGATT。其中 PHYLL 我们一般称之为 controllerHCI 之上的称之为 Host,而 HCI 是中间的接口层。

下面是对这几个层所负责的事的大致描述,如果需要了解更细致的信息,需要去蓝牙官网下载详细的 spec

  • PHY: 主要是射频相关的,BLE 有 40 个信道,从 2402MHz - 2480MHz

  • LL: 链路层,RF 控制层,用于控制设备的射频状态,控制芯片工作在 standby(准备)、advertising(广播)、scanning(监听/扫描), initiating(发起连接)、connected(已连接) 这五个状态中的一种。

  • HCI:主机控制接口层,通信层,向 hostcontroller 提供一个标准化的接口。该层可以由软件 API 实现或者使用硬件接口 uartspiusb 来控制。

  • L2CAP:逻辑链路控制及自适应协议层,相当于快递,将数据打包,为上层提供数据封装服务,可以让客户点对点的通信。

  • SM:安全管理层,提供配对和密钥的分发,实现安全连接和数据交换。

  • ATT:属性协议层,ATT 环境中,允许设备向另外一个设备展示一块特定的数据,称之为“属性”,展示“属性” 的设备称为服务器,与之配对的设备称为客户端。链路层状态(主机和从机)与设备的 ATT 角色是相互独立的,也就是说,主机设备可以是 ATT 服务器,也可以是 ATT 客户端,从机也一样。

  • GATT:通用属文件健配置层,从名字就能看出,GATT 是在 ATT 上面的一层结构,定义了使用 ATT的服务框架,GATT 规定了配置文件(鼎鼎有名的 profile)的结构,在 BLE 中,所有被 profile 或者服务用到的数据块都称为“特性, characteristic”两个建立连接的设备之间的所有数据通信都是通过 GATT 子程序处理,应用程序和 profile 直接使用 GATT 层,在后面具体的代码中,我们会经常见到 GATT,数据交互也是再GATT 层。

2. BLE 角色划分

BLE 协议栈,不同的层级里面有不同的角色划分:

  • LL:可以把设备分为主机和从机,从机广播,主机发起连接;

  • GAP:可以把设备分为中心设备和外围设备;

  • GATT:可以把设备分为服务端和客户端;

我们需要记住一点就是,这些划分相互是不受影响的,在这里,我们只讨论 GATT 层的角色。

GATT 其实是一种属性传输协议,简单的讲可以认为是一种属性传输的应用层协议,这个属性的结构其实非常简单,其实也就是由一个个 services 组成,每个 service 又由数量不等的 characteristic 组成,每个characteristic 又由很多其它的元素组成,就如下图所示。在这里,我们不进一次描述,只需要有个概念,我们会在其它文档中进一步介绍。

GATT 中,有 ServerClient 之分:

  • Server 属性数据库的存储的地方

  • Client 会发起与 server 的连接, 发现 service 上面的属性数据库。

GATT serverGATT client 这两种角色存在的阶段则是建立连接之后,根据对话地位的不同进行区分的,很容易理解的是,保有数据的那一方我们称之为 GATT server,访问数据的那一方我们称之为 GATT client

二、应用场景介绍

在这里,我们还用一个简单的例子进行说明GATT serverGATT client的工作方式,比如手机 APP和小米手环。

手机 APP 和小米手环建立连接之前,一般我们都会打开手机的蓝牙功能,然后打开手机 APP,用手机 APP 搜索周边的蓝牙设备,搜索的结果可能不止一个,但是手环应该是在设备列表里的,这个过程中手机负责扫描的任务,两者建立了GATT 连接后,手机扮演的角色就是 GATT client,然后,手机这里需要做服务发现,看看小米手环那里有哪些服务,手机可以从手环中读取各种等传感器数据(比如步数和心跳),两者交互的数据是保存在手环中的,此时手环就是 GATT server

三、用 ESP32-AT 实现GATT Client

从上面的知识背景和应用场景介绍中,可以了解到 GATT client 主要就是干这些事情:

  • 扫描设备
  • 建立连接
  • 服务发现
  • 数据传输

当然除了上面提到的,client 还会干很多其它的事情,比如:

  • 更新连接参数
  • 更新 MTU Szie
  • 发起加密

不过这些操作,在本文不会涉及,会在下面的文档中有进一步的介绍。

AT 指令的操作介绍:

1. 初始化

首先要做的就是初始化 BLE 的角色,在这里,角色就是指 GATT 层的角色。

AT+BLEINIT=1

OK

AT+BLEINIT 这条指令用来初始化 GATT 角色, 参数可以设置为 0,1,2

  • 0: 关闭 BLE
  • 1: 初始化为 client
  • 2: 初始化为 server

2. 扫描周边设备

AT+BLESCAN=1,3

OK

AT+BLESCAN 这条命令可以用来开启和关闭 BLE 扫瞄,第一个参数 1 代表开启扫描,如果是 0,代表关闭扫描。

第二个参数是可选参数,如果是关闭扫描,第二个参数不必输入,如果是开启扫描,第二个参数是扫描时间,这个时候缺省第二个参数或者设置为0,都代表持续扫描下去。

扫描的时候,是不阻塞的,随时可以下命令停止扫描。

3. 发起连接

AT+BLECONN=0,"12:34:56:78:90:12"

OK

第一个参数是连接的序列号,由用户随意设置,第二个参数是想要连接的对端设备地址,这里可以通过上面那条扫描的命令的得到。

如果用户想断开连接,可以调用这条命令:

AT+BLEDISCONN=0

OK

这里只需要输入连接序列号一个参数即可,如果忘记了连接的序号,可以查询:

AT+BLECONN?

OK

4. 服务发现

AT+BLEGATTCPRIMSRV=0

OK

这里的参数 0 代表连接的序列号,表示用户想去查那个 server 上的服务(主要服务)。

用户可以拿一块 ESP32 的模组,烧录 ESP32-AT 的固件,试一下。

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/espressif/article/details/81707792

智能推荐

python实现日历功能_Python如何绘制日历图和热力图-程序员宅基地

文章浏览阅读206次。本文以2019年全国各城市的空气质量观测数据为例,利用matplotlib、calmap、pyecharts绘制日历图和热力图。在绘图之前先利用pandas对空气质量数据进行处理。数据处理从网站下载的数据为逐小时数据,每天一个文件。如果要绘制全年的日历图或者热图,首先要将所有的数据进行合并处理。下载好数据之后,将数据解压到当前目录的2019文件夹内,然后处理数据:import globfrom d..._python绘制日历图4-6月份

Git 项目管理快速入门_git项目管理-程序员宅基地

文章浏览阅读451次。Git 项目管理快速入门,当安装完git之后应该做的第一件事就是设置用户名称与邮件地址。因为每一个git的提交都会使用这些信息,并且它会写入到你的每一次提交中,不可更改。TIP- 版本库又名为仓库,英文名repositiry- 可以简单理解为一个目录,这个目录里面的所有文件都可以被git管理起来,每个文件的修改、删除,git都能跟踪,以便任何时刻都可以追踪历史,或者再将某个时刻可以“还原”。_git项目管理

memmove函数详解 看这一篇就够了-C语言(函数讲解、函数实现、使用用法举例、作用、自己实现函数 )-程序员宅基地

文章浏览阅读1.7w次,点赞20次,收藏65次。memmove函数详解 看这一篇就够了-C语言(函数讲解、函数实现、使用用法举例、作用、自己实现函数 )_memmove

JVM内存池_jvm memory pool-程序员宅基地

文章浏览阅读1.3k次,点赞2次,收藏4次。JVM内存池根据jconsole工具提供,内存池大致可分为堆 年轻代   survivor space   eden space 老年代   old Gen非堆 Metaspace(起始部分 compressed Class space) codeCache结合《内存概述与JAVA进程内存》,进一步得出运行时数据区的组成通过《JVM内存模型》我们对运行时数据有了初步的..._jvm memory pool

VUE 使用 IView 实现Table Tree进行异步数据加载_iview table tree-程序员宅基地

文章浏览阅读2.3k次。VUE 使用 IView 实现Table Tree进行异步数据加载页面布局 <Table ref="thisTab" row-key="ID" :load-data="handleLoadData" :columns="columnsList" :data="familyList" :highlight-row="true" :disabled-hover=._iview table tree

matlab飞机飞行模型,使用 MATLAB/Simulink 设计无人机飞行控制系统(二、动力学模型建立)...-程序员宅基地

文章浏览阅读8.7k次,点赞5次,收藏110次。前面一篇文章已经讲解了如何获取无人机的气动参数,本篇文章讲解如何建立无人机动力学模型。建立无人机动力学模型前面我们用 MATLAB 绘制了飞机的几何外形,并调用 DATCOM计算了无人机的气动参数。我们现在看一看 DATCOM 输出的 .out 文件是什么样的。我们又该怎么利用它呢?通过 DATCOM 计算得到的气动参数文件这是一个非常复杂的文件,里面记录了我们之前定义的各种飞行包线内的工作点上,..._大飞机数学模型matlab

随便推点

Kubernetes入门——Longhorn简介-程序员宅基地

文章浏览阅读3.9k次。Longhorn是由Rancher创建的一款云原生的、轻量级、可靠且易用的开源分布式块存储系统。部署到K8s集群上之后,Longhorn会自动将集群中所有节点上可用的本地存储聚集为存储集群,然后利用这些存储管理分布式、带有复制功能的块存储,支持快照和数据备份。_longhorn

abcde依次进入一个队列_程序员考试复习题-程序员宅基地

文章浏览阅读1.4k次。程序员习题1)经过以下栈运算后,x的值是_____________。InitStack(s); Push(s,a); Push(s,b); Pop(s,x);GetTop(s,x);A. aB. bC. 1D. 02) 经过以下栈运算后,StackEmpty(s)的值是___________。InitStack(s); Push(s,a); Push(s,b); Pop(s,x); Pop(s,y..._经过以下队列运算后,队头的元素是( )。

白嫖是真香,耗时半年,终于整理完成这套Java架构进阶pdf_java架构 限流框架-程序员宅基地

文章浏览阅读143次。2020年的“金九银十”就这么来了,也到了该发福利的阶段了,花了我大半个月时间收整了全套的「Java架构进阶pdf」,这一波下来,刷完你就会知道,真真香啊,我的心血果然,没白费!请注意:关于全套的「Java架构进阶pdf」,我会从面试-筑基-框架-分布式-微服务-调优的等六个方面展开,顾及篇幅,只截图展示大概内容Java架构进阶之面试篇面试这一块,分别有基础、中级、高级等三部分,一步一步检测。基础面试中级开发面试高级面试Java架构进阶之技能导图篇筑基.._java架构 限流框架

阿里云数据库RDS备份恢复_rds.tar.gz恢复到阿里云rds-程序员宅基地

文章浏览阅读452次。阿里云数据库RDS备份恢复.准备本地环境需于线上数据库RDS版本一样。.去阿里云下载RDS备份wget -c ‘<备份文件外网下载地址>’ -O <自定义文件名>.tar.gz(登录阿里云>>云数据库RDS>>实例备份>>管 理>>备份恢复>>点击下载获取下载地址)-c:启用断点续传模式。-O:将下载的结果保存为指定的文件(使用URL中包含的文件后缀 .tar.gz 、.xb.gz 或 _qp.xb)。按照_rds.tar.gz恢复到阿里云rds

【F28335】第15章 串行通讯接口SCI_f28335 sci的空闲线多处理模式-程序员宅基地

文章浏览阅读933次,点赞21次,收藏22次。整理书本内容,梳理F28335的SCI模块相关知识。_f28335 sci的空闲线多处理模式

VB 函数 CallByName 的一些用法 (函数动态调用,回调函数)_vba callbyname函数-程序员宅基地

文章浏览阅读6.4k次。VB 函数 CallByName (函数动态调用,回调函数) 的一些用法最近编程,要用到函数指针来动态地调用函数,在C 语言里很容易实现,用函数指针就行了。VB里就没有这么方便了,查阅相关资料,讲起来均很复杂,也不容易明白,其中采用CallByName (函数动态调用,回调函数)是最方便的方法。将之归纳为几个例子,应用方法就一目了然了。 附:Visual B_vba callbyname函数

推荐文章

热门文章

相关标签