CC2530 中的ZigBee协议栈
CC2530 中的ZigBee协议栈
1.何为协议栈
ZigBee协议栈将各个层的协议集合在一起,并以函数的形式实现,并且向用户提供接口,用户能够直接调用。
它本身就为一个工程。
2. 如何使用
- 开始组网,通过调用协议栈的组网函数等来实现网络的建立以及其他节点的加入网络;
- 发送数据,节点通过调用协议栈的消息发送函数来实现数据的无线发送;
- 接收数据,节点调用协议栈的消息接收函数来实现无线数据接收功能。
这就是它最常用得方法,组网,发送数据,接收数据,在协议栈中都有相应得函数。
3.下载ZigBee协议栈
[官网下载](Z-STACK 驱动程序或库 | 德州仪器 TI.com.cn)
4.例程文件介绍

在下载的zigbee协议栈中随便找一个事例工程打开,打开工程后如上所示。
APP: 为应用层的代码,用户的工作空间就在这。
BDB: zigbee 3.0新增的特性。实现ZigBee BDB(Base Device Behavior,设备基础行为)功能
GP:实现ZigBee GP(Green Power,绿色能源)功能。
HAL:硬件抽象层,存放各种驱动程序。
MAC:媒体介质访问控制,实现物理层通信及IEEE 802.15.4协议。
MT:监视层,为监视协议栈各层的运行状态提供支持。
NWK: ZigBee网络层。
OSAL:操作系统抽象层。
Profile:存放ZigBee标准化定义及相关功能实现的源代码文件。
Security:实现安全相关服务。
Services:提供一些公共的、常用的功能。
Tools:存放工程配置相关的文件。
ZDO:存放ZDO(ZigBee Device Object,ZigBee设备对象)相关源代码文件。
ZMac:属于mac层的内容。
ZMain:存放主函数所在的源代码文件及系统硬件启动相关的源代码文件。
Output:存放工程编译/链接时输出的文件。

ZigBee网络设备类型有3种,分别是Coordinator(协调器),Router(路由器)和EndDevice(终端设备),图中选项卡选项的含义描述如下:
(1)CoordinatorEB : ZigBee协调器。
(2)RouterEB : ZigBee路由器。
(3)EndDeviceEB : ZigBee终端设备。
(4)EndDeviceEB-OTAClient : 支持OTA(Over The Air)空中升级的ZigBee终端设备。
(5)RouterEB-OTAClient : 支持OTA(Over The Air)空中升级的ZigBee路由器。
5 启动流程
5.1 main()
打开ZMain文件夹下的ZMain.c文件,其中有main()函数,为程序的入口函数,程序在这里开始运行

里面有很多初始化函数,下面只说这两个函数:

osal_init_system();//-----------初始化操作系统--------
osal_start_system(); //---------开始任务调度---------
5.2 osal_init_system()
先看osal_init_system()这个函数,先跳转到这个函数里面

来到这个函数,可以看到调用了osalInitTasks()这个函数,接着跳转到osalInitTasks()这个函数,

来到这个函数,可以看到有各个层的初始化代码,并且分给了他们任务ID,下面还会提到这个任务ID,到这每个层的初始化就结束了,每个层都有自己的任务ID。
5.3 osal_start_system()
这个函数轮询任务池的函数,跳转到osal_start_system()中
void osal_start_system( void )
{
#ifdef USE_ICALL
/* Kick off timer service in order to allocate resources upfront.
* The first timeout is required to schedule next OSAL timer event
* as well. */
ICall_Errno errno = ICall_setTimer(1, osal_msec_timer_cback,
(void *) osal_msec_timer_seq,
&osal_timerid_msec_timer);
if (errno != ICALL_ERRNO_SUCCESS)
{
ICall_abort();
}
#endif /* USE_ICALL */
#if !defined ( ZBIT ) && !defined ( UBIT )
//主循环
for(;;)
#endif
{
//系统轮询调度
osal_run_system();
#ifdef USE_ICALL
ICall_wait(ICALL_TIMEOUT_FOREVER);
#endif /* USE_ICALL */
}
}
在osal_start_system()函数的主循环中,循环调用了 osal_run_system()函数,该函数主要工作轮询任务池。osal_run_system()函数的定义OSAL.c文件中,跳转到osal_run_system()中

