0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

Free RTOS的任务通知-3

汽车电子技术 来源:玩转单片机 作者:Julian 2023-02-10 16:11 次阅读

每个任务都有一个32位的通知值,该值在创建任务时初始化为零。

配置相关资源

//为1时开启任务通知
#define configUSE_TASK_NOTIFICATIONS 1

发送

BaseType_t xTaskNotify( TaskHandle_t xTaskToNotify, 
                        uint32_t ulValue, 
                        eNotifyAction eAction );

参数

xTaskToNotify:被通知并使其通知值递增的任务句柄

ulValue:用于更新被通知任务的通知值

eAction:通知任务时要执行的操作

eNoAction = 0,通知任务而不更新其通知值

eSetBits,设置任务通知值中的位

eIncrement,增加任务的通知值

eSetvaluewithoverwrite,覆盖当前通知

eSetValueWithoutoverwrite 不覆盖当前通知

返回值

pdFAIL:当参数eAction设置为eSetValueWithoutOverwrite的时候,如果任务通知值没有更新成功就返回pdFAIL。

pdPASS: eAction 设置为其他选项的时候统一返回pdPASS。

接收

uint32_t ulTaskNotifyTake( BaseType_t xClearCountOnExit, TickType_t xTicksToWait );

参数

xClearCountOnExit:是否需要清零

xTicksToWait:等待时间

实验程序

#include "stm32f10x.h"
#include 
#include "FreeRTOS.h"
#include "task.h"

void KEY_Init(void)
{
  GPIO_InitTypeDef GPIO_InitStructure; //定义结构体变量  
  RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA|RCC_APB2Periph_GPIOE,ENABLE);
  
  GPIO_InitStructure.GPIO_Pin=GPIO_Pin_0;     //选择你要设置的IO口
  GPIO_InitStructure.GPIO_Mode=GPIO_Mode_IPD;//下拉输入  
  GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz;     //设置传输速率
  GPIO_Init(GPIOA,&GPIO_InitStructure);      /* 初始化GPIO */
  
  GPIO_InitStructure.GPIO_Pin=GPIO_Pin_3|GPIO_Pin_2|GPIO_Pin_4;
  GPIO_InitStructure.GPIO_Mode=GPIO_Mode_IPU;  //上拉输入
  GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz;
  GPIO_Init(GPIOE,&GPIO_InitStructure);
}

void USART_init(uint32_t bound)
{
  GPIO_InitTypeDef GPIO_InitStruct;   //定义GPIO结构体变量
  USART_InitTypeDef USART_InitStruct;   //定义串口结构体变量
  
  RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA|RCC_APB2Periph_USART1,ENABLE);   //使能GPIOC的时钟
  
  GPIO_InitStruct.GPIO_Pin=GPIO_Pin_9;   //配置TX引脚
  GPIO_InitStruct.GPIO_Mode=GPIO_Mode_AF_PP;   //配置PA9为复用推挽输出
  GPIO_InitStruct.GPIO_Speed=GPIO_Speed_50MHz;   //配置PA9速率
  GPIO_Init(GPIOA,&GPIO_InitStruct);   //GPIO初始化函数
  
  GPIO_InitStruct.GPIO_Pin=GPIO_Pin_10;   //配置RX引脚
  GPIO_InitStruct.GPIO_Mode=GPIO_Mode_IN_FLOATING;   //配置PA10为浮空输入
  GPIO_InitStruct.GPIO_Speed=GPIO_Speed_50MHz;   //配置PA10速率
  GPIO_Init(GPIOA,&GPIO_InitStruct);   //GPIO初始化函数
  
  
  USART_InitStruct.USART_Mode=USART_Mode_Tx|USART_Mode_Rx;   //发送接收模式
  USART_InitStruct.USART_Parity=USART_Parity_No;   //无奇偶校验
  USART_InitStruct.USART_BaudRate=bound;   //波特率
  USART_InitStruct.USART_StopBits=USART_StopBits_1;   //停止位1位
  USART_InitStruct.USART_WordLength=USART_WordLength_8b;   //字长8位
  USART_InitStruct.USART_HardwareFlowControl=USART_HardwareFlowControl_None;   //无硬件数据流控制
  USART_Init(USART1,&USART_InitStruct);   //串口初始化函数
  
  USART_Cmd(USART1,ENABLE);   //使能USART1
}

