BLE Mesh蓝牙组网技术详细解析之Access Layer访问层(六)

目录

一、什么是BLE Mesh Access Layer访问层?

二、Access payload

2.1 Opcode

三、Access layer behavior

3.1 Access layer发送消息的流程

3.2 Access layer接收消息的流程

3.3 Unacknowledged and acknowledged messages

3.3.1 Unacknowledged message

3.3.2 Acknowledged message

3.4 Example message sequence charts

3.4.1 Acknowledged Get

3.4.2 Acknowledged Set

3.4.3 Unacknowledged Set

3.4.4 Acknowledged set with periodic publishing

四、资料获取


一、什么是BLE Mesh Access Layer访问层?

BLE Mesh Access Layer是蓝牙Mesh协议栈的一部分,它主要负责以下几个方面的功能

  • 定义应用数据的格式,例如Opcode和参数字段。
  • 定义并控制在上层传输层中执行的应用数据的加密和解密,使用AppKey或DeviceKey作为密钥。
  • 在将数据上传到模型层之前,对来自上层传输层的数据进行验证,判断其是否适用于该网络和应用,例如检查地址和AppKey是否匹配。
  • 在将数据下发到上层传输层之前,根据模型层的指令,填充源地址、目的地址、TTL等字段,以及设置重传参数。

二、Access payload

Field Name

Size (octets)

Notes

Opcode

1, 2, or 3

Operation Code

Parameters

0 to 379

Application Parameters

  • Opcode:1字节、2字节或3字节的操作码,用于标识消息的类型和含义。
  • Parameters:一个可变长度的参数字段,用于携带消息的具体内容。参数字段的长度由Opcode和上层传输层的PDU的长度共同决定,最大为379字节,取决于Opcode的长度

一个access payload最多可以发送32个片段,每个片段12字节。这意味着最大值 包括TransMIC在内的字节数是384

对于4字节的TransMIC,access payload最大大小是380字节,因此对于单个字节的操作码,参数字段最多可以达到379字节。对于2字节的操作码,参数字段最多可以有378个字节。对于一个3字节的操作码,parameters字段最多可以有377个字节。

传输层可以将消息分割成多段PDU,以便在网络层上传输。下表显示了根据包的数量和TransMIC的大小而定的最大有用的应用包大小。

Number of Packets

Maximum useful access payload size (octets)

32 bit TransMIC

64 bit TransMIC

1

11 (unsegmented)

n/a

1

8 (segmented)

4 (segmented)

2

20

16

3

32

28

n

(n×12)-4

(n×12)-8

32

380

376

2.1 Opcode

  • 1字节操作码:由SIG定义,用于标准的模型消息,例如Generic OnOff Set,Generic Level Get等。1字节操作码的第一位为0,剩下的7位表示具体的操作码值。
  • 2字节操作码:由SIG定义,用于扩展的模型消息,例如Scene Store,Scene Delete等。2字节操作码的第一位为1,第二位为0,剩下的14位表示具体的操作码值。
  • 3字节操作码:由供应商定义,用于自定义的模型消息,例如Vendor Model Status,Vendor Model Indication等。3字节操作码的第一位为1,第二位为1。其中的2个字节被指定为厂商ID(CID),该部分在表中以“z”表示。在整个mesh网络中,每个厂商ID最多支持64个厂商操作码。用户可用的只有6位(xxxxxx),因此总共是64个操作码。

Opcode Format

Notes

0xxxxxxx (excluding 01111111)

1-octet Opcodes

01111111

Reserved for Future Use

10xxxxxx xxxxxxxx

2-octet Opcodes

11xxxxxx zzzzzzzz zzzzzzzz

3-octet Opcodes

三、Access layer behavior