代码太长,就贴出来一部,需要注意的就下面几行:
do {
if (tasksEvents[idx]) // Task is highest priority that is ready.
{
break;
}
} while (++idx < tasksCnt);
if (idx < tasksCnt)
{
uint16 events;
halIntState_t intState;
HAL_ENTER_CRITICAL_SECTION(intState);
events = tasksEvents[idx];
tasksEvents[idx] = 0; // Clear the Events for this task.
HAL_EXIT_CRITICAL_SECTION(intState);
activeTaskID = idx;
events = (tasksArr[idx])( idx, events );
activeTaskID = TASK_NO_TASK;
HAL_ENTER_CRITICAL_SECTION(intState);
tasksEvents[idx] |= events; // Add back unprocessed events to the current task.
HAL_EXIT_CRITICAL_SECTION(intState);
}
这是检测每个层是否有任务,如果有就去执行这个任务,执行之后,接着轮询,检测是否有任务,idx就是上述提到的任务ID。
do {
if (tasksEvents[idx]) // Task is highest priority that is ready.
{
break;
}
} while (++idx < tasksCnt);
这段代码中tasksEvents[idx]就是之前

申请的空间,用于存放每个层是否有任务的一个事件组之类的,如果有任务,这个层的tasksEvents[idx]就会被置1,读者可以跳转函数看看,
events = tasksEvents[idx];
...
activeTaskID = idx;
events = (tasksArr[idx])( idx, events );
activeTaskID = TASK_NO_TASK;
...
这个就是把任务的标志为给events,然后调用tasksArr [idx] (idx,events),这个函数就会根据idx的值调用相应层的事件处理函数.
6 应用层
每一个层次都有一个对应的任务来处理本层次的事务,例如MAC层对应一个MAC层的任务、网络层对应一个网络层的任务、HAL对应一个HAL的任务,以及应用层对应一个应用层的任务等,这些各个层次的任务构成一个任务池,这个任务池也就是上节课讲到的tasksEvents数组,如图所示。

下面打开APP下的