int fputc(int ch,FILE *f)   //printf重定向函数
{
  USART_SendData(USART1,(uint8_t)ch);   //发送一字节数据
  while(USART_GetFlagStatus(USART1,USART_FLAG_TXE) == RESET);   //等待发送完成
  return ch;
}

#define START_TASK_PRIO 1      //任务优先级
#define START_STK_SIZE 128      //任务堆栈大小
TaskHandle_t StartTask_Handler;   //任务句柄
void Start_Task(void *pvParameters);//任务函数

#define Send_TASK_PRIO 2       //任务优先级
#define Send_STK_SIZE 50       //任务堆栈大小
TaskHandle_t SendTask_Handler;     //任务句柄
void Send_Task(void *p_arg);     //任务函数

#define Read_TASK_PRIO 3       //任务优先级
#define Read_STK_SIZE 50       //任务堆栈大小
TaskHandle_t ReadTask_Handler;     //任务句柄
void Read_Task(void *p_arg);     //任务函数

int main( void ) 
{
  NVIC_PriorityGroupConfig(NVIC_PriorityGroup_4);//设置系统中断优先级分组 4

  KEY_Init();
  USART_init(9600);
  
  //创建开始任务
  xTaskCreate(
    (TaskFunction_t )Start_Task,     //任务函数
    (const char* )"Start_Task",     //任务名称
    (uint16_t )START_STK_SIZE,       //任务堆栈大小
    (void* )NULL,             //传递给任务函数的参数
    (UBaseType_t )START_TASK_PRIO,     //任务优先级
    (TaskHandle_t* )&StartTask_Handler  //任务句柄 
  );
  vTaskStartScheduler();  //开启调度
}
//开始任务函数
void Start_Task(void *pvParameters)
{
  taskENTER_CRITICAL();   //进入临界区
  
  //创建 Send_Task 任务
  xTaskCreate(
    (TaskFunction_t )Send_Task, 
    (const char* )"Send_Task", 
    (uint16_t )Send_STK_SIZE, 
    (void* )NULL,
    (UBaseType_t )Send_TASK_PRIO,
    (TaskHandle_t* )&SendTask_Handler
  );
  //创建 Read_Task 任务
  xTaskCreate(
    (TaskFunction_t )Read_Task, 
    (const char* )"Read_Task", 
    (uint16_t )Read_STK_SIZE, 
    (void* )NULL,
    (UBaseType_t )Read_TASK_PRIO,
    (TaskHandle_t* )&ReadTask_Handler
  );
  vTaskDelete(StartTask_Handler); //删除开始任务
  taskEXIT_CRITICAL();   //退出临界区
}

//Send_Task 任务函数
void Send_Task(void *pvParameters)
{
  uint32_t Value = 0x01;
  while(1)
  {
    /* 原型:BaseType_t xTaskNotify( TaskHandle_t xTaskToNotify, 
                    uint32_t ulValue, 
                    eNotifyAction eAction ); 
    * eNoAction = 0,通知任务而不更新其通知值。
    * eSetBits,     设置任务通知值中的位。
    * eIncrement,   增加任务的通知值。
    * eSetvaluewithoverwrite,覆盖当前通知
    * eSetValueWithoutoverwrite 不覆盖当前通知
    * 
    * pdFAIL:当参数eAction设置为eSetValueWithoutOverwrite的时候,
    * 如果任务通知值没有更新成功就返回pdFAIL。
    * pdPASS: eAction 设置为其他选项的时候统一返回pdPASS。
    */
    /* 触发一个事件 */
    xTaskNotify( (TaskHandle_t  ) ReadTask_Handler,  //接收任务通知的任务句柄
           (uint32_t    ) Value,      //要触发的事件
           (eNotifyAction ) eSetBits);    //设置任务通知值中的位
    Value <<= 1;
    vTaskDelay(1000);
  }
}

//Read_Task 任务函数
void Read_Task(void *pvParameters)
{
  uint32_t Return = 0;
  while(1)
  {
    //获取任务通知 ,没获取到则一直等待
    Return = ulTaskNotifyTake( pdTRUE, portMAX_DELAY );
    printf("Return = %#xn",Return);
    vTaskDelay(1000);
  }
}

实验现象

pYYBAGPl_BuAVILfAAB04f33IfI881.png


--END--


声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • FreeRTOS
    +关注

    关注

    12

    文章

    484

    浏览量

    62178
  • 任务
    +关注

    关注

    1

    文章

    20

    浏览量

    8537
  • 初始化
    +关注

    关注

    0

    文章

    50

    浏览量

    11858
