Zigbee协议栈进行数据发送

Zigbee协议栈进行数据发送Zigbee协议栈进行数据发送是调用AF_DataRequest这个函数,该函数会调用协议栈里面与硬件相关的函数最终将数据通过天线发送出去。  afStatus_tAF_DataRequest(afAddrType_t*dstAddr,//目的地址指针                             endPointDesc_t*srcEP, //发送节点的端点描述符

大家好,又见面了,我是你们的朋友全栈君。Zigbee协议栈进行数据发送是调用AF_DataRequest这个函数,该函数会调用协议栈里面与硬件相关的函数最终将数据通过天线发送出去。

   afStatus_t AF_DataRequest(afAddrType_t *dstAddr, //目的地址指针

                             endPointDesc_t *srcEP, //发送节点的端点描述符指针
                             uint16cID,   //ClusID 簇ID号

                             uint16len,  //发送数据的长度

                              uint8*buf,  //指向存放发送数据的缓冲区指针

                             uint8*transID,//传输序列号,该序列号随着信息的发送而增加                                        uint8 options, //发送选项

                             uint8 radius  //最大传输半径(发送的跳数)

                            )

参数1:afAddrType_t *dstAddr该参数包含了目的节点的网络地址、端点号及数据传送的模式,如单播、广播或多播等。

    afAddrType_t 是个结构体如下:

    typedefstruct
   {

     union
    {

      uint16shortAddr;      //用于标识该节点网络地址的变量
     }addr;
    afAddrMode_taddrMode;  //用于指定数据传送模式,单播、多播还是广播
    byteendPoint;          //端点号
   }afAddrType_t;  // 其定义在AF.h中

 因为在Zigbee中,数据包可以单点传送(unicast),多点传送(multicast)或者广播传送,所以必须有地址模式参数。一个单点传送数据包只发送给一个设备,多点传送数据包则要传送给一组设备,而广播数据包则要发送给整个网络的所有节点。因此上述结构体中的afAddrMode_taddrMode 就是用于指定数据传送模式,是个枚举类型,可以设置为以下几个值,

typedef enum
{

  afAddrNotPresent =AddrNotPresent,   //表示通过绑定关系指定目的地址
  afAddr16Bit =Addr16Bit,   //单播发送
  afAddrGroup =AddrGroup,   //组播
  afAddrBroadcast = AddrBroadcast //广播
} afAddrMode_t;

enum
{

  AddrNotPresent = 0,
  AddrGroup = 1,
  Addr16Bit = 2,
  Addr64Bit = 3,
  AddrBroadcast = 15
};
 注意:ZigBee设备有两种类型的地址。一种是64IEEE地址(物理),即MAC地址,另一种是16位网络地址。64位地址使全球唯一的地址,设备将在它的生命周期中一直拥有它。它通常由制造商或者被安装时设置。这些地址由IEEE来维护和分配。16为网络地址是当设备加入网络后由协调器或路由器分配的。它在网络中是唯一的,用来在网络中鉴别设备和发送数据。

参数2:endPointDesc_t *srcEP是发送节点的端点描述符指针,在Zigbee网络中,可以通过网络地址找到某个具体的节点,但是具体到某个节点,还有不同的端口(每个节点上最多可支持240个端口),不同节点的端口间可以相互通信。如节点1的端口1可以给节点2的控制端口1发led控制命令,也可以给节点2的端口2发采集命令,但是同一个节点上的端口的网络地址是相同的,所以仅仅通过网络地址无法区分节点1的端口1是与节点2的哪个端口进行通信,因此在发送数据时不但要指定网络地址,还要指点端口号。

因此得出结论:使用网络地址来区分不同的节点,使用端口号区分同一节点上的端口

typedef struct
{

  byte endPoint;  //端点号 
 byte *task_id;  //哪一个任务的端点号(调用任务的ID).
 SimpleDescriptionFormat_t*simpleDesc;  //描述一个Zigbee设备节点,称为简单设备描述符
 afNetworkLatencyReq_tlatencyReq;      //枚举结构,这个字段必须为nolatencyreqs

} endPointDesc_t;      // 其定义在AF.h中

typedefstruct 
  byte          EndPoint;               //EP 
  uint16        AppProfId;              //应用规范ID 
  uint16        AppDeviceId;           //特定规范ID的设备类型 
  byte         AppDevVer:4;           //特定规范ID的设备的版本 
  byte          Reserved:4;             //AF_V1_SUPPORTusesforAppFlags:4. 
  byte         AppNumInClusters;     //输入簇ID的个数 
  cId_t        *pAppInClusterList;     //输入簇ID的列表 
  byte         AppNumOutClusters;      //输出簇ID的个数 
  cId_t        *pAppOutClusterList;    //输出簇ID的列表 
}SimpleDescriptionFormat_t;    //
 其定义在AF.h中