3.1 Access layer发送消息的流程

  • 首先,访问层接收到来自模型层的消息,消息包含操作码(Opcode)和有效载荷(Payload)。操作码是用于标识消息的类型和含义的一个字节或多个字节的值,有效载荷是消息的主要内容,包含了模型的状态值、参数、属性等信息。
  • 然后,访问层根据下发来的模型ID找到对应的存储信息,包括源地址(SRC)、目的地址(DST)、应用密钥(AppKey)等。源地址是该模型对应的元素的单播地址,目的地址是该模型设置的发布地址,应用密钥是该模型绑定的用于加密和解密的密钥。
  • 接着,访问层将消息和存储信息一起传递给上层传输层,上层传输层会根据应用密钥对消息进行加密和认证,生成一个传输层数据单元(Upper Transport PDU),并为每个消息分配一个序列号(SEQ)用于防护中继/重放攻击。
  • 然后,上层传输层将传输层数据单元和其他信息传递给下层传输层,下层传输层会根据消息的长度决定是否需要对消息进行分段,如果需要分段,会生成多个分段的传输层数据单元(Segmented Upper Transport PDU),并为每个分段分配一个分段索引(SEG)用于重组消息。
  • 接着,下层传输层将分段的传输层数据单元和其他信息传递给网络层,网络层会根据网络密钥(NetKey)对分段的传输层数据单元进行加密和混淆,生成一个网络层数据单元(Network PDU),并为每个网络层数据单元分配一个网络标识符(IVI)用于识别网络。
  • 然后,网络层将网络层数据单元和其他信息传递给承载层,承载层会根据承载类型(Advertising Bearer或GATT Bearer)对网络层数据单元进行封装,生成一个承载层数据单元(Bearer PDU),并通过底层的BLE协议栈将承载层数据单元发送出去。
  • 最后,承载层会根据消息的类型(Acknowledged Message或Unacknowledged Message)决定是否需要对消息进行重传,如果需要重传,会根据重传参数(重传次数、重传间隔等)进行重传,直到收到应答消息或达到重传次数上限为止。
/*源自开源协议栈NimBLE*/
static int model_send(struct bt_mesh_model *model,struct bt_mesh_net_tx *tx, bool implicit_bind,struct os_mbuf *msg,const struct bt_mesh_send_cb *cb, void *cb_data)
{BT_DBG("net_idx 0x%04x app_idx 0x%04x dst 0x%04x", tx->ctx->net_idx,tx->ctx->app_idx, tx->ctx->addr);BT_DBG("len %u: %s", msg->om_len, bt_hex(msg->om_data, msg->om_len));if (!bt_mesh_is_provisioned()) {BT_ERR("Local node is not yet provisioned");return -EAGAIN;}if (net_buf_simple_tailroom(msg) < 4) {BT_ERR("Not enough tailroom for TransMIC");return -EINVAL;}if (msg->om_len > BT_MESH_TX_SDU_MAX - 4) {BT_ERR("Too big message");return -EMSGSIZE;}if (!implicit_bind && !model_has_key(model, tx->ctx->app_idx)) {BT_ERR("Model not bound to AppKey 0x%04x", tx->ctx->app_idx);return -EINVAL;}return bt_mesh_trans_send(tx, msg, cb, cb_data);
}int bt_mesh_model_send(struct bt_mesh_model *model,struct bt_mesh_msg_ctx *ctx,struct os_mbuf *msg,const struct bt_mesh_send_cb *cb, void *cb_data)
{struct bt_mesh_net_tx tx = {.sub = bt_mesh_subnet_get(ctx->net_idx),.ctx = ctx,.src = bt_mesh_model_elem(model)->addr,.xmit = bt_mesh_net_transmit_get(),.friend_cred = 0,};return model_send(model, &tx, false, msg, cb, cb_data);
}

3.2 Access layer接收消息的流程

  • 首先,访问层从承载层接收到来自底层低功耗蓝牙协议栈的消息,消息包含操作码(Opcode)和有效载荷(Payload)。操作码是用于标识消息的类型和含义的一个字节或多个字节的值,有效载荷是消息的主要内容,包含了模型的状态值、参数、属性等信息。
  • 然后,访问层根据消息中的网络密钥索引(NetKeyIndex)和应用密钥索引(AppKeyIndex)找到对应的网络密钥(NetKey)和应用密钥(AppKey),并将消息传递给上层传输层。上层传输层会根据网络密钥和应用密钥对消息进行解密和认证,还原出原始的操作码和有效载荷。
  • 接着,访问层根据消息的目的地址(DST)字段,找到当前节点中订阅了该地址或者元素地址为该地址的所有模型,再判断该消息与该模型是否绑定了同一个应用密钥,满足了要求再上报至模型层,模型再进行操作码和有效载荷的检查。
