当前位置:首页 > > strongerHuang
[导读]本文详细讲述了一个知识点:FreeRTOS的直接任务(消息)通知,这样做的目的就是减少RAM占用空间并加快执行速度。


关注、星标公众,不错过精彩内容

作者:strongerHuang

微信公众号:strongerHuang


之前分享了《FreeRTOS V10.4.0更新了哪些功能?》,今天就来详细讲述其中的一个知识点:FreeRTOS的直接任务(消息通知,这样做的目的就是减少RAM占用空间并加快执行速度。


一、写在前面

几乎所有RTOS操作系统都提供了队列和信号量的功能,对于大部分新手来说,使用队列和信号量是必备技能。


但是,在大多数情况下,他们都是使用“中介对象”进行通信,而并非“直接任务消息”通信。


通过“中介对象”进行通信,每一组队列或信号量都会分配一段内存(消息缓冲区和流缓冲区)。就存在一个问题,如果队列或信号量比较多,势必造成更大的内存开支。


但是,如果通过本文说的“直接消息”通信,会节约很多内存。


二、什么是直接任务通知?
大多数任务间通信方法都通过中介对象,例如队列,信号量或事件组。发送任务写入通信对象,接收任务从通信对象读取。

比如FreeRTOS的队列通信,首先创建队列之前要定义一个队列:
QueueHandle_t xQueue;
xQueue = xQueueCreate(10, sizeof( /* 长度 */ ) );

而这个队列包含了很多中介对象:


大家可以算一下这个“中介对象”会占用多少RAM空间?


通过一个代码示意图理解中介对象通信:


直接任务通知:
当使用直接任务通知时,顾名思义,发送任务将通知直接发送给接收任务,而无需中介对象。

通过一个代码示意图理解:


从FreeRTOS V10.4.0开始,每个任务都有一系列通知。每个通知都包含一个32位值和一个布尔状态,它们一起仅消耗5个字节的RAM。


就像任务可以阻止二进制信号量等待该信号量变为“可用”一样,任务可以阻止通知以等待该通知的状态变为“待处理”。同样,就像任务可以阻止计数信号量以等待该信号量的计数变为非零一样,任务可以阻止通知以等待该通知的值变为非零。下面的第一个示例演示了这种情况。


通知不仅可以传达事件,还可以通过多种方式传达数据。


三、进一步分析直接任务通知
通过对比FreeRTOS V10.4.0和之前版本,你会发现V10.4.0多了一些API,比如ulTaskNotifyTake / ulTaskNotifyTakeIndexed:


在官网也有针对这些API的详细介绍和说明,以及应用代码例子:


直接任务通信API说明地址:

https://www.freertos.org/RTOS-task-notification-API.html

(公号不支持外链接,请复制链接到浏览器打开)

四、使用直接任务通知性能优势和使用限制
任务通知的灵活性使它们可以在需要创建单独的队列、 二进制信号量、 数信号量或事件组的情况下使用。

与使用中介对象(例如信号量)来取消阻止任务相比,使用直接通知取消阻止RTOS任务的速度快了45% (来自官方数据) ,并且使用的RAM更少。

当然,有这些性能优势,也肯定一些限制
  • 仅当只有一个任务可以作为事件的接收者时,才可以使用RTOS任务通知。但是,在大多数实际使用情况下都可以满足此条件,例如中断使执行任务处理的任务中断时,该任务将处理该中断接收的数据。


  • 仅在使用RTOS任务通知代替队列的情况下:接收任务可以在“阻塞”状态下等待通知(因此不占用任何CPU时间),而发送任务不能在“阻塞”状态下等待消息。如果发送无法立即完成,则发送完成。


五、使用方法
使用方法其实很简单,只要你会使用RTOS的队列、信号量,基本看一眼官方例子就能使用。

我这里也拿官方例子说明一下:
/* main() 创建的两个任务的原型 */static void prvTask1( void *pvParameters );static void prvTask2( void *pvParameters );
/* 处理由main() 创建的任务的句柄 */static TaskHandle_t xTask1 = NULL, xTask2 = NULL;
/* 创建两个任务,来回发送通知,然后启动RTOS调度程序 */void main( void ){ xTaskCreate( prvTask1, “Task1”, 200, NULL, tskIDLE_PRIORITY, &xTask1 ); xTaskCreate( prvTask2, “Task2”, 200, NULL, tskIDLE_PRIORITY, &xTask2 ); vTaskStartScheduler();}/*———————————————————–*/
/* prvTask1() 使用API的“索引”版本 */static void prvTask1( void *pvParameters ){ for( ;; ) { /* 发送通知到prvTask2() ,使其脱离“已阻止”状态。*/ xTaskNotifyGiveIndexed( xTask2, 0 );
        /* 阻止等待prvTask2() 通知此任务 */ ulTaskNotifyTakeIndexed( 0, pdTRUE, portMAX_DELAY ); }}/*———————————————————–*/
/* prvTask2()使用API的原始版本(不带“索引”) */static void prvTask2( void *pvParameters ){ for( ;; ) {        /* 等待prvTask1()通知此任务 */ ulTaskNotifyTake( pdTRUE, portMAX_DELAY );
        /* 向prvTask1()发送通知,使它退出“已阻止”状态 */ xTaskNotifyGive( xTask1 ); }}

本文就介绍到这里,更多内容请参看官方文档:

https://www.freertos.org/2020/09/decrease-ram-footprint-and-accelerate-execution-with-freertos-notifications.html


推荐阅读:
MCU堆栈空间,你设置了多少?
为什么不建议使用DS-5开发STM32?
从基础描述无刷直流电机的工作原理

关注 微信公众号『strongerHuang』,后台回复“1024”查看更多内容,回复“加群”按规则加入技术交流群。


长按前往图中包含的公众号关注

免责声明:本文内容由21ic获得授权后发布,版权归原作者所有,本平台仅提供信息存储服务。文章仅代表作者个人观点,不代表本平台立场,如有问题,请联系我们,谢谢!

本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

LED驱动电源的输入包括高压工频交流(即市电)、低压直流、高压直流、低压高频交流(如电子变压器的输出)等。

关键字: 驱动电源

在工业自动化蓬勃发展的当下,工业电机作为核心动力设备,其驱动电源的性能直接关系到整个系统的稳定性和可靠性。其中,反电动势抑制与过流保护是驱动电源设计中至关重要的两个环节,集成化方案的设计成为提升电机驱动性能的关键。

关键字: 工业电机 驱动电源

LED 驱动电源作为 LED 照明系统的 “心脏”,其稳定性直接决定了整个照明设备的使用寿命。然而,在实际应用中,LED 驱动电源易损坏的问题却十分常见,不仅增加了维护成本,还影响了用户体验。要解决这一问题,需从设计、生...

关键字: 驱动电源 照明系统 散热

根据LED驱动电源的公式,电感内电流波动大小和电感值成反比,输出纹波和输出电容值成反比。所以加大电感值和输出电容值可以减小纹波。

关键字: LED 设计 驱动电源

电动汽车(EV)作为新能源汽车的重要代表,正逐渐成为全球汽车产业的重要发展方向。电动汽车的核心技术之一是电机驱动控制系统,而绝缘栅双极型晶体管(IGBT)作为电机驱动系统中的关键元件,其性能直接影响到电动汽车的动力性能和...

关键字: 电动汽车 新能源 驱动电源

在现代城市建设中,街道及停车场照明作为基础设施的重要组成部分,其质量和效率直接关系到城市的公共安全、居民生活质量和能源利用效率。随着科技的进步,高亮度白光发光二极管(LED)因其独特的优势逐渐取代传统光源,成为大功率区域...

关键字: 发光二极管 驱动电源 LED

LED通用照明设计工程师会遇到许多挑战,如功率密度、功率因数校正(PFC)、空间受限和可靠性等。

关键字: LED 驱动电源 功率因数校正

在LED照明技术日益普及的今天,LED驱动电源的电磁干扰(EMI)问题成为了一个不可忽视的挑战。电磁干扰不仅会影响LED灯具的正常工作,还可能对周围电子设备造成不利影响,甚至引发系统故障。因此,采取有效的硬件措施来解决L...

关键字: LED照明技术 电磁干扰 驱动电源

开关电源具有效率高的特性,而且开关电源的变压器体积比串联稳压型电源的要小得多,电源电路比较整洁,整机重量也有所下降,所以,现在的LED驱动电源

关键字: LED 驱动电源 开关电源

LED驱动电源是把电源供应转换为特定的电压电流以驱动LED发光的电压转换器,通常情况下:LED驱动电源的输入包括高压工频交流(即市电)、低压直流、高压直流、低压高频交流(如电子变压器的输出)等。

关键字: LED 隧道灯 驱动电源
关闭