天猫精灵蓝牙mesh协议栈demo解析

简介: 本文档简单主要介绍如何基于天猫精灵蓝牙mesh协议栈进行产品开发,对目前代码中的light demo进行说明。

一、项目代码路径与编译

1.1 项目代码路径
light demo项目工程位于协议栈代码app/example/bluetooth/light目录下。如果需要创建一个新工程,可以在bluetooth目录下参照light工程创建对应的工程文件夹。

1.2 编译指令
天猫精灵蓝牙mesh协议栈编译指令为:aos make bluetooth.project_name@chip_name
light demo的编译指令为aos make bluetooth.light@chip_name。
比如采用telink8250芯片,对应编译指令aos make bluetooth.light@tc825x
或者采用nordic10040芯片,对应编译指令aos make bluetooth.light@pca10040
注:以上编译指令如果对编译环境有要求,需要联系芯片厂商提供对应对编译环境安装包。编译生成对bin文件位于根目录out目录下。

二、应用层接口

基于天猫精灵蓝牙mesh协议栈开发一个产品,原则上对接实现3个软件接口即可完整对接整个蓝牙mesh功能。
需要对接的3个软件接口如下:
• 设备模型定义
• 设备默认组播设置
• 协议栈事件响应
2.1 设备模型定义
根据设备需要实现的具体功能,定义设备模型;demo中定义的灯模型具备开关、亮度调节、色温调节功能,所以定义的设备模型如下:
Primary Element Configuration Server Model

Health Server Model
Generic OnOff Server Model
Lightness Server Model
CTL Server Model
Vendor Server Model

注:上述模型中,Lightness Server依赖于Generic Level Server,CTL Server依赖于CTL Setup Server。
代码如下:
static struct bt_mesh_model element_models[] = {

BT_MESH_MODEL_CFG_SRV(),
BT_MESH_MODEL_HEALTH_SRV(),

MESH_MODEL_GEN_ONOFF_SRV(&g_elem_state[0]),
MESH_MODEL_GEN_LEVEL_SRV(&g_elem_state[0]),
MESH_MODEL_LIGHTNESS_SRV(&g_elem_state[0]),
MESH_MODEL_CTL_SRV(&g_elem_state[0]),
MESH_MODEL_CTL_SETUP_SRV(&g_elem_state[0]),

};

2.2 设备默认组播设置
根据灯品类产品规范,灯需要默认绑定组播地址为0xC000和0xCFFF。
代码如下:

#define DEFAULT_MESH_GROUP1 0xC000
#define DEFAULT_MESH_GROUP2 0xCFFF

void mesh_sub_init(u16_t *p_sub)
{

uint16_t sub_list[CONFIG_BT_MESH_MODEL_GROUP_COUNT];
memset(sub_list, 0, sizeof(sub_list));
#ifdef DEFAULT_MESH_GROUP1
    sub_list[0] = DEFAULT_MESH_GROUP1;
#endif
#ifdef DEFAULT_MESH_GROUP2
    sub_list[1] = DEFAULT_MESH_GROUP2;
#endif
memcpy(p_sub, sub_list, sizeof(sub_list));

}

2.3 协议栈事件响应
该软件接口主要处理协议栈触发的具体事件,代码如下:
void user_event(E_GENIE_EVENT event, void *p_arg)
{

E_GENIE_EVENT next_event = event;

switch(event) {
    case GENIE_EVT_SW_RESET:
    case GENIE_EVT_HW_RESET_START:
        BT_DBG_R("FLASH x5");
        led_flash(5);
        reset_light_para();
        break;
    case GENIE_EVT_SDK_MESH_INIT:
        user_init();
        if (!genie_reset_get_flag()) {
            next_event = GENIE_EVT_SDK_ANALYZE_MSG;
        }
        break;
    case GENIE_EVT_SDK_MESH_PROV_SUCCESS:
        BT_DBG_R("FLASH x3");
        led_flash(3);
        break;
    case GENIE_EVT_SDK_TRANS_CYCLE:
    case GENIE_EVT_SDK_ACTION_DONE:
    {
        S_ELEM_STATE *p_elem = (S_ELEM_STATE *)p_arg;
        _led_set(p_elem->elem_index, p_elem->state.onoff[T_CUR], p_elem->state.actual[T_CUR], p_elem->state.temp[T_CUR]);
        if(event == GENIE_EVT_SDK_ACTION_DONE)
            save_light_state(p_elem);
        break;
    }
    case GENIE_EVT_SDK_INDICATE:
        break;
    case GENIE_EVT_SDK_VENDOR_MSG:
    {
        vendor_model_msg_handle((vnd_model_msg *)p_arg);
        break;
    }
    case GENIE_EVT_HW_RESET_DONE:
        printk("GENIE_EVT_HW_RESET_DONE\n");
        break;
    default:
        break;
}

if(next_event != event) {
    genie_event(next_event, p_arg);
}

}

2.4 协议栈事件说明
GENIE_EVT_SW_RESET
GENIE_EVT_HW_RESET_START
GENIE_EVT_SDK_MESH_INIT
GENIE_EVT_SDK_MESH_PROV_SUCCESS
GENIE_EVT_SDK_TRANS_CYCLE
GENIE_EVT_SDK_ACTION_DONE
GENIE_EVT_SDK_INDICATE
GENIE_EVT_SDK_VENDOR_MSG
GENIE_EVT_HW_RESET_DONE