/*BLE Mesh访问层接收数据子函数*/
/*源自开源协议栈NimBLE*/
void bt_mesh_model_recv(struct bt_mesh_net_rx *rx, struct os_mbuf *buf)
{struct bt_mesh_model *models, *model;const struct bt_mesh_model_op *op;u32_t opcode;u8_t count;int i;BT_DBG("app_idx 0x%04x src 0x%04x dst 0x%04x", rx->ctx.app_idx,rx->ctx.addr, rx->ctx.recv_dst);BT_DBG("len %u: %s", buf->om_len, bt_hex(buf->om_data, buf->om_len));if (get_opcode(buf, &opcode) < 0) {BT_WARN("Unable to decode OpCode");return;}BT_DBG("OpCode 0x%08x", (unsigned) opcode);for (i = 0; i < dev_comp->elem_count; i++) {struct bt_mesh_elem *elem = &dev_comp->elem[i];struct net_buf_simple_state state;/* SIG models cannot contain 3-byte (vendor) OpCodes, and* vendor models cannot contain SIG (1- or 2-byte) OpCodes, so* we only need to do the lookup in one of the model lists.*/if (BT_MESH_MODEL_OP_LEN(opcode) < 3) {models = elem->models;count = elem->model_count;} else {models = elem->vnd_models;count = elem->vnd_model_count;}op = find_op(models, count, opcode, &model);if (!op) {BT_DBG("No OpCode 0x%08x for elem %d", opcode, i);continue;}if (!model_has_key(model, rx->ctx.app_idx)) {continue;}if (!model_has_dst(model, rx->ctx.recv_dst)) {continue;}if (buf->om_len < op->min_len) {BT_ERR("Too short message for OpCode 0x%08x", opcode);continue;}/* The callback will likely parse the buffer, so* store the parsing state in case multiple models* receive the message.*/net_buf_simple_save(buf, &state);op->func(model, &rx->ctx, buf);net_buf_simple_restore(buf, &state);}
}

3.3 Unacknowledged and acknowledged messages

  • 消息的类型:消息可以分为需要应答的消息(Acknowledged Message)和不需要应答的消息(Unacknowledged Message)。需要应答的消息要求接收端回复一个状态消息作为应答,不需要应答的消息则不需要回复。
  • 消息的目标地址:消息可以发送到单播地址(Unicast Address)、组播地址(Group Address)或虚拟地址(Virtual Address)。目标地址决定了消息的发布和订阅方式,以及消息的中继和重传机制。
  • 消息的操作码(Opcode):操作码是用于标识消息的类型和含义的一个字节或多个字节的值。操作码的长度和值由消息的第一位和第二位决定,操作码的值可以是预定义的或者厂商自定义的。
  • 消息的有效载荷(Payload):有效载荷是消息的主要内容,包含了模型的状态值、参数、属性等信息。有效载荷的长度和格式由操作码和模型定义决定。

3.3.1 Unacknowledged message

unAcknowledged Message是一种不需要接收端回复应答的消息。它的定义如下

  • unAcknowledged Message的操作码(Opcode)的第一位(bit 7)必须为0,表示操作码的长度为一个字节。
  • unAcknowledged Message的有效载荷(Payload)的长度和格式由操作码和模型定义决定,通常包含了目标状态(Target State)或查询参数(Query Parameters)等信息。
  • unAcknowledged Message的发送端(Client Model)会在发送消息后不等待接收端(Server Model)的应答消息(Status Message),而是直接结束消息发送过程。

3.3.2 Acknowledged message

