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无线传 感网 ...
随机推荐
- 如何基于three.js(webgl)引擎架构,实现3D密集架库房,3D档案室(3d机器人取档、机器人盘点、人工查档、设备巡检)
前言: 这是最好的时代,也是最坏的时代:是充满挑战的时代,也是充满机遇的时代.是科技飞速的时代,也是无限可能的时代. 近年来,人工智能(AI)技术的飞速发展已经席卷了全球,不断突破着技术边界,为各行 ...
- bootstrap与javascript
1.bootstrap依赖 bootstrap依赖javascript类库,jQuery 下载jQuery,在页面上应用jQuery 在页面上应用bootstrap的js类库 <script s ...
- python装饰器保留原有函数名称和属性functools.wraps()
# python装饰器在实现的时候,被装饰后的函数其实已经是另外一个函数了(函数名等函数属性会发生改变),为了不影响,python的functools包中提供了一个叫wraps的decorator来消 ...
- 【图论#02】岛屿系列题(数量、周长、最大面积),flood fill算法的代码实现与优化
岛屿数量 给你一个由 '1'(陆地)和 '0'(水)组成的的二维网格,请你计算网格中岛屿的数量. 岛屿总是被水包围,并且每座岛屿只能由水平方向和/或竖直方向上相邻的陆地连接形成. 此外,你可以假设该网 ...
- 第130篇:BOM(window对象)
好家伙,本篇为<JS高级程序设计>第十二章"BOM"学习笔记 什么是BOM? BOM(Browser Object Model) 是指浏览器对象模型,是用于描述这种 ...
- 【Azure 应用服务】Python fastapi Function在Azure中遇见AttributeError异常(AttributeError: 'AsgiMiddleware' object has no attribute 'handle_async')
问题描述 参考文档"Using FastAPI Framework with Azure Functions", 使用FastAPI 模块在Function中实现API请求.通过V ...
- Taurus.MVC WebMVC 入门开发教程6:路由配置与路由映射
前言: 在本篇 Taurus.MVC WebMVC 入门开发教程的第六篇文章中, 我们将讨论如何配置路由并映射到控制器和操作方法. 路由是决定应用程序如何响应客户端请求的重要组成部分,因此在 Web ...
- redis开启多端口
Centos安装多端口的redis服务 背景 redis默认端口6379,由于开发需要,key有重复.于是另起端口6380. 配置服务过程 1.新建/etc/redis6380.conf,内容如下: ...
- 第142篇:原生js实现响应式原理
好家伙,狠狠地补一下代码量 本篇我们来尝试使用原生js实现vue的响应式 使用原生js,即代表没有v-bind,v-on,也没有v-model,所有语法糖我们都用原生实现 1.给输入框绑个变量 & ...
- git拉项目, 1.新建目录 2 git clone 地址 . (重点最后的点)