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


    引言

    我们在裸机开发中,每个函数之间进行数据通信往往采用全局变量。而在嵌入式开发中。我们在进行进程间通信的时候,往往采用消息队列。对于操作系统来说,消息队列是非常重要的一个数据结构。本文将介绍一下,如何使用消息队列进行通信。

    介绍

    消息队列概念

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

    FreeRTOS中的消息队列函数

    1. 设定消息队列的格式:osMessageQDef(myQueue, len, size);
    • myQueue是消息队列的名称。
    • len是消息队列的长度(有几个消息)
    • size是每个消息的大小,也就是每个元素的格式
    1. 创建消息:osMessageCreate(osMessageQ(myQueue01), NULL);
      创建消息的函数,实际上是调用了FreeRTOS的osMessageCreate()函数,只不过HAL库进行了封装。
    2. 向消息队列发送消息
      我们这里来介绍在中断中发送消息。使用函数xQueueSendFromISR(QueueHandle,&Res,time);
      其中:
    • QueueHandle:消息队列的句柄
    • &Res:要发送的数据的地址
    • time:阻塞时间,就是如果消息队列满的时候,任务应该阻塞多久
    1. 接收消息队列中的消息
      xQueueReceive(QueueHandle,&queue_buffer,time);
    • QueueHandle:消息队列的句柄
    • &queue_buffer:接收的消息要存放在的地址
    • time:阻塞时间,就是如果消息队列空的时候,任务应该阻塞多久
    1. 查询消息队列中消息的数量
      uxQueueMessagesWaiting(myQueue01Handle),可以返回消息队列(句柄为myQueue01Handle)中消息的数量,返回值为整数。

    实例

    需求分析

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

    发送消息

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

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

    接收消息

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

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

    现象

    PC端发送123456789,MCU回复。

    Receive a frame data
    count 8,LEDTask Receive 12345678

    后续

    欢迎关注公众号:物联网知识
    其中使用到的软件均可在公众号回复关键字获得。

  • 相关阅读:
    Java面试题-Redis-第三天(缓存更新策略-由旁路缓存策略衍生出的一系列问题)
    故障诊断 | MATLAB实现GRNN广义回归神经网络故障诊断
    文本摘要实战:基于句子相似度矩阵构建图结构实现文本摘要 代码+数据
    C++之结构体以及通讯录管理系统
    【Unity3D】灯光组件Light
    入侵事件应急案例
    Transformer模型对应的Tokenizer类型
    Coursera自动驾驶1.5——纵向控制
    智能汽车HMI“火了”
    CS231n-2022 Module1: 神经网络3:Learning之参数更新
  • 原文地址:https://blog.csdn.net/qq_44629109/article/details/127492641