zigbee 学习笔记
Posted cynchanpin
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了zigbee 学习笔记相关的知识,希望对你有一定的参考价值。
在德州仪器的站点:http://www.ti.com.cn/tool/cn/z-stack上下载安装zigbee2007协议栈版,我的是ZStack-CC2530-2.3.0-1.4.0。
以下演示一个简单的用zigbee通信的样例:
完毕这个样例须要两个zigbee的模块,一个用作协调器(Coordinator)(起建立zigbee网络和与上位机通信的作用)一个用作终端设备(Enddevice)(起採集数据。增加建立的zigbee网络),这里须要实现的功能是先由协调器建立网络,终端设备增加网络。然后终端设备发送“QSS”给协调器,协调器进行推断,假设接收到QSS,那么闪烁LED2,否则点亮LED2。
删除在App目录中的几个文件,仅仅剩下OSAL_SampleApp.c。加入三个文件Coordinator.c,Coordinator.h,Enddevice.c。
须要在OSAL_SampleApp.c中加入#include "Coordinator.h"。
编译Coordinator时须要在workspace以下的下拉表中选择CoordinatorEB,然后右键单击Enddevice.c,在弹出的对话框中选择Exclude from build,将终端设备的程序排除在编译范围内,相同在编译Enddevice的程序时也须要将Coordinator.c排除在编译范围内。
ZigBee网络传输数据流程:
1、终端节点发送信标请求;
2、协调器已经建立ZigBee无线网络。在ZigBee无线网络中。协调器的地址必然是0x0000,。
3、终端节点发送增加网络请求;
4、协调器对终端节点的增加网络请求作出应答
5、终端节点接收到协调器的应答后,发送数据请求(Data Request)。请求协调器分配网络地址。
6、协调器对终端节点的数据请求作出应答
7、协调器将分配的网络地址发送给终端节点
之后终端节点就使用分配的网络地址和协调器通信了。
OSAL(Operating System Abstraction Layer)操作系统抽象层
每一个应用程序对象执行在不同的port上。因此port的作用就是区分不同的应用程序对象。能够把一个应用程序对象看成为一个任务,因此,须要一个机制来实现任务的切换、同步和相互排斥。这就是OSAL产生的根源。
它的主要功能有:
任务注冊、初始化和启动;
任务间的同步、相互排斥;
中断处理。
储存器分配和管理。
在Coordinator.c中GenericApp_Init是任务的初始化函数。GenericApp_ProcessEvent负责处理传递给此任务的事件,推断事件类型,然后运行对应的事件处理函数。
OSAL负责调度整个任务的执行,假设有事件发生了,则会调用对应的事件处理函数进行处理。
那么事件和任务的事件处理函数是怎样联系起来的呢?
ZigBee的做法是:建立一个事件表,保存各个任务相应的事件,建立还有一个函数表,保存各个任务事件处理函数的地址,然后将这两张表建立某种相应的关系,当某事发生时则查找函数表找到相应的事件处理函数就可以。
Zigbee协议中三个变量最重要:
Uint8 tasksCnt-保存任务的总数;
Uint16 *tackEvents指向事件表的首地址。
pTaskEventHandlerFn taskArr[]-数组中的每一项都是一个函数指针,指向事件处理函数;
Typedef unsigned short (*pTaskEventHandlerFn )(unsigned char task_id,unsigned short event);
总结一下OSAL的工作原理:
通过tasksEvents指针訪问事件表的每一项,假设有事件发生。则查找函数表找到事件处理函数进行处理,处理完后。继续訪问事件表,查看是否有事件发生。无限循环。
能够说OSAL是基于事件驱动的轮询式操作系统。
OSAL的消息队列:
将事件和数据封装成一个消息,将消息发送到消息队列,然后在事件处理函数中就能够使用osal_msg_receive,从消息队列中得到该消息。OSAL维护了一个消息队列。每一个消息都会被放到这个消息队列中去。当任务接收到事件后,能够从消息队列中获取属于自己的消息,然后调用消息处理函数进行对应的处理就可以。
OSAL加入新任务:
OSAL_SampleApp.c中,能够找到数组tasksArr[]和函数osalInitasks。tasksArr[]数组中存放了全部任务的事件处理函数的地址。osalInitasks是OSAL的任务初始化函数。全部任务的初始化工作都在这里完毕,而且自己主动给每一个任务分配一个ID。
因此要加入一个新任务,仅仅须要编写两个函数:
新任务的初始化函数。
新任务的事件处理函数。
Coordinator.h
/*******************************
这里定义一些关于zigbee的设备描叙符的宏
**************************************/
#ifndef COORDINATOR_H
#define COORDINATOR_H
#include "ZComDef.h"//zigbee common definition
#define GENERICAPP_ENDPOINT 10 //port
#define GENERICAPP_PROFID 0x0F04
#define GENERICAPP_DEVICEID 0x0001
#define GENERICAPP_DEVICE_VERSION 0
#define GENERICAPP_FLAGS 0
#define GENERICAPP_MAX_CLUSTERS 1
#define GENERICAPP_CLUSTERID 1
extern void GenericApp_Init(byte task_id);
extern UINT16 GenericApp_ProcessEvent(byte task_id,UINT16 events);
#endif
/*********************************************************************/
Coordinator.c
/*********************************************************************
* INCLUDES
*/
#include "OSAL.h"
#include "ZGlobals.h"
#include "AF.h"
#include "aps_groups.h"
#include "ZDApp.h"
#include "Coordinator.h"//这是新加的
#include "OnBoard.h"
/* HAL *///硬件抽象层
#include "hal_lcd.h"
#include "hal_led.h"
#include "hal_key.h"
#include "MT_UART.h"
#include "MT_APP.h"
#include "MT.h"
/*********************************************************************
* GLOBAL VARIABLES
*/
// This list should be filled with Application specific Cluster IDs.
const cId_t GENERICAPP_ClusterList[GENERICAPP_MAX_CLUSTERS] =
{
GENERICAPP_CLUSTERID
};
const SimpleDescriptionFormat_t GenericApp_SimpleDesc =
{
GENERICAPP_ENDPOINT, // int Endpoint;
GENERICAPP_PROFID, // uint16 AppProfId[2];
GENERICAPP_DEVICEID, // uint16 AppDeviceId[2];
GENERICAPP_DEVICE_VERSION, // int AppDevVer:4;
GENERICAPP_FLAGS, // int AppFlags:4;
GENERICAPP_MAX_CLUSTERS, // uint8 AppNumInClusters;
(cId_t *)GENERICAPP_ClusterList, // uint8 *pAppInClusterList;
0, // uint8 AppNumInClusters;
(cId_t *)NULL // uint8 *pAppInClusterList;
};//简单的设备节点描叙符
endPointDesc_t GenericApp_epDesc;//定义一个节点描叙符
byte GenericApp_TaskID; //任务优先级
byte GenericApp_TransID; //数据发送序列号
typedef struct
{
byte endPoint;
byte *task_id;
SimpleDescriptionFormat_t *simpleDesc;
afNetworkLatencyReq_t latenncyReq;
}ndPointDesc_t;
void GenericApp_MessageMSGCB( afIncomingMSGPacket_t *pckt );//消息处理函数
void GenericApp_SendTheMessage( void );//数据发送函数
static void rxCB(uint8 port,uint8 event);
//任务初始化函数
void GenericApp_Init(byte task_id )
{
halUARTCfg_t uartConfig;
GenericApp_TaskID = task_id;//初始化优先级
GenericApp_TransID = 0;//初始化序列号
// Fill out the endpoint description.//初始化设备节点描叙符
GenericApp_epDesc.endPoint = GENERICAPP_ENDPOINT;
GenericApp_epDesc.task_id = &GenericApp_TaskID;
GenericApp_epDesc.simpleDesc =
(SimpleDescriptionFormat_t *)&GenericApp_SimpleDesc;
GenericApp_epDesc.latencyReq = noLatencyReqs;
// Register the endpoint description with the AF
//注冊设备节点描叙符,仅仅有注冊之后才干够使用OSAL的服务
afRegister( &GenericApp_epDesc );
}
//消息处理函数,完毕对接收数据的处理,固定格式
UINT16 GenericApp_ProcessEvent( uint8 task_id, uint16 events )
{
afIncomingMSGPacket_t *MSGpkt;//定义一个接收消息结构体的指针MSGpkt
if ( events & SYS_EVENT_MSG )
{
MSGpkt=(afIncomingMSGPacket_t *)osal_msg_receive( GenericApp_TaskID );
while ( MSGpkt )
{
switch ( MSGpkt->hdr.event )//对消息进行推断
{
// Received when a messages is received (OTA) for this endpoint
case AF_INCOMING_MSG_CMD://无线数据
GenericApp_MessageMSGCB( MSGpkt );//对数据进行处理
break;
default:
break;
}
// Release the memory释放内存
osal_msg_deallocate( (uint8 *)MSGpkt );
// Next - if one is available再次从消息队列中接收消息
MSGpkt=(afIncomingMSGPacket_t*)osal_msg_receive( GenericApp_TaskID );
}
// return unprocessed events
return (events ^ SYS_EVENT_MSG);
}
return 0;
}
//数据处理函数
void GenericApp_MessageMSGCB( afIncomingMSGPacket_t *pkt )
{
unsigned char buffer[4] = " ";
switch ( pkt->clusterId )
{
case GENERICAPP_CLUSTERID:
osal_memcpy(buffer,pkt->cmd.Data,3);//将接收到的数据复制到buffer区
if((buffer[0] == ‘Q‘) || (buffer[1] == ‘S‘) || (buffer[2] == ‘S‘))
{
HalLedBlink(HAL_LED_2,0,50,500);//闪灯
}
else
{
HalLedSet(HAL_LED_2,HAL_LED_MODE_ON);//点灯
}
break;
}
}
Enddevice.c
/*********************************************************************
* INCLUDES
*/
#include "OSAL.h"
#include "ZGlobals.h"
#include "AF.h"
#include "aps_groups.h"
#include "ZDApp.h"
#include "Coordinator.h"//新增协调器头文件
#include "OnBoard.h"
/* HAL *///硬件抽象层
#include "hal_lcd.h"
#include "hal_led.h"
#include "hal_key.h"
#include "MT_UART.h"
#include "MT_APP.h"
#include "MT.h"
// This list should be filled with Application specific Cluster IDs.
const cId_t GENERICAPP_ClusterList[GENERICAPP_MAX_CLUSTERS] =
{
GENERICAPP_CLUSTERID
};
const SimpleDescriptionFormat_t GenericApp_SimpleDesc =
{
GENERICAPP_ENDPOINT, // int Endpoint;
GENERICAPP_PROFID, // uint16 AppProfId[2];
GENERICAPP_DEVICEID, // uint16 AppDeviceId[2];
GENERICAPP_DEVICE_VERSION, // int AppDevVer:4;
GENERICAPP_FLAGS, // int AppFlags:4;
0,
(cId_t *)NULL, // uint8 *pAppInClusterList;
GENERICAPP_MAX_CLUSTERS, // uint8 AppNumInClusters;
(cId_t *)GENERICAPP_ClusterList // uint8 *pAppInClusterList;
};//简单的设备节点描叙符
endPointDesc_t GenericApp_epDesc;//节点描叙符
byte GenericApp_TaskID; //任务优先级
byte GenericApp_TransID; //数据发送序列号。在协议栈中每发送一个数据包。
//该发送序列号加1,能够用来在数据接收端查看数据包的序列号来计算丢包率
devStates_t GenericApp_NwkState;//记录该设备的状态
void GenericApp_MessageMSGCB( afIncomingMSGPacket_t *pckt );//消息处理函数
void GenericApp_SendTheMessage( void );//数据发送函数
void GenericApp_Init(byte task_id )//任务初始化函数
{
GenericApp_TaskID = task_id;//初始化了任务优先级(任务优先级由协议栈的操作系统OSAL分配)
GenericApp_NwkState = DEV_INIT;//将设备状态初始化为DEV_INT,表示该节点没有连接到ZigBee网络
GenericApp_TransID = 0;//将发送数据包的序列号初始化为0,
// Fill out the endpoint description.对节点描叙符进行的初始化
GenericApp_epDesc.endPoint = GENERICAPP_ENDPOINT;
GenericApp_epDesc.task_id = &GenericApp_TaskID;
GenericApp_epDesc.simpleDesc =
(SimpleDescriptionFormat_t *)&GenericApp_SimpleDesc;
GenericApp_epDesc.latencyReq = noLatencyReqs;
// Register the endpoint description with the AF,注冊节点描叙符
afRegister( &GenericApp_epDesc );
}
//消息处理函数,完毕接收数据的处理
uint16 GenericApp_ProcessEvent( uint8 task_id, uint16 events )
{
afIncomingMSGPacket_t *MSGpkt;//定义一个指向消息结构的指针
if ( events & SYS_EVENT_MSG )
{
MSGpkt = (afIncomingMSGPacket_t *)osal_msg_receive( GenericApp_TaskID );
while ( MSGpkt )
{
switch ( MSGpkt->hdr.event )//推断消息类型
{
// Received when a messages is received (OTA) for this endpoint
case ZDO_STATE_CHANGE:
GenericApp_NwkState = (devStates_t)(MSGpkt->hdr.status);//读取设备节点类型
if(GenericApp_NwkState == DEV_END_DEVICE)
{
GenericApp_SendTheMessage();//发送无线数据
}
break;
default:
break;
}
// Release the memory
osal_msg_deallocate( (uint8 *)MSGpkt );
// Next - if one is available
MSGpkt = (afIncomingMSGPacket_t *)osal_msg_receive( GenericApp_TaskID );
}
// return unprocessed events
return (events ^ SYS_EVENT_MSG);
}
return 0;
}
//发送消息函数
void GenericApp_SendTheMessage(void )
{
unsigned char theMessageData[4] = "QSS";
afAddrType_t my_DstAddr;//目标地址
my_DstAddr.addrMode = (afAddrMode_t)Addr16Bit;//将发送数据模式设置为单播
my_DstAddr.endPoint = GENERICAPP_ENDPOINT;//初始化port
my_DstAddr.addr.shortAddr = 0x00;//协调器的网络地址是固定的,为0x00
AF_DataRequest(&my_DstAddr,&GenericApp_epDesc,//调用AF_DataRequest函数发送数据
GENERICAPP_CLUSTERID,
3,
theMessageData,
&GenericApp_TransID,
AF_DISCV_ROUTE,
AF_DEFAULT_RADIUS);
HalLedBlink(HAL_LED_2,0,50,500);//闪灯
}
參考书目:
王小强 欧阳骏 黄宁淋编著《Zigbee无线传感器网络设计与实现》2012化学工业出版社
以上是关于zigbee 学习笔记的主要内容,如果未能解决你的问题,请参考以下文章
ZigBee案例笔记 -- IAR for 8051工程创建
ZigBee案例笔记 -- IAR for 8051工程创建