相关文章
|
3月前
|
编译器 C语言 C++
栈区的非法访问导致的死循环(x64)
这段内容主要分析了一段C语言代码在VS2022中形成死循环的原因,涉及栈区内存布局和数组越界问题。代码中`arr[15]`越界访问,修改了变量`i`的值,导致`for`循环条件始终为真,形成死循环。原因是VS2022栈区从低地址到高地址分配内存,`arr`数组与`i`相邻,`arr[15]`恰好覆盖`i`的地址。而在VS2019中,栈区先分配高地址再分配低地址,因此相同代码表现不同。这说明编译器对栈区内存分配顺序的实现差异会导致程序行为不一致,需避免数组越界以确保代码健壮性。
44 0
栈区的非法访问导致的死循环(x64)
232.用栈实现队列,225. 用队列实现栈
在232题中,通过两个栈(`stIn`和`stOut`)模拟队列的先入先出(FIFO)行为。`push`操作将元素压入`stIn`,`pop`和`peek`操作则通过将`stIn`的元素转移到`stOut`来实现队列的顺序访问。 225题则是利用单个队列(`que`)模拟栈的后入先出(LIFO)特性。通过多次调整队列头部元素的位置,确保弹出顺序符合栈的要求。`top`操作直接返回队列尾部元素,`empty`判断队列是否为空。 两题均仅使用基础数据结构操作,展示了栈与队列之间的转换逻辑。
|
5月前
|
网络协议
为何UDP协议不可靠?DNS为何选择UDP?
总的来说,UDP和TCP各有优势,选择哪种协议取决于应用的具体需求。UDP可能不如TCP可靠,但其简单、快速的特性使其在某些场景下成为更好的选择。而DNS就是这样的一个例子,它利用了UDP的优势,以实现快速、高效的名字解析服务。
267 14
|
8月前
|
存储 C语言 C++
【C++数据结构——栈与队列】顺序栈的基本运算(头歌实践教学平台习题)【合集】
本关任务:编写一个程序实现顺序栈的基本运算。开始你的任务吧,祝你成功!​ 相关知识 初始化栈 销毁栈 判断栈是否为空 进栈 出栈 取栈顶元素 1.初始化栈 概念:初始化栈是为栈的使用做准备,包括分配内存空间(如果是动态分配)和设置栈的初始状态。栈有顺序栈和链式栈两种常见形式。对于顺序栈,通常需要定义一个数组来存储栈元素,并设置一个变量来记录栈顶位置;对于链式栈,需要定义节点结构,包含数据域和指针域,同时初始化栈顶指针。 示例(顺序栈): 以下是一个简单的顺序栈初始化示例,假设用C语言实现,栈中存储
331 77
|
7月前
|
存储 缓存 网络协议
DNS协议详解
通过本文,您可以全面了解DNS协议的各个方面,从而更好地理解和应用这一重要的互联网基础服务。
1482 44
|
6月前
|
编解码 监控 网络协议
RTSP协议规范与SmartMediaKit播放器技术解析
RTSP协议是实时流媒体传输的重要规范,大牛直播SDK的rtsp播放器基于此构建,具备跨平台支持、超低延迟(100-300ms)、多实例播放、高效资源利用、音视频同步等优势。它广泛应用于安防监控、远程教学等领域,提供实时录像、快照等功能,优化网络传输与解码效率,并通过事件回调机制保障稳定性。作为高性能解决方案,它推动了实时流媒体技术的发展。
320 5
|
7月前
|
算法 调度 C++
STL——栈和队列和优先队列
通过以上对栈、队列和优先队列的详细解释和示例,希望能帮助读者更好地理解和应用这些重要的数据结构。
152 11
|
7月前
|
网络协议 Linux Go
自己动手编写tcp/ip协议栈1:tcp包解析
学习tuntap中的tun的使用方法,并使用tun接口解析了ip包和tcp包,这是实现tcp/ip协议栈的第一步。
152 15
|
7月前
|
DataX
☀☀☀☀☀☀☀有关栈和队列应用的oj题讲解☼☼☼☼☼☼☼
### 简介 本文介绍了三种数据结构的实现方法:用两个队列实现栈、用两个栈实现队列以及设计循环队列。具体思路如下: 1. **用两个队列实现栈**: - 插入元素时,选择非空队列进行插入。 - 移除栈顶元素时,将非空队列中的元素依次转移到另一个队列,直到只剩下一个元素,然后弹出该元素。 - 判空条件为两个队列均为空。 2. **用两个栈实现队列**: - 插入元素时,选择非空栈进行插入。 - 移除队首元素时,将非空栈中的元素依次转移到另一个栈,再将这些元素重新放回原栈以保持顺序。 - 判空条件为两个栈均为空。
|
8月前
|
存储 C++ 索引
【C++数据结构——栈与队列】环形队列的基本运算(头歌实践教学平台习题)【合集】
【数据结构——栈与队列】环形队列的基本运算(头歌实践教学平台习题)【合集】初始化队列、销毁队列、判断队列是否为空、进队列、出队列等。本关任务:编写一个程序实现环形队列的基本运算。(6)出队列序列:yzopq2*(5)依次进队列元素:opq2*(6)出队列序列:bcdef。(2)依次进队列元素:abc。(5)依次进队列元素:def。(2)依次进队列元素:xyz。开始你的任务吧,祝你成功!(4)出队一个元素a。(4)出队一个元素x。
236 13
【C++数据结构——栈与队列】环形队列的基本运算(头歌实践教学平台习题)【合集】

推荐镜像

更多
  • DNS