Acknowledged Message是一种需要接收端回复一个状态消息作为应答的消息。它的定义如下

  • Acknowledged Message的操作码(Opcode)的第一位(bit 7)必须为0,表示操作码的长度为一个字节。
  • Acknowledged Message的有效载荷(Payload)的长度和格式由操作码和模型定义决定,通常包含了目标状态(Target State)或查询参数(Query Parameters)等信息。
  • Acknowledged Message的发送端(Client Model)会在发送消息后等待一段时间(Acknowledged Message Timeout),如果在这段时间内没有收到接收端(Server Model)的应答消息(Status Message),则会重发消息,直到收到应答消息或达到重发次数上限(Acknowledged Message Retransmissions)为止。

3.4 Example message sequence charts

3.4.1 Acknowledged Get

Acknowledged Get是一种用于从服务器端请求状态信息的消息类型。它属于Acknowledged MSG的一种,也就是说,当服务器端收到这个消息后,需要回复一个对应的Status消息,告诉客户端当前的状态值。这样可以保证客户端和服务器端的状态同步,也可以避免消息丢失或重复的问题。

例如,如果客户端想要知道一个灯的亮度,它可以发送一个Generic Level Get消息,这是一种Acknowledged Get消息,它的Opcode是0x8205。服务器端收到这个消息后,会回复一个Generic Level Status消息,它的Opcode是0x8206,它的Payload包含了当前的亮度值。

3.4.2 Acknowledged Set

Acknowledged Set是一种用于向服务器端发送状态设置请求的消息类型。它也属于Acknowledged MSG的一种,也就是说,当服务器端收到这个消息后,需要回复一个对应的Status消息,告诉客户端状态是否设置成功。这样可以保证客户端和服务器端的状态同步,也可以避免消息丢失或重复的问题。

例如,如果客户端想要设置一个灯的亮度,它可以发送一个Generic Level Set消息,这是一种Acknowledged Set消息,它的Opcode是0x8207。它的Payload包含了要设置的亮度值和一个可选的Transition Time和Delay参数,用于控制灯的渐变效果。服务器端收到这个消息后,会回复一个Generic Level Status消息,它的Opcode是0x8206,它的Payload包含了当前的亮度值。

3.4.3 Unacknowledged Set

Unacknowledged Set是一种用于向服务器端发送状态设置请求的消息类型。它不属于Acknowledged MSG,也就是说,当服务器端收到这个消息后,不需要回复一个对应的Status消息,也不会把当前状态的改变后的结果通过Publish地址向四周广播。这样可以减少网络层的负载,提高传输效率,但也可能导致客户端和服务器端的状态不同步,或者消息丢失或重复的问题。

例如,如果客户端想要设置一个灯的亮度,它可以发送一个Generic Level Set Unacknowledged消息,这是一种Unacknowledged Set消息,它的Opcode是0x8208。它的Payload包含了要设置的亮度值和一个可选的Transition Time和Delay参数,用于控制灯的渐变效果。服务器端收到这个消息后,不会回复任何消息,也不会广播当前的亮度值。

3.4.4 Acknowledged set with periodic publishing

Acknowledged set with periodic publishing是一种用于向服务器端发送状态设置请求并启用周期性状态发布的消息类型。它属于Acknowledged MSG的一种,也就是说,当服务器端收到这个消息后,需要回复一个对应的Status消息,告诉客户端状态是否设置成功,并且按照设定的时间间隔和发布地址,定期广播当前的状态值。这样可以保证客户端和服务器端的状态同步,也可以让其他订阅了发布地址的节点获取服务器端的状态信息。

例如,如果客户端想要设置一个灯的亮度,并且让灯每隔5秒钟发布一次当前的亮度值,它可以发送一个Generic Level Set消息,这是一种Acknowledged Set消息,它的Opcode是0x8207。它的Payload包含了要设置的亮度值和一个可选的Transition Time和Delay参数,用于控制灯的渐变效果。它还需要配置灯的发布地址和发布周期,比如发布地址为0xC000(所有节点的组播地址),发布周期为5秒。服务器端收到这个消息后,会回复一个Generic Level Status消息,它的Opcode是0x8206,它的Payload包含了当前的亮度值。同时,服务器端会每隔5秒钟向0xC000地址发送一个Generic Level Status消息,让其他节点知道当前的亮度值

四、资料获取

