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)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • freeradius ad认证(掉证书)

    http://blog.csdn.net/lassewang/article/details/9159543

    2022年4月12日
    55
  • 多级时间轮定时器_时间轮与哈希表定时

    多级时间轮定时器_时间轮与哈希表定时时间轮简述顾名思义,时间轮就像一个轮子,在转动的时候外界会指向轮子不同的区域,该区域就可以被使用。因此只要将不同时间的定时器按照一定的方法散列到时间轮的不同槽(即时间轮划分的区域)之中,就可以实现在运转到某个槽时,进行判断该定时器是否已经到达运行时间(需要判断是由于有的定时器并非在这一圈就需要运行,可能需要后面几圈才会运行。从图中也可以看出,每个槽中的定时器是以(双向)链表…

    2022年9月29日
    2
  • get请求关于url长度过长问题_get请求关于url长度过长问题

    get请求关于url长度过长问题_get请求关于url长度过长问题今天在写一个php相应jsonp请求的功能时,发现当url中包含的请求参数过长时会返回414错误。如下图414Request-URITooLarge414Request-URITooLargenginx在网上查询之后,浏览器和服务器对url长度都有限制,现总结如下。1、IEIE浏览器(MicrosoftInternetExplorer)对url长度限制是2083(2K+53)…

    2022年8月24日
    13
  • linux 修改文件内容 vi命令

    linux 修改文件内容 vi命令vi编辑器是所有Unix及Linux系统下标准的编辑器,介绍一下它的用法和一小部分指令。由于对Unix及Linux系统的任何版本,vi编辑器是完全相同的,因此您可以在其他任何介绍vi的地方进一步了解它。Vi也是Linux中最基本的文本编辑器,学会它后,您将在Linux的世界里畅行无阻。在所有的命令执行前,先按Esc1、vi的基本概念   基本上vi可以分为三种状态,分别是命令模式(

    2022年7月26日
    8
  • pycharm怎么装第三方库jieba_pycharm安装配置教程

    pycharm怎么装第三方库jieba_pycharm安装配置教程一、测试环境测试机型:window10pycharm版本:2020.1.1专业版二、安装步骤1.打开pycharm,点击最上方菜单项中的File,并找到Setings如图:2.点击Setings,在Project中找到自己的项目(1),随后右边的显示框中会显示如图页面,点击ProjectInterpreter(2):3.点击之后就会出现如下图页面,中间表格中的Package表示你已经添加的包,Version表示当前版本,latestversion表示最新的版本。添加新.

    2022年8月25日
    3
  • Java中获取时间戳

    Java中获取时间戳**Java语言中关于三种时间戳的获取之心得**最近项目开发过程中发现了项目中获取时间戳的业务。而获取时间戳有以下三种方式,首先先声明推荐使用System类来获取时间戳,下面一起看一看三种方式。1.System.currentTimeMillis()System类中的currentTimeMillis()方法是三种方式中效率最好的,运行时间最短。开发中如果设计到效率问题,推荐使用此种方式获取。System.currentTimeMillis()2.newDate().getTime()除

    2022年5月3日
    38

发表回复

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

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