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

【STM32 FreeRTOS】任务通知

任务通知简介

任务通知:用来通知任务的,任务控制块中的结构体成员变量ulNotifiedValue(32位)就是这个通知值。

	#if( configUSE_TASK_NOTIFICATIONS == 1 )volatile uint32_t ulNotifiedValue;volatile uint8_t ucNotifyState;#endif

任务控制块TCB里有两个成员变量,一个uint32_t的表示通知值,一个uint8_t的用来表示通知状态。

  • 使用队列、信号量、事件标志组时都需要另外创建一个结构体,通过中间的结构体进行间接通信。
  • 使用任务通知时,任务结构体TCB中就包含了内部对象,可以直接接收别人发过来的通知。

任务通知值的更新方式:

  • 不覆盖接收任务的通知值
  • 覆盖接收任务的通知值
  • 更新接收任务通知值的一个或多个bit
  • 增加接收任务的通知值

只要合理,灵活的利用任务通知的特点,可以在一些场合中替代队列、信号量和事件标志组。

任务通知的优势:

  • 效率更高:使用任务通知向任务发送事件或数据比使用队列、事件标志组或信号量快得多。
  • 使用内存更小:使用任务通知时无需额外创建结构体

任务通知的劣势:

  • 无法发送数据给ISR:ISR没有任务结构体,所以无法给ISR发送数据。但是ISR可以使用任务通知发数据给任务。
  • 无法广播给多个任务:任务通知只能是被指定的一个任务接收并处理。
  • 无法缓存多个数据:任务结构体中只有一个任务通知值,只能保存一个数据。
  • 发送受阻不支持阻塞:发送发无法进入阻塞状态等待。

任务通知状态:

#define taskNOT_WAITING_NOTIFICATION	( ( uint8_t ) 0 )
#define taskWAITING_NOTIFICATION		( ( uint8_t ) 1 )
#define taskNOTIFICATION_RECEIVED		( ( uint8_t ) 2 )
  • 任务未等待通知,任务通知默认的初始化状态
  • 等待通知,接收方已经准备好了(调用了接收任务通知函数),等待发送方发给个通知
  • 等待接收,发送方已经发送出去(调用了发送任务通知函数),等待接收方接收

任务通知相关API函数介绍

发送通知

函数描述
xTaskNotify发送通知,带有通知值
xTaskNotifyAndQuery发送通知,带有通知值并且保留接收任务的原有通知值
xTaskNotifyGive发送通知,不带通知值
xTaskNotifyFromISR
xTaskNotifyAndQueryFromISR
vTaskNotifyGiveFromISR

接收通知

函数描述
ulTaskNotifyTake获取任务通知,可以设置在退出该函数的时候将任务通知值清零或者减一。
xTaskNotifyWait获取任务通知,可获取通知值和清除通知值的指定位
  • 当任务通知用作于信号量时,使用ulTaskNotifyTake函数获取信号量。
  • 当任务用作于事件标志组或队列时,使用xTaskNotifyWait函数来获取。

模拟二值信号量和计数信号量

