当前位置: 首页 > news >正文

zigbee笔记:十、ZStack(2.3.0-1.4.0)的OSAL使用分析

        zigbee笔记:九中,我们已经学会了利用模板,定制自己的个性开发工程,本文为协议栈(ZStack-CC2530-2.3.0-1.4.0)代码使用分析笔记,来进一步掌握协议栈的使用。

一、协议栈使用知识点

        1、协调器、路由器、终端代码下载

        通过选项卡选择不同的节点类型工程(分别是协调器、路由节点和终端节点工程),就会导致参与编译的配置文件不同,配置不同就会导致部分宏定义不同,宏定义不同就会导致工程里某些宏的值不同。这些宏值的不同会影响程序执行的流程,就会导致出现不同的功能,,如下图所示。

        在选项卡里每一种版本模块都貌似有2个分为不带Pro和带Pro。zigbee协议在发展过程中经历了3个版本,其中zigbee在7出的是最新的规范这个规范有2个版本 zigbee2007和 zigbeePro他们之间的区别是,功能基本相同,Pro版本功能稍微强大咱们一般都选 Pro 。

        2、信道选择与网络PANID设置

        在工程配置文件f8wConfig.cfg中, 参数-DDEFAULT_CHANLIST和-DZDAPP_CONFIG_PAN_ID。

        (1)-DDEFAULT_CHANLIST 参数

        信道编码采用64位独热码,当需要多信道通信时,可以采用或运算,如编码为0x00001800表示采用11和12信道,把结果作为 DEFAULT CHANLIST值。

        路由器和终端:可以在参与或操作的这些信道上选择一个相对于我来说最佳的网络,加入进去。

        协调器: 可以在参与或操作的这些信道上选择一个最佳的信道并在这个信道上创建自己的Zigbee网络()。

        (2)-DZDAPP_CONFIG_PAN_ID 参数

 -DZDAPP_CONFIG_PAN_ID值非0xFFFF时:

         路由器和终端:必须要加入到PANID为参数值这样一个Zigbee无线局域网。

        协调器:我要创建一个网络,并且把这个参数值作为这个网络的PANID。

-DZDAPP_CONFIG_PAN_ID值为0xFFFF时:

        路由器和终端:在加入网络的时候没有PANID的限制。

        协调器:可以随机生成一个值,把这个随机值作为这个网络的PANID。

(3)补充:

        问1:   当2个模块下载相同的协调器代码,并且指走的PANID参数值为非0xFFFF,后果会怎样?

       答: 先上电的模块可以创建0xFFFB这样-个zigbee网络,后上电的模块创建一个在DXFFF8基础上加1的网络。

        问2:网络创建成功后,协调器的网络短地址固定为0x0000

        3、引脚初始化准备

        启动osal之前main函数中还初始化调用了TI的一些硬件实验,其中很多引脚的功能已经被初始化为片上外设了,所以我们使用时要自己修改,不能漏步骤。如果添加自己的一些外设初始化要放在osal_start_system函数前面,WatchDogEnable之后,如初始化LED,如果缺少配置为普通IO模式这一步骤,可能会导致LED无法使用。

        4、OSAL操作系统

        OSAL(Operating System Abstraction Layer),能够实现任务调度的微操作系统,在OSAL操作系统中非常重要的两个概念是任务和事件,还有消息也比较重要