收藏 人收藏

    评论

    相关推荐

    FreeTRTOS可以通过哪些方法向接收任务更新通知

    每个RTOS任务都有一个32位的通知值,任务创建时,这个值被初始化为0。
    的头像 发表于 09-15 09:28 935次阅读

    【CW32移植Free-RTOS】CW32开发者扶持计划

    CW32配置Free-RTOS全过程,CW32开发者扶持计划
    的头像 发表于 04-18 09:38 6150次阅读
    【CW32移植<b class='flag-5'>Free-RTOS</b>】CW32开发者扶持计划

    在energia 编译器中使用free_rtos或者ti_rtos

    请问在energia编译器里,有使用free_rtos或者ti_rtos的案例程序吗?
    发表于 05-14 11:11

    Free RTOS移植问题的解决办法?

    按照原子哥的free rtos抑制说明文档做第一个工程,改完代码编译出现xPortSysTickHandler()函数未定义,头文件中加入task.h和FreeRTOS.h。实在找不出问题所在,,,there must be anther problem???
    发表于 06-11 07:57

    任务通知发送出问题怎么办

    大家好。我在STM32cubemx里使用FreeRTOS,开启定时器3中断,然后发送一个任务通知给TIM3_Task。但是无论是在回调函数里使用vTaskNotifyGiveFromI
    发表于 06-18 04:35

    任务通知的问题如何解决

    在一个最高优先级任务1里面发送通知:xTaskNotify((TaskHandle_t)BatTask_Handler,//接收任务通知的电池任务
    发表于 07-13 10:36

    LEDs状态灯任务(线程)设计 (基于RTOS

    LEDs状态灯任务(线程)设计(基于RTOS
    的头像 发表于 03-12 11:30 2355次阅读

    FreeRTOS的直接任务(消息)通知

            之前分享了《FreeRTOS V10.4.0更新了哪些功能?》,今天就来详细讲述其中的一个知识点:FreeRTOS的直接任务(消息)通知,这样做的目的就是减少RAM占用空间并加快执行
    的头像 发表于 01-07 09:37 4583次阅读

    FreeRTOS系列第14篇---FreeRTOS任务通知

    每个RTOS任务都有一个32位的通知值,任务创建时,这个值被初始化为0。RTOS任务
    发表于 01-26 17:49 5次下载
    FreeRTOS系列第14篇---FreeRTOS<b class='flag-5'>任务</b><b class='flag-5'>通知</b>

    FreeRTOS任务通知简介及实现

    每个系统任务都会有一个任务通知。然后每个任务通知都具有挂起或者未挂起的状态,以及32位的通知。常
    的头像 发表于 12-06 16:24 2094次阅读

    Free RTOS任务通知-1

    每个任务都有一个32位的通知值,该值在创建任务时初始化为零。
    的头像 发表于 02-10 16:05 885次阅读
    <b class='flag-5'>Free</b> <b class='flag-5'>RTOS</b>的<b class='flag-5'>任务</b><b class='flag-5'>通知</b>-1

    Free RTOS任务通知-2

    每个任务都有一个32位的通知值,该值在创建任务时初始化为零。
    的头像 发表于 02-10 16:08 854次阅读
    <b class='flag-5'>Free</b> <b class='flag-5'>RTOS</b>的<b class='flag-5'>任务</b><b class='flag-5'>通知</b>-2

    FreeRTOS任务通知简介

    任务通知简介 任务通知在 FreeRTOS 中是一个可选的功能,要使用任务通知的话就需要将宏co
    的头像 发表于 07-30 11:34 742次阅读

    FreeRTOS任务通知通用发送函数

    发送任务通知 任务通知通用发送函数 任务任务通知
    的头像 发表于 07-30 11:43 726次阅读
    FreeRTOS<b class='flag-5'>任务</b><b class='flag-5'>通知</b>通用发送函数

    使用任务通知提高RTOS应用的效率

    在实时嵌入式系统中,性能和资源效率是决定设计成败的关键因素。传统的实时操作系统(RTOS)提供了如队列、信号量和事件组机制,实现任务之间的同步和通信。FreeRTOS/SAFERTOS还提供一种方法可以使这些过程更快、更轻量化,即任务
    的头像 发表于 12-27 14:54 132次阅读