通过点击以下链接,您可以获取BLE Mesh模块原理图、源代码以及开发资料。链接地址将为您提供详细的文件资料,以供您进行参考和使用。

如果您在使用过程中遇到任何问题或疑虑,欢迎加我QQ ,一起探讨技术问题,我的QQ号是986571840,加的时候请注明CSDN。

BLE Mesh蓝牙组网模块 - 硬创社 (jlc.com)icon-default.png?t=N7T8https://x.jlc.com/platform/detail/001d23cba7b64b0d9df5b9b69720fadb

感谢各位用户点赞、分享、在看,这些行为让知识得以更加广泛地传播,从而让更多人受益。

请在转载作品时注明出处,严禁抄袭行为。
 

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

python3 批量创建zabbix主机

一、简介 此程序是python调用zabbix API 批量创建监控主机的脚本。所有格式参考zabbix 官网API。地址如下&#xff1a; https://www.zabbix.com/documentation/6.0/zh/manual/api/reference二、创建zabbixAPI包 1.config.py 其中Get_token函数是为了获取访问zabbix API所需…

java内存屏障

参考&#xff1a;https://blog.csdn.net/weixin_73077810/article/details/132804522 内存屏障主要用于解决线程的可见性、有序性问题 代码案例&#xff1a; ReentrantLock保证可见性的原理 在 lock.lock() 和 lock.unlock() 时&#xff0c;都会操作 AbstractQueuedSy…

设置gazebo内sdf,urdf文件路径的可能变量

IGN_GAZEBO_RESOURCE_PATH/home/actorsun/ws/install/ros_gz_sim_demos/share 输入printenv |grep -i ros 其中-i 表示忽略大小写

算法训练营Day39(动态规划)

62.不同路径 62. 不同路径 - 力扣&#xff08;LeetCode&#xff09; class Solution {public int uniquePaths(int m, int n) {//1dp数组 m n代表位置&#xff0c;dp[m][n]代表到达这里的途径个数int [][] dp new int[m][n];//3初始化for(int i 0;i<n;i){dp[0][i] 1;}f…

SqueezeNet:通过紧凑架构彻底改变深度学习

一、介绍 在深度学习领域&#xff0c;对效率和性能的追求往往会带来创新的架构。SqueezeNet 是神经网络设计的一项突破&#xff0c;体现了这种追求。本文深入研究了 SqueezeNet 的复杂性&#xff0c;探讨其独特的架构、设计背后的基本原理、应用及其对深度学习领域的影响。 在创…

【Python机器学习】线性模型——用于多分类的线性模型

很多线性分类模型只使用与二分类问题&#xff0c;将二分类算法推广到多分类算法的一种常见方法是“一对其余”方法。在“一对其余”方法中&#xff0c;对每个类别都学习一个二分类模型&#xff0c;将这个类别和其他类别尽量区分&#xff0c;这样就生成了与类别数相同的二分类模…

Spring之循环依赖底层源码(一)

文章目录 一、简介1. 回顾2. 循环依赖3. Bean的生命周期回顾4. 三级缓存5. 解决循环依赖的思路 二、源码分析三、相关问题1. Async情况下的循环依赖解析2. 原型Bean情况下的循环依赖解析3. 构造方法导致的循环依赖解析 一、简介 1. 回顾 前面首先重点分析了Spring Bean的整个…

力扣刷题记录(29)LeetCode:695、1020、130

695. 岛屿的最大面积 这道题和计算岛屿周长类似&#xff0c;在这里dfs的功能就是由一块陆地出发&#xff0c;找出这块陆地所在的岛屿并返回岛屿面积。 class Solution { public:int dfs(vector<vector<int>>& grid,int i,int j){if(i<0||i>grid.size())…

JavaScript获取后端json数据创建表格