void StartMyTask1(void *argument)
{printf("StartMyTask1\r\n");uint32_t notify_value = 0;for(;;){//这里实际是取通知值,只要通知值大于0的时候就不会阻塞,可以执行到下面//第一个参数设置为true,表示执行完该函数后,会将通知值设置为0,//这意味着如果再没有接收通知,那么会一直阻塞在这里。//函数返回值,表示通知值再设置为0之前的值(一般情况下为1)notify_value = ulTaskNotifyTake(pdTRUE,portMAX_DELAY);if(0 != notify_value)printf("task1 notify_value=%d\r\n",notify_value);vTaskDelay(pdMS_TO_TICKS(500));//500ms}
}void NotifyMyTask1FromISR()
{printf("NotifyMyTask1FromISR\r\n");BaseType_t xHigherPriorityTaskWoken = pdFALSE;//这里是在中断服务函数中调用的vTaskNotifyGiveFromISR(myTask1Handle,&xHigherPriorityTaskWoken);portYIELD_FROM_ISR( xHigherPriorityTaskWoken );
}
NotifyMyTask1FromISR
task1 notify_value=1
NotifyMyTask1FromISR
task1 notify_value=1
NotifyMyTask1FromISR
task1 notify_value=1
NotifyMyTask1FromISR
task1 notify_value=1
NotifyMyTask1FromISR
NotifyMyTask1FromISR
task1 notify_value=2
NotifyMyTask1FromISR
NotifyMyTask1FromISR
task1 notify_value=2
NotifyMyTask1FromISR
task1 notify_value=1
NotifyMyTask1FromISR
NotifyMyTask1FromISR
NotifyMyTask1FromISR
task1 notify_value=3
NotifyMyTask1FromISR
NotifyMyTask1FromISR
NotifyMyTask1FromISR
task1 notify_value=3

可以看到,正常情况下,执行一次NotifyMyTask1FromISR就会执行一次ulTaskNotifyTake。但是如果NotifyMyTask1FromISR函数执行的频率很快,ulTaskNotifyTake函数执行不过来,所以通知值不是1(执行通知函数的时候实际上是将通知值加一),ulTaskNotifyTake函数执行一次后还是会阻塞,因为执行一次后将通知值设置为0了。如果将ulTaskNotifyTake函数的第一个参数设置为pdFALSE,表示执行完后将通知值减一,这样其实就是模拟计数信号量了。

void StartMyTask1(void *argument)
{printf("StartMyTask1\r\n");uint32_t notify_value = 0;for(;;){//这里实际是取通知值,只要通知值大于0的时候就不会阻塞,可以执行到下面//第一个参数设置为false,表示执行完该函数后,会将通知值减1,//函数返回值,表示通知值在减1之前的值notify_value = ulTaskNotifyTake(pdFALSE,portMAX_DELAY);if(0 != notify_value)printf("task1 notify_value=%d\r\n",notify_value);vTaskDelay(pdMS_TO_TICKS(500));//500ms}
}
NotifyMyTask1FromISR
task1 notify_value=1
NotifyMyTask1FromISR
task1 notify_value=1
NotifyMyTask1FromISR
task1 notify_value=1
NotifyMyTask1FromISR
task1 notify_value=1
NotifyMyTask1FromISR
NotifyMyTask1FromISR
NotifyMyTask1FromISR
task1 notify_value=3
task1 notify_value=2
task1 notify_value=1

模拟消息邮箱

void NotifyMyTask1FromISR(uint32_t notify_value)
{printf("NotifyMyTask1FromISR\r\n");BaseType_t xHigherPriorityTaskWoken = pdFALSE;//这里是在中断服务函数中调用的//将参数值当作通知值发送给任务xTaskNotifyFromISR(myTask1Handle,notify_value,eSetValueWithOverwrite,&xHigherPriorityTaskWoken);portYIELD_FROM_ISR( xHigherPriorityTaskWoken );
}void StartMyTask1(void *argument)
{printf("StartMyTask1\r\n");uint32_t notify_value = 0;for(;;){//第一个参数表示接收前不清零通知值的bit//第二个参数表示接收后清零通知值的所有bitxTaskNotifyWait(0x00,0xFFFFFFFF,&notify_value,portMAX_DELAY);printf("task1 notify_value=%d\r\n",notify_value);vTaskDelay(pdMS_TO_TICKS(500));//500ms}
}
NotifyMyTask1FromISR
task1 notify_value=1
NotifyMyTask1FromISR
task1 notify_value=2
NotifyMyTask1FromISR
task1 notify_value=3
NotifyMyTask1FromISR
task1 notify_value=4

可以将参数作为通知值传递给任务。

模拟时间标志组

void NotifyMyTask1FromISR(uint32_t notify_value)
{//notify_value表示第几位设置为1,其实可以一次设置多位,这里不演示//notify_value == 2,则表示第二位置1,那么通知值 = (1<<2)printf("NotifyMyTask1FromISR\r\n");BaseType_t xHigherPriorityTaskWoken = pdFALSE;//模拟事件标志组其实就是利用通知值的每一位作为事件的Flag,//发送和接收的时候,都只是修改和判断对应bitxTaskNotifyFromISR(myTask1Handle,1<<notify_value,eSetBits,&xHigherPriorityTaskWoken);portYIELD_FROM_ISR( xHigherPriorityTaskWoken );
}void StartMyTask1(void *argument)
{printf("StartMyTask1\r\n");uint32_t notify_value = 0;for(;;){//第一个参数表示接收前不清零通知值的bit//第二个参数表示接收后清零通知值的所有bitxTaskNotifyWait(0x00,0xFFFFFFFF,&notify_value,portMAX_DELAY);printf("task1 notify_value=%d\r\n",notify_value);if((notify_value & (1<<0)) != 0){printf("bit0 event... \r\n");}else if((notify_value & (1<<1)) != 0){printf("bit1 event... \r\n");}else if((notify_value & (1<<2)) != 0){printf("bit2 event... \r\n");}vTaskDelay(pdMS_TO_TICKS(500));//500ms}
}
NotifyMyTask1FromISR
task1 notify_value=4
bit2 event... 
NotifyMyTask1FromISR
task1 notify_value=2
bit1 event... 
NotifyMyTask1FromISR
task1 notify_value=1
bit0 event... 
NotifyMyTask1FromISR
task1 notify_value=4
bit2 event... 

其实,关于xTaskNotifyWait函数的第二个参数,准确来说不是执行完xTaskNotifyWait函数后清零比特位,而是在下次给通知值赋值的时候吧???

相关文章:

【STM32 FreeRTOS】任务通知

任务通知简介 任务通知&#xff1a;用来通知任务的&#xff0c;任务控制块中的结构体成员变量ulNotifiedValue&#xff08;32位&#xff09;就是这个通知值。 #if( configUSE_TASK_NOTIFICATIONS 1 )volatile uint32_t ulNotifiedValue;volatile uint8_t ucNotifyState;#endi…...

51单片机学习

定时器流水灯 #include <REGX52.H> #include "Timer0.h" #include "Key.h" #include <INTRINS.H> unsigned char KeyNum,LEDMode; void main() { P20xFE; Timer0Init(); while(1) { KeyNumKey(); if(KeyNum)…...

vue项目实现postcss-pxtoremvue大屏适配

1.安装依赖 npm install postcss-pxtorem autoprefixer postcss-loader --save-dev # 或者 yarn add postcss-pxtorem autoprefixer postcss-loader --dev2.配置 PostCSS 在项目根目录下创建一个 .postcssrc.js 文件&#xff0c;并添加以下配置&#xff1a; module.exports …...

如何打造爆款游戏?开发由你操刀,运维交由我托管,合作共赢创造更大成功

Linode提供的云计算服务都有哪里的哪些人在用&#xff0c;又都用来做什么&#xff1f;简而言之&#xff1a;世界各地&#xff01;各行各业&#xff01;&#xff01;丰富多彩&#xff01;&#xff01;&#xff01; 今天我们将关注云计算在游戏行业的应用。在这篇文章里&#xf…...

颈部按摩仪语音播报芯片方案,高品质语音IC,NV080D

想要利用碎片化的时间按摩肩颈&#xff0c;颈部按摩仪是很好的选择。然而&#xff0c;随着科技的不断进步&#xff0c;一些新的技术也开始被应用于颈部按摩仪中&#xff0c;以提升它们的功能和用户体验。 例如&#xff0c;NV080D语音播放芯片在颈部按摩仪上的应用&#xff0c;…...

Opencv模板匹配

使用OpenCV和C来识别彩色图片中的特定物体&#xff0c;如黑桃♠&#xff0c;通常涉及几个步骤&#xff1a;预处理图像、特征提取、对象检测等。下面是一个基本的示例代码&#xff0c;演示如何使用OpenCV的模板匹配方法来识别图片中的黑桃♠。 函数原型 void matchTemplate(Inp…...

JavaScript DOM事件监听器:深入解析与实践应用

引言 在JavaScript中&#xff0c;DOM&#xff08;文档对象模型&#xff09;事件监听器是与用户交互的核心机制之一。它们允许开发者响应用户的行为&#xff0c;如点击、滚动、输入等&#xff0c;从而创建动态和交互式的网页。本文将深入探讨DOM事件监听器的工作原理、类型以及…...

iOS的App启动详细过程(底层知识)

1.虚拟内存 & ASLR 在早期计算机中数据是直接通过物理地址访问的&#xff0c;这就造成了下面两个问题 1.内存不够用 2.数据安全问题 内存不够 ---> 虚拟内存 虚拟内存就是通过创建一张物理地址和虚拟地址的映射表来管理内存&#xff0c;提高了CPU利用率&#xff0c;…...

【轨物推荐】创新者的钥匙:如何破解创新的最大难题

原创 混沌学园 混沌学园 2021年12月27日 19:55 本文重点在于&#xff0c;将纷繁复杂现象中&#xff0c;针对创新的灵魂问题“不创新是等死&#xff0c;创新是找死”&#xff0c;给出本质上的解读、解析和解答。将创新在保守和发散的两难选择中&#xff0c;下探本质找到那个价值…...

SpringCloud的能源管理系统-能源管理平台源码

介绍 基于SpringCloud的能源管理系统-能源管理平台源码-能源在线监测平台-双碳平台源码-SpringCloud全家桶-能管管理系统源码 软件架构...

Mybatis获取主键自增的方法

原本的方法 使用原本的JDBC去获取主键自增的方法如下图所示&#xff1a; 在这段代码中&#xff0c;通过连接对象 conn 的 prepareStatement 方法创建了一个PreparedStatement对象&#xff0c;并将SQL语句和 RETURN_GENERATED_KEYS 常量作为参数传递给该方法。这意味着执行SQL…...

strip 、objdump、objcopy 差异与区别

strip 、objdump、objcopy 差异与区别 strip 命令用于从已编译的可执行文件或目标文件中移除调试信息和其他非必要数据。这可以减小文件的大小&#xff0c;并且有助于保护源代码不被轻易反编译。通常&#xff0c;在发布软件时会使用 strip 命令来减少二进制文件的体积 objdump…...

本地phpstudy部署算命系统,用户端是H5页面,支持微信支付宝支付,支持微信支付宝登录

算命系统本地部署教程 0. 技术架构1. 启动Apache、MySQL服务2. 创建前台和后台两个网站3. Navicat连接数据库4. 登录后台是长这个样子5. 登录前台登录样子6. 代码结构是 0. 技术架构 前端&#xff1a;HTMLCSSJquery 后端&#xff1a;PHP 数据库&#xff1a;MySQL 1. 启动Ap…...

APP上架苹果App Store被拒原因及解决方案

苹果官方商店的审核大致分为三部分&#xff1a;预审、机审和人工审核&#xff0c;审核流程包括&#xff1a;初步提交、审核队列、审核过程、结果通知等步骤。#iOS App Store审核是非常严格的&#xff0c;很多很多APP&#xff0c;并不是能够一次上架成功&#xff0c;大部分APP都…...

docker-compose的下载

方式一&#xff1a;使用二进制文件安装。 GitHub官方下载安装&#xff08;推荐&#xff0c;速度慢但不会出错&#xff09; 1.GitHub官方下载&#xff08;测试&#xff0c;已成功&#xff09; curl -SL https://github.com/docker/compose/releases/download/v2.29.1/docker-c…...

h3c虚拟园区网概述

虚拟园区网概述 H3C虚拟园区网的解决方案 企业园区网作为企业网络的核心部分&#xff0c;连接了企业总部的办公、生产、研发、财务等多种重要的机 构。在网络建设中占有重要的地位。园区网内部终端种类众多,接入用户数量庞大,对网络的性能、可 靠性、可管理性都有较高的要求…...

云原生和安装Ubuntu 22系统

一.云原生简介 2004年开始&#xff0c;Google已在内部大规模地使用容器技术。 2008年&#xff0c;Google将 Cgroups合并进入了Linux内核。2013年&#xff0c;Docker项目正式发布。2014年&#xff0c;Kubernetes项目正式发布。2015年&#xff0c;由Google、Redhat 以及微软等大…...

HTTP代理IP如何助力旅游大数据领域?怎么建立安全的代理隧道连接?

在旅游行业&#xff0c;数据的准确性和实时性至关重要。随着大数据分析的兴起&#xff0c;HTTP代理IP成为了这一领域中不可或缺的工具。本文将探讨HTTP代理IP如何帮助旅游大数据领域&#xff0c;并介绍如何建立安全的代理隧道连接。 1. HTTP代理IP在旅游大数据领域的应用 1.1…...

AIGC从入门到实战:基础理论【核心算法与模型】

历史导读&#xff1a; 三、AIGC从入门到实战:基础理论【模型微调】 二、AIGC从入门到实战:AIGC基础理论 一、AIGC从入门到实战&#xff1a;为什么要了解AIGC 核心算法与模型 在人工智能生成内容(AIGC)领域&#xff0c;技术的飞速进步催生了一系列高效且创新的生成模型&…...

极狐GitLab 17.2发布了哪些 JH-Only 的功能?

极狐GitLab 是 GitLab 在中国的发行版&#xff0c;专门面向中国程序员和企业提供企业级一体化 DevOps 平台&#xff0c;用来帮助用户实现需求管理、源代码托管、CI/CD、安全合规&#xff0c;而且所有的操作都是在一个平台上进行&#xff0c;省事省心省钱。可以一键安装极狐GitL…...

css中的高度塌陷

CSS高度塌陷&#xff08;或称为高度坍塌&#xff09;是指在某些特定情况下&#xff0c;元素的高度无法被正确计算或显示的现象。这通常发生在具有浮动属性的元素或使用绝对定位的元素周围。 原因 高度塌陷通常发生在父元素包含着一个或多个浮动元素时。由于浮动元素被移出了正…...

怎样使用sudo的时候不需要输入密码?

在Ubuntu等Linux系统下&#xff0c;经常要在个人账户使用sudo命令来执行一些需要root权限的命令&#xff0c;但是需要输入该账户的密码&#xff0c;有时候显得很繁琐&#xff0c; 那么怎样使用sudo的时候不需要输入密码呢&#xff1f; 有如下两种方法&#xff1a; 常规方法1…...

kettle的Javascript组件获取T-1天和T+1天

// 获取T-1的时间 var currentDate new Date(); currentDate.setDate(currentDate.getDate() - 1); var currentYear currentDate.getFullYear(); var currentMonth (0 (currentDate.getMonth() 1)).slice(-2); var currentDay (0 currentDate.getDate()).slice(-2); va…...

YoloV8改进策略:Block改进|LeYOLO,一种用于目标检测的新型可扩展且高效的CNN架构|复现LeYolo,轻量级Yolo改进

摘要 LeYOLO是在YOLO系列&#xff0c;特别是可能受到YOLOv8启发的基础上进行的一系列改进&#xff0c;旨在提升目标检测模型的高效性、可扩展性和精度。其主要特点包括&#xff1a; 高效骨干网络缩放方法&#xff1a; LeYOLO借鉴了倒置瓶颈&#xff08;Inverted Bottleneck&am…...

ODX(Open Diagnostic Data Exchange)简介

ODX(Open Diagnostic Data Exchange)是一种由ASAM制定的开放标准,用于描述和交换ECU(电子控制单元)诊断数据,广泛应用于车辆诊断。ODX文件采用XML格式,包含通讯参数,如ISO15765-2/3时间参数。 ASAM(Association for Standardisation of Automation and Measuring Syst…...

记一次CSDN认证模块后端未校验漏洞

前言 作为一个程序员&#xff0c;一直充满好奇心&#xff0c;没事就喜欢找找漏洞&#xff0c;试想一下某些程序是否存在某些鉴权等漏洞&#xff0c;目前该漏洞已提交官方&#xff0c;且影响不大&#xff0c;现分享分析过程用于各位技术学习。 漏洞分析 https://i.csdn.net/#…...

【图机器学习系列】(一)图机器学习简介

微信公众号&#xff1a;leetcode_algos_life&#xff0c;代码随想随记 小红书&#xff1a;412408155 CSDN&#xff1a;https://blog.csdn.net/woai8339?typeblog &#xff0c;代码随想随记 GitHub: https://github.com/riverind 抖音【暂未开始&#xff0c;计划开始】&#xf…...

全网最详细,从一堆字符串,精确抓取想要日期时间的实战2.0

前言: 前面我们知道了&#xff0c;怎么从一堆带有中文、英文、日期时间的字符串里面抓取需要的日期时间&#xff0c;但是我们实现的只是抓取第一个日期时间&#xff0c;那我们怎么实现&#xff0c;抓取第二个&#xff0c;或者任一一个日期时间呢&#xff1f; 一、思路分析 1、数…...

24/8/15算法笔记 dp策略迭代 价值迭代

策略迭代&#xff1a; 策略迭代从某个策略开始&#xff0c;计算该策略下的状态价值函数。它交替进行两个步骤&#xff1a;策略评估&#xff08;Policy Evaluation&#xff09;和策略改进&#xff08;Policy Improvement&#xff09;。在策略评估阶段&#xff0c;计算给定策略下…...

【MMdetection改进】换遍MMDET主干网络之SwinTransformer-Tiny(基于MMdetection)

OpenMMLab 2.0 体系中 MMYOLO、MMDetection、MMClassification、MMSelfsup 中的模型注册表都继承自 MMEngine 中的根注册表&#xff0c;允许这些 OpenMMLab 开源库直接使用彼此已经实现的模块。 因此用户可以在MMYOLO 中使用来自 MMDetection、MMClassification、MMSelfsup 的主…...