(1)OSAL的启动流程

        如下图所示,系统启动后,先完成一系列的初始化,然后进入如无轮询主循环。

        在zstack协议栈工程目录ZMAIN下的ZMain.c包含了主程序的入口地址

        (2) 两个重要的函数

        main函数中完成了一系列的初始,其中对开发者比较重要有两个函数

        1)OSAL初始化函数 osal_init_system()

       函数osal_init_system() 其中包含的任务初始化函数osalInitTasks(),完成系统任务和用户自定义任务

         函数osalInitTasks()包含的任务初始化函数完成系统任务和用户自定义任务

        2)系统启动函数 osal_start_system(),其功能是不断查询任务事件数组,如果有事件产生,则根据事件在taskArr数组中查询任务事件处理函数进行处理。从tasksEvents数组中tasksEvent[0](0号任务事件变量)去读一直读到tasksEvent[8],如果发现哪一个任务事件变量值为非0,就会跳出循环,此时的idx值就是它的任务ID。跳出上面的循环后,此时就会将idx中存的任务ID和event中存的任务事件变量值,通过函数指针的方法调用相应的任务事件处理函数。

        (3) OSAL(操作系统抽象层)原理

        1)应用层是一个任务,它有一个系统分配给它的数值唯一的编号叫做任务ID。

        2)任务可以处理事件,处理事件的这些代码都在一个函数里,这个函数叫做应用层任务事件处理函数。应用层任务还有一个2个字节的变量,这个变量叫做任务事件变量,如果事件变量和某个事件的宏值&操作为1,那么表示应用层任务将要处理这个事件。

        3)系统在运行的时候,会不断地读取应用层任务事件变量,如果应用层任务事件变量为非0,就会认为应用层任务有事件需要处理,此时就会调用应用层任务事件处理函数StarryApp_ProcessEvent,并将任务事件变量的值传给events,在处理函数中会将events分别与应用层定义的所有任务事件宏值进行与操作,如果值为1,就会去执行这个事件处理的相应代码;当发现任务事件变量为0的时候,就会认为应用层任务没有事件需要处理。

       4) 函数 uint8 osal_set_event( uint8 task_id, uint16 event_flag ) 为事件产生函数,调用执行某个任务中的某个事件, para1是任务ID,para2是任务中的某个事件宏。

        本质是,把taskID(应用层任务)的任务事件变量中对应于宏event_flag值为1的那个位变为1,这样的话首先这个任务事件变量就变成了非0,系统在检测到任务事件变量为非0,就会去调用任务事件处理函数***App_ProcessEvent。在这个函数里,任务事件变量肯定和任务宏的值与操作为1,所以必定会执行这个事件相关的代码。

        5)在ZStack里按照代码功能来划分,分成不同的层,比如,硬件操作相关硬件层,网络相关的代码叫网络层,自己写应用程序部分叫应用层。几乎每一个层都是一个任务,系统为每一个任务分配一个字节的唯一数值编号, 我们把这个数值编号叫做任务ID(task_id)。每一个任务都能处理一些他们能够处理的事物,叫做事件(event_flag)。TI在设计的时候把任务事件处理函数和任务事件变量都封装成了数组:

        函数数组={任务事件处理函数A,任务事件处理函数B,任务事件处理函数C...}

        变量数组={任务事件变量A,任务事件变量B,任务事件变量C...}

        6)任务ID,系统不是随便分配的,我们可以通过这个值立即找到这个任务自己的任务事件处理函数和任务事件变量。任务ID就是数组的下标,通过下标就可以找到相应的任务事件处理函数和任务事件变量。由函数osalInitTasks()包含的任务初始化函数完成系统任务和用户自定义任务,详见 4(2)。

        7)事件是系统需要处理的事情,必须有对应的任务进行处理,事件采用16位独热码,即每种任务可以有16种事件(event_flag)。例如,在应用层,系统事件已被定义,占用编码为0x8000,还有15种用户可定义的事件。

        8)所有Zstack协议栈在稳定工作时,它的行为是在for(;;){}死循环里不断去读所有任务的任务事件变量。如果发现所有任务的任务事件变量都为0,没问题,说明所有任务都没有事件将要去处理;不断重复这个过程,直到发现,某个任务事件变量不为0,就会通过不为0的这个任务ID找到这个任务事件处理函数,去处理相应的事件。

        9) 当事件数量较多,编码无法表示时,可采用消息,系统事件就是采用消息进行事件传递的。 OSAL任务与事件的关系 ,如下图所示。

        10) 消息*******************************************

        4、事件处理函数的编写

        以应用层任务的事件为例, 事件处理函数要在xxApp.c里的xx应用事件处理函数中编写(xxApp_ProcessEvent),如下图所示。

        5、函数总结:

        (1)uint8 osal_set_event( uint8 task_id, uint16 event_flag );

        事件产生函数,调用执行某个任务中的某个事件, para1是任务ID,para2是任务中的事件

        (2)uint8 osal_start_timerEx( uint8 task_id, uint16 event_flag,unit16 timeout_value );

        事件定时器触发函数,等待一定时间执行某个任务中的某个事件,para1是任务编号,para2是参数是要处理的事务,para3是是时间(毫秒)。

        (3)uint8 osal_msg_send( uint8 destination_taskID, uint8 *msg_ptr );

发送消息的函数,para1是要接收此Message的目标任务ID,para2是所要发送的消息Message指针。

        (4)MSGpkt =(afIncomingMSGPacket t *)osal_msg_receive( SampleApp TasklD );

        接收消息的函数,para1是任务ID的参数,用于指定从哪个任务的消息队列中接收消息。osal_msg_receive函数从任务的消息队列中取出一个消息,该消息是一个指向afIncomingMSGPacket_t 类型的指针,然后将其强制类型转换为 afIncomingMSGPacket_t 类型(osal_msg_receive()函数返回的是一个指向 void 类型的指针),最后将其赋值给 MSGpkt 变量,这个操作通常在一个任务中用于接收其他任务发送的消息,以便进行进一步的处理。

    