怎么在前端获取后端数据生成表格json $.ajax({url: /Resource/GetResource,data: { searchText: searchText },success: function (response) {/*searchResult.innerHTML response;*/console.log(输入框 &#xff1a;, response);// 假设你有一个具有 id 为 "tableContai…

表格封装之 useForm 封装

在日常开发中&#xff0c;后端管理系统中增删改查的开发大多是重复机械式的工作&#xff0c;为了减少解放自己的双手&#xff0c;我们可以对这部分工作的内容进行封装。 一般的增删改查页面&#xff0c;由顶部搜索区&#xff0c;中部表格区&#xff0c;底部功能区&#xff08;…

Unity 面试篇|(二)Unity基础篇 【全面总结 | 持续更新】

目录 1.Unity3d脚本从唤醒到销毁有着一套比较完整的生命周期&#xff0c;列出系统自带的几个重要的方法。2.Unity3D中的碰撞器和触发器的区别&#xff1f;3.物体发生碰撞的必要条件&#xff1f;4.简述Unity3D支持的作为脚本的语言的名称&#xff1f;5. .Net与Mono的关系&#x…

镜头选型和计算

3.5 补充知识 一、单像元分辨率&#xff08;单像素精度&#xff09; 单像素精度是表示视觉系统综合精度的指标&#xff0c;表示一个像元对应检测目标的实际物理尺寸&#xff0c;是客户重点关注的 视觉系统参数&#xff1b; 计算公式1&#xff1a;单像素精度视野范围FOV/相机分辨…

Unity 点击对话系统(含Demo)

点击对话系统 可实现点击物体后自动移动到物体附近&#xff0c;然后弹出对话框进行对话。 基于Unity 简单角色对话UI脚本的编写&#xff08;新版UI组件&#xff09;和Unity 关于点击不同物品移动并触发不同事件的结合体&#xff0c;有兴趣可以看一下之前文章。 下边代码为U…

【数据库原理】(11)SQL数据查询功能

基本格式 SELECT [ALL|DISTINCT]<目标列表达式>[,目标列表达式>]... FROM <表名或视图名>[,<表名或视图名>] ... [ WHERE <条件表达式>] [GROUP BY<列名 1>[HAVING <条件表达式>]] [ORDER BY <列名 2>[ASC DESC]];SELECT: 指定要…

QT上位机开发(文本编辑器的界面开发)

【 声明&#xff1a;版权所有&#xff0c;欢迎转载&#xff0c;请勿用于商业用途。 联系信箱&#xff1a;feixiaoxing 163.com】 文本编辑器是编程开发中经常使用到的一个软件&#xff0c;比如说notepad就是其中一种。这里说编写一个文本编辑器&#xff0c;并不是说真的要写一个…

Linux 部署 AI 换脸

我使用的系统是 Ubuntu 20.04 文章实操主要分为以下几个部分 1、python 环境安装 2、下载 FaceFusion 上传服务器 3、创建 python 虚拟环境 4、下载 FaceFusion 依赖&#xff08;这里的命令执行时间会很长&#xff0c;够你睡午觉了&#xff09; 5、运行 FaceFusion 6、开…

python基础—网络编程

网络基本协议 TCP协议 UDP协议 二者对比&#xff1a; 连接性&#xff1a; TCP是面向连接的协议&#xff0c;需要在传输数据之前先进行三次握手建立连接。而UDP是无连接的协议&#xff0c;可以直接发送数据&#xff0c;无需事先建立连接。 可靠性&#xff1a; TCP提供了数…

js数组元素的排序

JavaScript 中的数组可以使用多种方法进行排序。下面是一些常见的排序方法&#xff1a; sort() 方法 sort() 方法用于对数组的元素进行排序。默认情况下&#xff0c;sort() 方法将数组元素转换为字符串&#xff0c;然后按照字符的 Unicode 码点进行排序。这可能导致一些不符合…

Golang拼接字符串性能对比

g o l a n g golang golang的 s t r i n g string string类型是不可修改的&#xff0c;对于拼接字符串来说&#xff0c;本质上还是创建一个新的对象将数据放进去。主要有以下几种拼接方式 拼接方式介绍 1.使用 s t r i n g string string自带的运算符 ans ans s2. 使用…

如何将手机中termux用电脑的vnc显示

在电脑中我们同样需要下载 vnc 这里填写手机上的 IP&#xff1a;端口号 我的是 10.11.166.219:5902 下面填名字然后 手机端 输入sshd开始ssh这边就可以连接啦