typedef enum
{

  noLatencyReqs,
  fastBeacons,
  slowBeacons
} afNetworkLatencyReq_t;

参数3 uint16cID 是ClusID簇ID号,一个Zigbee节点有很多属性,一个簇实际上是一些相关命令和属性的集合,在整个网络中,每个簇都有唯一的簇ID,也就是用来标识不同的控制操作的命令号。

参数4 uint16len 是送数据的长度

参数5uint8 *buf  指向发送数据缓冲的指针

参数6uint8 *transID该参数是指向发送序号的指针,每发送一个数据包,该发送序号会自动加1,因此在接收端可以查看接收数据包的序号来计算丢包率。

参数7 uint8options 发送选项,有如下选项 

   #defineAF_FRAGMENTED     0x01 
   #defineAF_ACK_REQUEST     0x10 //要求APS应答,这是应用层的应答,只在直接发送(单播)时使用。
   #defineAF_DISCV_ROUTE     0x20 //总要包含这个选项
   #defineAF_EN_SECURITY    0x40 
   #defineAF_SKIP_ROUTING    0x80 //设置这个选项将导致设备跳过路由而直接发送消息。终点设备将不向其父亲发送消息。在直接发送(单播)和广播消息时很好用。

参数8uint8radius 最大的跳数,取默认值AF_DEFAULT_RADIUS

返回值:afStatus_t类型 枚举型,

typedef enum
{

  afStatus_SUCCESS,
  afStatus_FAILED = 0x80,
  afStatus_MEM_FAIL,
  afStatus_INVALID_PARAMETER
} afStatus_t;

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/145943.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • QuartusII添加IP核[通俗易懂]

    QuartusII添加IP核[通俗易懂]Tools->

    2022年10月15日
    4
  • 进程调度算法;先来先服务调度算法、短作业优先调度算法、时间片轮转调度算法「建议收藏」

    进程调度算法;先来先服务调度算法、短作业优先调度算法、时间片轮转调度算法「建议收藏」一、实验目的和要求1.了解进程调度算法的特点2.掌握进程调度算法,如先来先服务调度算法(firstcomefirstserved,FCFS)、短作业优先调度算法(shotjobfirst,SJF)、时间片轮转调度算法。二、实验内容设计模拟实现FCFS、SJF、时间片轮转调度算法的C语言程序1.FCFS算法:按照作业/进程进入队列的先后顺序进行挑选,先进入的将先进行…

    2022年9月29日
    2
  • 一些非常有价值的资料网站有哪些_比较有价值的网站

    一些非常有价值的资料网站有哪些_比较有价值的网站最近在搜集资料的时候,发现了很多好的资料网站,不忍独享,陆续整理出来,贴这里。我先有一个贴一个,等到攒到足够多的时候,我再重新进行整理:电影类:IMDb:http://www.imdb.com/(美国著名互联网电影资料库(InternetMovieDatabase,简称IMDb)是一个关于电影演员、电影、电视节目、电视明星、电子游戏和电影制作的在线数据库。亚马逊公司旗下网站)烂番茄…

    2025年9月1日
    9
  • 推荐.Net、C# 逆向反编译四大工具利器(请勿用来非法行为)[通俗易懂]

    推荐.Net、C# 逆向反编译四大工具利器(请勿用来非法行为)[通俗易懂]在项目开发过程中,估计也有人和我遇到过同样的经历:运行环境出现了重大Bug亟需解决、或者由于电脑挂了、旧代码覆盖新代码,而在这种情况下,我们不能直接在当前的代码中修改这个Bug然后发布,这会导致更严重的问题,因为相当于版本回退了。还有电脑挂了代码整个都没有,这种情况下我们只能只能利用一些逆向的技巧和工具了来解析在服务器发布好的dll。那么你只是单纯的修改一个.Net程序集中的某个方法或功能,而

    2022年6月22日
    25
  • Navicat 4 激活码_通用破解码

    Navicat 4 激活码_通用破解码,https://javaforall.net/100143.html。详细ieda激活码不妨到全栈程序员必看教程网一起来了解一下吧!

    2022年3月16日
    72
  • 贪心算法——C++实现经典案例——初级

    贪心算法——C++实现经典案例——初级最简单的钱币找零问题 这个问题在我们的日常生活中很普遍 假设 1 元 2 元 5 元 10 元 20 元 50 元 100 元的纸币分别有 c0 c1 c2 c3 c4 c5 c6 张 现在要用这些钱来支付 K 元 至少要用多少张纸币 用贪心算法的思想 很显然 每一步尽可能用面值大的纸币即可 在日常生活中我们自然而然也是这么做的 在程序中已经事先将 Value 按照从小到大的顺序排好 贪心分析 贪心算

    2025年12月7日
    3

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号