二、常见注意和错误解决

        1、编译时Error[Pa045]: **** has no prototype ,prototypes取消勾选

        在协议栈中添加使用自己写程序的时候,编译时要把设置中的prototypes取消勾选。

        2、编译基于协议栈代码生成hex文件

        (1)首先设置选项,如下图所示

        (2)取消f8w2530.xcl文件内的部分代码,如下图所示

        3、 Warning[w2]: Symbol ?PBANK is redefined in command-line

        (1) IAR  编译zigbee工程时 报的警告如下:

         (2)可以在配置文件中修改,Tools文件夹下的f8w2530.xcl中屏蔽掉-D?PBANK=93这一行。

        (3)重新全部编译

        4、Warning[Pe223]: function **** declared implicitly 

        这个告警是提示我,我调用的这个 **** 函数没有进行函数声明,一般出现这种情况就是没有调用该函数声明的头文件,或者是没有对其进行外部声明。 解决办法

        1、将使用的函数的头文件包含进去。

        2、对其使用关键字"extern"声明

extern void ****(void);

        3、还有一种就是如果只在一个文件中调用函数,就将这个函数定义移到最前面也可以解决。

参考链接

(1)lesson7上协议栈的使用_1_哔哩哔哩_bilibili

(2)ZigBee协议栈 -- Zstack协议栈(Zstack2.5.1a)-CSDN博客

(3)zigbee之Zstack协议栈使用_z-stack协议栈-CSDN博客

(4)lesson7-3 OSAL操作系统抽象层原理_osal运行机制-CSDN博客

(5)lesson7-2 Zigbee协议栈的使用_zigbee协议栈如何创建一个新任务-CSDN博客

相关文章:

  • 北京网站建设多少钱?
  • 辽宁网页制作哪家好_网站建设
  • 高端品牌网站建设_汉中网站制作
  • Spark SQL Catalyst工作流程
  • 如何高效记录并整理编程学习笔记?
  • 【论文阅读】Far3D: Expanding the Horizon for Surround-view 3D Object Detection
  • Apache SeaTunnel 2.3.5 Zeta-Server集群环境搭建与使用
  • 【安卓】调用摄像头和相册
  • Ubuntu18.04 配置EtherCAT主站IGH SOEM
  • css实现水滴效果图
  • MPU6050详细介绍
  • 初始化React Native项目
  • 独立开发者系列(42)——MYSQL语句使用和进阶
  • SQL Server 2022的游标
  • 20240813在荣品RK3588S-AHD开发板的预置Android13中挂载ext4格式的256GB的TF卡
  • 【智能家居设备集成】智能家居设备的集成与互联互通
  • Revit二次开发_使用InnoSetup打包插件
  • Vue2.0 项目实战篇\部署篇
  • 「前端」从UglifyJSPlugin强制开启css压缩探究webpack插件运行机制
  • 2018天猫双11|这就是阿里云!不止有新技术,更有温暖的社会力量
  • ECMAScript 6 学习之路 ( 四 ) String 字符串扩展
  • JavaScript 奇技淫巧
  • 百度地图API标注+时间轴组件
  • 从tcpdump抓包看TCP/IP协议
  • 给初学者:JavaScript 中数组操作注意点
  • 回顾 Swift 多平台移植进度 #2
  • 设计模式 开闭原则
  • 我与Jetbrains的这些年
  • 想晋级高级工程师只知道表面是不够的!Git内部原理介绍
  • 硬币翻转问题,区间操作
  • ionic入门之数据绑定显示-1
  • 分布式关系型数据库服务 DRDS 支持显示的 Prepare 及逻辑库锁功能等多项能力 ...
  • ​​​​​​​ubuntu16.04 fastreid训练过程
  • ​LeetCode解法汇总518. 零钱兑换 II
  • ​Linux·i2c驱动架构​
  • # Apache SeaTunnel 究竟是什么?
  • # Java NIO(一)FileChannel
  • # 安徽锐锋科技IDMS系统简介
  • (175)FPGA门控时钟技术
  • (Java岗)秋招打卡!一本学历拿下美团、阿里、快手、米哈游offer
  • (Pytorch框架)神经网络输出维度调试,做出我们自己的网络来!!(详细教程~)
  • (阿里云万网)-域名注册购买实名流程
  • (附源码)spring boot校园拼车微信小程序 毕业设计 091617
  • (附源码)springboot金融新闻信息服务系统 毕业设计651450
  • (每日持续更新)信息系统项目管理(第四版)(高级项目管理)考试重点整理 第13章 项目资源管理(七)
  • (实战篇)如何缓存数据
  • (太强大了) - Linux 性能监控、测试、优化工具
  • (一)springboot2.7.6集成activit5.23.0之集成引擎
  • .[hudsonL@cock.li].mkp勒索加密数据库完美恢复---惜分飞
  • .bat批处理(三):变量声明、设置、拼接、截取
  • .NET 8 编写 LiteDB vs SQLite 数据库 CRUD 接口性能测试(准备篇)
  • .Net core 6.0 升8.0
  • .Net OpenCVSharp生成灰度图和二值图
  • .NET Remoting Basic(10)-创建不同宿主的客户端与服务器端
  • .NET 材料检测系统崩溃分析
  • .net 反编译_.net反编译的相关问题
  • .net安装_还在用第三方安装.NET?Win10自带.NET3.5安装
  • .Net转Java自学之路—基础巩固篇十三(集合)