你的浏览器版本过低,可能导致网站不能正常访问!
为了你能正常使用网站功能,请使用这些浏览器。

基于STM32与FreeRTOS的消息传递详解(HAL库)

[复制链接]
STMCU小助手 发布时间:2022-12-1 18:00
引言
我们在裸机开发中,每个函数之间进行数据通信往往采用全局变量。而在嵌入式开发中。我们在进行进程间通信的时候,往往采用消息队列。对于操作系统来说,消息队列是非常重要的一个数据结构。本文将介绍一下,如何使用消息队列进行通信。

介绍
消息队列概念

队列又称消息队列,是一种常用于任务间通信的数据结构,队列可以在任务与任务间、中断和任务间传递信息,实现了任务接收来自其他任务或中断的不固定长度的消息,任务能够从队列里面读取消息,当队列中的消息是空时,读取消息的任务将被阻塞,用户还可以指定阻塞的任务时间 xTicksToWait,在这段时间中,如果队列为空,该任务将保持阻塞状态以等待队列数据有效。当队列中有新消息时,被阻塞的任务会被唤醒并处理新消息;当等待的时间超过了指定的阻塞时间,即使队列中尚无有效数据,任务也会自动从阻塞态转为就绪态。消息队列是一种异步的通信方式。

在FreeRTOS中的消息队列函数
1.设定消息队列的格式:osMessageQDef(myQueue, len, size);
myQueue是消息队列的名称。
len是消息队列的长度(有几个消息)
size是每个消息的大小,也就是每个元素的格式

2.创建消息:osMessageCreate(osMessageQ(myQueue01), NULL);
创建消息的函数,实际上是调用了FreeRTOS的osMessageCreate()函数,只不过HAL库进行了封装。

3.向消息队列发送消息
我们这里来介绍在中断中发送消息。使用函数xQueueSendFromISR(QueueHandle,&Res,time);
其中:
QueueHandle:消息队列的句柄
&Res:要发送的数据的地址
time:阻塞时间,就是如果消息队列满的时候,任务应该阻塞多久

4.接收消息队列中的消息
xQueueReceive(QueueHandle,&queue_buffer,time);
QueueHandle:消息队列的句柄
&queue_buffer:接收的消息要存放在的地址
time:阻塞时间,就是如果消息队列空的时候,任务应该阻塞多久

5.查询消息队列中消息的数量
uxQueueMessagesWaiting(myQueue01Handle),可以返回消息队列(句柄为myQueue01Handle)中消息的数量,返回值为整数。

实例
需求分析

此样例我们使用PC充当上位机,上位机发送数据后,在串口中断函数中将接收到的PC数据发送在消息队列myQueue01Handle中,之后在一个接收线程中接收这个消息的内容,并通过串口将接受到的消息的大小和内容输出出来。

发送消息
当上位机PC下发数据后,串口中断函数将接收到的数据发送在消息队列中。

  1. void USART3_IRQHandler(void)
  2. {
  3.         uint8_t Res;
  4. if(__HAL_UART_GET_FLAG(&huart3,UART_FLAG_RXNE)!=RESET)//检测到有单个字节的中断
  5. {
  6.         HAL_UART_Receive(&huart3,&Res,1,0Xffff);
  7.         xQueueSendFromISR(myQueue01Handle,&Res,0)//发送消息
  8. }
  9. else if(__HAL_UART_GET_FLAG(&huart3,UART_FLAG_IDLE)!=RESET)//空闲中断(代表这一帧数据传输完了)
  10. {
  11.         printf(" Receive a frame data.");
  12.         __HAL_UART_CLEAR_IDLEFLAG(&huart3)
  13. }
复制代码

接收消息
我们创建一个任务,此任务的重要功能就是接收消息队列中的消息。我们将接受到的消息的大小和内容通过串口发送出来。没有消息的时候,一直实现LED的闪烁。

  1. void LEDToggleTesk(void const * argument)
  2. {
  3.   /* USER CODE BEGIN LEDToggleTesk */
  4.   BaseType_t xReturn=pdTRUE;//定义一个创建消息返回值,默认为pdTRUE
  5.   UBaseType_t num_queue ;
  6.   uint8_t Res[20];//存放我们接收到的一包数据
  7.   uint8_t queue_buffer;
  8.   int i=0;//接收数组下标
  9.   for(;;)
  10.   {
  11.     i=0;
  12.     HAL_GPIO_TogglePin(GPIOA,GPIO_PIN_5);
  13.     num_queue=uxQueueMessagesWaiting(myQueue01Handle);//获取消息队列中有多少数据
  14.     while(num_queue--)
  15.     {
  16.       xReturn=xQueueReceive(myQueue01Handle,&queue_buffer,0);//将消息队列中的数据放在queue_buffer中
  17.       if(xReturn)
  18.       Res[i++]=queue_buffer;  
  19.     }
  20.     if(i!=0)
  21.     printf(" count %d,LEDTask Receive %s",i,Res);//输出接收消息的大小和内容
  22.     osDelay(500);
  23.   }
  24. }
复制代码

现象

PC端发送123456789,MCU回复Receive a frame data
count 8,LEDTask Receive 12345678


640.png

转载自: 跋扈洋
收藏 评论0 发布时间:2022-12-1 18:00

举报

0个回答

所属标签

相似分享

官网相关资源

关于
我们是谁
投资者关系
意法半导体可持续发展举措
创新与技术
意法半导体官网
联系我们
联系ST分支机构
寻找销售人员和分销渠道
社区
媒体中心
活动与培训
隐私策略
隐私策略
Cookies管理
行使您的权利
官方最新发布
STM32N6 AI生态系统
STM32MCU,MPU高性能GUI
ST ACEPACK电源模块
意法半导体生物传感器
STM32Cube扩展软件包
关注我们
st-img 微信公众号
st-img 手机版