打开的例程不同,名字可能不同,不过,这无所谓
找到这两个函数
void zclSampleSw_Init( byte task_id )
{
zclSampleSw_TaskID = task_id;
...
MT_UartInit();
MT_UartRegisterTaskID(task_id);
...
#endif
}
uint16 zclSampleSw_event_loop( uint8 task_id, uint16 events )
{
afIncomingMSGPacket_t *MSGpkt;
(void)task_id; // Intentionally unreferenced parameter
if ( events & SYS_EVENT_MSG )
{
while ( (MSGpkt = (afIncomingMSGPacket_t *)osal_msg_receive( zclSampleSw_TaskID )) )
{
switch ( MSGpkt->hdr.event )
{
...
default:
break;
}
// Release the memory
osal_msg_deallocate( (uint8 *)MSGpkt );
}
// return unprocessed events
return (events ^ SYS_EVENT_MSG);
}
// Rejoin Event
if ( events & SAMPLEAPP_REJOIN_EVT )
{
bdb_StartCommissioning(BDB_COMMISSIONING_MODE_NWK_STEERING |
BDB_COMMISSIONING_MODE_FINDING_BINDING );
return ( events ^ SAMPLEAPP_REJOIN_EVT );
}
...
#endif
// Discard unknown events
return 0;
}
每个层都有这两个函数,读者可以找找,看一下,一个初始化函数,一个事件处理函数,里面的参数前面也有提到,看一看。
7 结束
这里只讲了一些启动的大致过程,对于事件发送函数,如何处理函数及发送接收函数未提到,只是想让读者及笔者有一个大致方向,下面推荐一些更为详细的.
推荐
[推荐 一](Zigebee复习_void genericapp_messagemsgcb( afincomingmsgpacket_-CSDN博客)
[推荐二](ZigBee CC2530学习(二)认识协议栈_cc2530协议栈-CSDN博客)
[学习网站](课程简介 - ZigBee 3.0 开发指南 (topthink.com))
CC2530 中的ZigBee协议栈的更多相关文章
- ZigBee协议栈中AES加密算法
原文地址:ZigBee协议栈中AES加密算法作者:大浪淘沙 Z-stack对Zigbee2006提供了全面的支持,功能之强大,性能稳定.安全性高,说到安全性是我们今天的主题.CC2430硬件支持128 ...
- [ZigBee] 16、Zigbee协议栈应用(二)——基于OSAL的无线控制LED闪烁分析(下)
说在前面:上一篇介绍了无线LED闪烁实现的OSAL部分,本篇介绍如何实现无线数据收发及数据处理: 上一篇是用SI跟着流程查看源码,我个人认为以架构的思维去了解代码能让人更清晰 ::ZMain.c程序入 ...
- [ZigBee] 15、Zigbee协议栈应用(一)——Zigbee协议栈介绍及简单例子(长文,OSAL及Zigbee入门知识)
1.Zigbee协议栈简介 协议是一系列的通信标准,通信双方需要按照这一标准进行正常的数据发射和接收.协议栈是协议的具体实现形式,通俗讲协议栈就是协议和用户之间的一个接口,开发人员通过使用协议栈来使用 ...
- 第1章 ZigBee协议栈初始化网络启动流程
作者:宋老师,华清远见嵌入式学院讲师. ZigBee的基本流程:由协调器的组网(创建PAN ID),终端设备和路由设备发现网络以及加入网络. 基本流程:main()->osal_init_sys ...
- Zigbee协议栈--Z-Stack的使用
使用方法简介:一般情况下用户只需要额外添加三个文件就可以完成一个项目.一个是主文件,存放具体的任务事件处理函数:一个是这个主文件的头文件:另外一个是以Osal开头的操作系统接口文件,是专门存放任务处理 ...
- Zigbee协议栈OSAL层API函数【转载】
OSAL层提供了很多的API来对整个的协议栈进行管理.主要有下面的几类:信息管理.任务同步.时间管理.中断管理.任务管理.内存管理.电源管理以及非易失存储管理.看到这些管理是不是感 ...
- CC2530中串口波特率改为9600时单个数据包来不及接收的解决方案
在调试CC2530过程中发现波特率改为9600时,单个包仅有3个Byte时,接收DMA就会启动 因而数据包被强迫拆分成多个,显然只要将接收DMA启动延时做到足够大即可. 具体修改内容如下图所示: 经过 ...
- cc2530中单片机的通用I/O接口
cc2530中有21个输入/输出引脚. 这些引脚可以设置为通用I/O或者设置为外设I/O.(其实这里的外设还是不太懂到底指什么,网上说输入设备,但是通用I/O也可以输入啊,为什么要弄外设I/O?) 其 ...
- 基于ZigBee的家居控制系统的设计与应用
基于ZigBee的家居控制系统的设计与应用 PPT简介:http://pan.baidu.com/s/1i38PC6D 摘 要 智能家居是未来家居的发展方向,其利用先进的网络技术.计算机技术和无线通 ...
- 三、ZigBee无线网络工具
CC2530概述 CC2530是德州仪器Ti公司用于2.4-GHz IEEE 802.15.4.ZigBee 和 RF4CE 应用的一个真正的片上系统(SoC)解决方案,是作为ZigBee无线传 感网 ...
随机推荐
- 识别主机名和IP地址
文章来源:https://oracle-base.com/articles/misc/identifying-host-names-and-addresses Identifying Host Nam ...
- 简单了解HTTP、Websocket和Netty
前言 伴随着网络的快速发展,网络通讯越来越重要,通讯的快捷.安全.方便影响着用户的体验.本文将探讨这些技术的原理.特点以及在实际应用中的应用场景. 1.HTTTP(超文本传输协议) HTTP是一种传输 ...
- 玩转SpringBoot:动态排除Starter配置,轻松部署
引言 在软件开发中,进行本地单元测试是一项常规且必要的任务.然而,在进行单元测试时,有时需要启动一些中间件服务,如Kafka.Elasticjob等.举例来说,我曾经遇到过一个问题:项目中使用了Red ...
- 2021-10-25 css中零值0后面是否要省略单位
原理 在css中如果值为0,可以省略单位. 在css应用场景中,有可能是多端多人维护.即可能维护的人有A及B及C-,应用场景中有电脑端及手机端及小程序及打印机之类的. 结论 个人认为不要省略单位,不要 ...
- Vue 3 的 setup语法糖到底是什么东西?
前言 我们每天写vue3项目的时候都会使用setup语法糖,但是你有没有思考过下面几个问题.setup语法糖经过编译后是什么样子的?为什么在setup顶层定义的变量可以在template中可以直接使用 ...
- .NET周刊【2月第3期 2024-02-25】
国内文章 4.1k Star!全面的C#/.NET/.NET Core学习.工作.面试指南 https://www.cnblogs.com/Can-daydayup/p/18027117 DotNet ...
- 我的第一个项目(十三) :组件间传值的一些方案(vuex,eventbus,localStorage)
好家伙, 先说一下我的需求,我要组件间传值 1.eventBus 前端兄弟组件传值eventbus无法使用 不报错也不触发,就很奇怪 //eventBus.js import Vue from & ...
- 【Azure Function App】在ADF(Azure Data Factory)中调用 Azure Function 时候遇见 Failed to get MI access token
问题描述 在ADF(Azure Data Factory)中,调用Azure Function App中的Function,遇见了 Failed to get MI access token Ther ...
- C++ //案列-评委打分 //(容器添加 删除 算法排序 随机数 字符串追加)描述:5名选手 ABCDE,10个评委分别对每一位选手打分,去除最高分,去除评委中的 //的最低分,取平均分
1 #include<iostream> 2 #include<string> 3 #include<deque> 4 #include<vector> ...
- Java与sql中的字符串表示
在 Java 中,双引号 "" 用于表示字符串字面量,而单引号 '' 用于表示字符字面量.这意味着在 Java 中,您可以使用双引号来包围包含任意数量字符的字符串,包括零个字符(空 ...