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化学工业出版社

时间: 2024-08-24 07:32:26

zigbee 学习笔记的相关文章

ZigBee学习笔记一

初学ZigBee, 记录笔记,交流分享, 如有指教,不胜感激! 1: ZigBee是一种协议,由ZigBee联盟制定. ZigBee联盟是一个高速增长的非牟利业界组织,成员包括国际著名半导体生产商.技术提供者.代工生产商以及最终使用者. 2:所谓协议,即一种通讯方式. 例如,人与人之间的通讯方式是 普通话.普通话就是一种协议. 在日常生活中,网吧局域网,运用一种叫 以太网协议,用于不同计算机之间的交流. 而ZigBee 是用于无线传感器只见交流的一种通讯方式. 3:无线传感(器)网络(WSN):

ZigBee学习笔记第一章《点亮第一个LED》

 作为一个物联网专业的学生,感觉不学好这些物联网相关的知识都对不起自己,恩,学习开始!!!        CC2530有21个数字输入/输出引脚,分别是P01-P07.P10-P17.P20-P24.特别需要注意CC2530管脚的输入输出和51单片机不一样,看图的对比嘛.                                                           恩,这个有个基本了解后我们再去了解自己手里的ZigBee模块的硬件电路连接,这样就可以开始准备写程序了,下图就

[XBee] ZigBee学习笔记

转自:http://blog.csdn.net/wanghanjiett/article/details/6931867 几个重要概念: node(节点): 在zigbee堆栈中最多有三种节点:Coordinator(协调器).Router(路由器).End Device(终端). 其中Coordinator负责选择工作频段,建立网络,允许子设备加入网络:Router负责传递消息,允许子设备加入网络:End Device只负责收发消息. 一个网络中可以有若干Router和End Device,但

vector 学习笔记

vector 使用练习: /**************************************** * File Name: vector.cpp * Author: sky0917 * Created Time: 2014年04月27日 11:07:33 ****************************************/ #include <iostream> #include <vector> using namespace std; int main

Caliburn.Micro学习笔记(一)----引导类和命名匹配规则

Caliburn.Micro学习笔记(一)----引导类和命名匹配规则 用了几天时间看了一下开源框架Caliburn.Micro 这是他源码的地址http://caliburnmicro.codeplex.com/ 文档也写的很详细,自己在看它的文档和代码时写了一些demo和笔记,还有它实现的原理记录一下 学习Caliburn.Micro要有MEF和MVVM的基础 先说一下他的命名规则和引导类 以后我会把Caliburn.Micro的 Actions IResult,IHandle ICondu

jQuery学习笔记(一):入门

jQuery学习笔记(一):入门 一.JQuery是什么 JQuery是什么?始终是萦绕在我心中的一个问题: 借鉴网上同学们的总结,可以从以下几个方面观察. 不使用JQuery时获取DOM文本的操作如下: 1 document.getElementById('info').value = 'Hello World!'; 使用JQuery时获取DOM文本操作如下: 1 $('#info').val('Hello World!'); 嗯,可以看出,使用JQuery的优势之一是可以使代码更加简练,使开

[原创]java WEB学习笔记93:Hibernate学习之路---Hibernate 缓存介绍,缓存级别,使用二级缓存的情况,二级缓存的架构集合缓存,二级缓存的并发策略,实现步骤,集合缓存,查询缓存,时间戳缓存

本博客的目的:①总结自己的学习过程,相当于学习笔记 ②将自己的经验分享给大家,相互学习,互相交流,不可商用 内容难免出现问题,欢迎指正,交流,探讨,可以留言,也可以通过以下方式联系. 本人互联网技术爱好者,互联网技术发烧友 微博:伊直都在0221 QQ:951226918 -----------------------------------------------------------------------------------------------------------------

Activiti 学习笔记记录(三)

上一篇:Activiti 学习笔记记录(二) 导读:上一篇学习了bpmn 画图的常用图形标记.那如何用它们组成一个可用文件呢? 我们知道 bpmn 其实是一个xml 文件

HTML&CSS基础学习笔记8-预格式文本

<pre>标签的主要作用是预格式化文本.被包围在 pre 标签中的文本通常会保留空格和换行符.而文本也会呈现为等宽字体. <pre>标签的一个常见应用就是用来表示计算机的源代码.当然你也可以在你需要在网页中预显示格式时使用它. 会使你的文本换行的标签(例如<h>.<p>)绝不能包含在 <pre> 所定义的块里.尽管有些浏览器会把段落结束标签解释为简单地换行,但是这种行为在所有浏览器上并不都是一样的. 更多学习内容,就在码芽网http://www.