STM32与Freertos入门(六)队列

1、队列介绍

队列是FreeRTOS提供的一种重要的通信机制,用于在任务之间传递数据。

FreeRTOS队列是一种先进先出(FIFO)的数据结构,用于在任务之间传递消息或数据项。它允许一个任务将数据项发送到队列,而另一个任务则可以从队列中接收这些数据项。

以下是FreeRTOS队列的一些重要特点和用法:

  1. 队列创建:可以使用FreeRTOS提供的API函数创建队列。在创建队列时,需要指定队列的最大长度和每个数据项的大小。

  2. 发送数据:任务可以使用xQueueSend()函数将数据项发送到队列中。如果队列已满,发送操作将被阻塞,直到有空间可用。

  3. 接收数据:任务可以使用xQueueReceive()函数从队列中接收数据项。如果队列为空,接收操作将被阻塞,直到有数据可用。

  4. 队列优先级:FreeRTOS队列还支持优先级功能。可以为队列设置优先级,使得在同时有多个任务等待发送或接收数据时,根据优先级决定哪个任务先进行操作。

  5. 队列长度:通过查询队列的长度,可以了解当前队列中待处理的数据项数量。这对于任务调度和资源管理非常有用。

使用FreeRTOS队列时,需要注意以下几点:

  • 需要合理设置队列的长度,确保不会超过任务处理能力。
  • 当队列满或空时,任务可能会被阻塞,因此要小心处理该种情况以避免死锁。
  • 在多任务环境下,要注意同步和竞争条件,以确保数据的正确性和一致性。

总的来说,FreeRTOS队列提供了一种方便和高效的任务间通信机制,可以在嵌入式系统中实现数据传递和同步。它是FreeRTOS强大功能的一部分,帮助开发人员编写可靠的实时应用程序。

2、队列测试

2.1 任务创建

2.2 队列建立

3、队列API

 3.1 创建队列

QueueHandle_t xQueueCreate( UBaseType_t uxQueueLength, UBaseType_t uxItemSize );
  • 参数:
    • uxQueueLength:队列可同时容纳的最大项目数 。
    • uxItemSize:存储队列中的每个数据项所需的大小(以字节为单位)。
  • 返回值: 
    • 如果队列创建成功,则返回所创建队列的句柄 。 如果创建队列所需的内存无法分配,则返回 NULL。

 3.2 写队列

BaseType_t xQueueSend(
              QueueHandle_t xQueue,
              const void * pvItemToQueue,
              TickType_t xTicksToWait
            );
  • 参数:
    • xQueue:队列的句柄,数据项将发送到此队列。
    • pvItemToQueue:待写入数据
    • xTicksToWait:阻塞超时时间
  • 返回值:
    • 如果成功写入数据,返回 pdTRUE,否则返回 errQUEUE_FULL。

  3.3读队列

BaseType_t xQueueReceive(
               QueueHandle_t xQueue,
               void *pvBuffer,
               TickType_t xTicksToWait
             );
  • 参数:
    • xQueue:待读取的队列
    • pvItemToQueue:数据读取缓冲区
    • xTicksToWait:阻塞超时时间
  • 返回值:
    • 成功返回 pdTRUE,否则返回 pdFALSE。

4、KEIL 5代码设计

4.1 写队列函数

void StartTask_send(void const * argument)
{
  /* USER CODE BEGIN StartTask_send */
	uint16_t buf = 100;
	BaseType_t status;
  /* Infinite loop */
  for(;;)
  {
		if(key==0){ 
			osDelay(20);
			if(key==0){
				printf("key按下\r\n");
				status=xQueueSend(myQueueHandle,&buf,0);
				if(status == pdTRUE){
					printf("写入队列成功,写入值%d\r\n",buf);
				}
				else{
					printf("写入失败\r\n");
				}
			}	
			while(key==0);
		}
		osDelay(10);
  }
  /* USER CODE END StartTask_send */
}

4.2 读队列函数

void StartTask_receive(void const * argument)
{
  /* USER CODE BEGIN StartTask_receive */
	uint16_t buf;
	BaseType_t status;
  /* Infinite loop */
  for(;;)
  {
    if(key2==0){ 
			osDelay(20);
			if(key2==0){
				printf("key2按下\r\n");
				status=xQueueReceive(myQueueHandle,&buf,0);
				if(status == pdTRUE){
					printf("读取队列成功,读出值%d\r\n",buf);
				}
				else{
					printf("读取失败\r\n");
				}
			}	
			while(key2==0);
		}
		osDelay(10);
  }
  /* USER CODE END StartTask_receive */
}

freertos.c代码如下:

/* USER CODE BEGIN Header */
/**
  ******************************************************************************
  * File Name          : freertos.c
  * Description        : Code for freertos applications
  ******************************************************************************
  * @attention
  *
  * Copyright (c) 2023 STMicroelectronics.
  * All rights reserved.
  *
  * This software is licensed under terms that can be found in the LICENSE file
  * in the root directory of this software component.
  * If no LICENSE file comes with this software, it is provided AS-IS.
  *
  ******************************************************************************
  */
/* USER CODE END Header */

/* Includes ------------------------------------------------------------------*/
#include "FreeRTOS.h"
#include "task.h"
#include "main.h"
#include "cmsis_os.h"

/* Private includes ----------------------------------------------------------*/
/* USER CODE BEGIN Includes */

/* USER CODE END Includes */

/* Private typedef -----------------------------------------------------------*/
/* USER CODE BEGIN PTD */

/* USER CODE END PTD */

/* Private define ------------------------------------------------------------*/
/* USER CODE BEGIN PD */

/* USER CODE END PD */

/* Private macro -------------------------------------------------------------*/
/* USER CODE BEGIN PM */

/* USER CODE END PM */

/* Private variables ---------------------------------------------------------*/
/* USER CODE BEGIN Variables */

/* USER CODE END Variables */
osThreadId Task_sendHandle;
osThreadId Task_receiveHandle;
osMessageQId myQueueHandle;

/* Private function prototypes -----------------------------------------------*/
/* USER CODE BEGIN FunctionPrototypes */

/* USER CODE END FunctionPrototypes */

void StartTask_send(void const * argument);
void StartTask_receive(void const * argument);

void MX_FREERTOS_Init(void); /* (MISRA C 2004 rule 8.1) */

/* GetIdleTaskMemory prototype (linked to static allocation support) */
void vApplicationGetIdleTaskMemory( StaticTask_t **ppxIdleTaskTCBBuffer, StackType_t **ppxIdleTaskStackBuffer, uint32_t *pulIdleTaskStackSize );

/* USER CODE BEGIN GET_IDLE_TASK_MEMORY */
static StaticTask_t xIdleTaskTCBBuffer;
static StackType_t xIdleStack[configMINIMAL_STACK_SIZE];

void vApplicationGetIdleTaskMemory( StaticTask_t **ppxIdleTaskTCBBuffer, StackType_t **ppxIdleTaskStackBuffer, uint32_t *pulIdleTaskStackSize )
{
  *ppxIdleTaskTCBBuffer = &xIdleTaskTCBBuffer;
  *ppxIdleTaskStackBuffer = &xIdleStack[0];
  *pulIdleTaskStackSize = configMINIMAL_STACK_SIZE;
  /* place for user code */
}
/* USER CODE END GET_IDLE_TASK_MEMORY */

/**
  * @brief  FreeRTOS initialization
  * @param  None
  * @retval None
  */
void MX_FREERTOS_Init(void) {
  /* USER CODE BEGIN Init */

  /* USER CODE END Init */

  /* USER CODE BEGIN RTOS_MUTEX */
  /* add mutexes, ... */
  /* USER CODE END RTOS_MUTEX */

  /* USER CODE BEGIN RTOS_SEMAPHORES */
  /* add semaphores, ... */
  /* USER CODE END RTOS_SEMAPHORES */

  /* USER CODE BEGIN RTOS_TIMERS */
  /* start timers, add new ones, ... */
  /* USER CODE END RTOS_TIMERS */

  /* Create the queue(s) */
  /* definition and creation of myQueue */
  osMessageQDef(myQueue, 16, uint16_t);
  myQueueHandle = osMessageCreate(osMessageQ(myQueue), NULL);

  /* USER CODE BEGIN RTOS_QUEUES */
  /* add queues, ... */
  /* USER CODE END RTOS_QUEUES */

  /* Create the thread(s) */
  /* definition and creation of Task_send */
  osThreadDef(Task_send, StartTask_send, osPriorityNormal, 0, 128);
  Task_sendHandle = osThreadCreate(osThread(Task_send), NULL);

  /* definition and creation of Task_receive */
  osThreadDef(Task_receive, StartTask_receive, osPriorityBelowNormal, 0, 128);
  Task_receiveHandle = osThreadCreate(osThread(Task_receive), NULL);

  /* USER CODE BEGIN RTOS_THREADS */
  /* add threads, ... */
  /* USER CODE END RTOS_THREADS */

}

/* USER CODE BEGIN Header_StartTask_send */
/**
  * @brief  Function implementing the Task_send thread.
  * @param  argument: Not used
  * @retval None
  */
/* USER CODE END Header_StartTask_send */
void StartTask_send(void const * argument)
{
  /* USER CODE BEGIN StartTask_send */
	uint16_t buf = 100;
	BaseType_t status;
  /* Infinite loop */
  for(;;)
  {
		if(key==0){ 
			osDelay(20);
			if(key==0){
				printf("key按下\r\n");
				status=xQueueSend(myQueueHandle,&buf,0);
				if(status == pdTRUE){
					printf("写入队列成功,写入值%d\r\n",buf);
				}
				else{
					printf("写入失败\r\n");
				}
			}	
			while(key==0);
		}
		osDelay(10);
  }
  /* USER CODE END StartTask_send */
}

/* USER CODE BEGIN Header_StartTask_receive */
/**
* @brief Function implementing the Task_receive thread.
* @param argument: Not used
* @retval None
*/
/* USER CODE END Header_StartTask_receive */
void StartTask_receive(void const * argument)
{
  /* USER CODE BEGIN StartTask_receive */
	uint16_t buf;
	BaseType_t status;
  /* Infinite loop */
  for(;;)
  {
    if(key2==0){ 
			osDelay(20);
			if(key2==0){
				printf("key2按下\r\n");
				status=xQueueReceive(myQueueHandle,&buf,0);
				if(status == pdTRUE){
					printf("读取队列成功,读出值%d\r\n",buf);
				}
				else{
					printf("读取失败\r\n");
				}
			}	
			while(key2==0);
		}
		osDelay(10);
  }
  /* USER CODE END StartTask_receive */
}
/* Private application code --------------------------------------------------*/
/* USER CODE BEGIN Application */

/* USER CODE END Application */

  • 0
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: STM32是一款32位嵌入式微控制器系列,其中HAL(硬件抽象层)是一个软件库,用于与STM32微控制器的硬件进行高级的交互操作。FreeRTOS(实时操作系统)是一个开源的实时操作系统内核,用于实现多任务处理和任务调度。串口消息队列是HAL和FreeRTOS的结合应用,用于实现串口通信过程中的消息传递和任务调度。 在STM32中,通过HAL库中的串口接口可以实现与其他设备的串口通信。而使用FreeRTOS的串口消息队列,可以将串口接收到的消息经过处理后,以队列形式存储,等待任务调度进行处理。该消息队列可以是有大小限制的,根据实际需求进行设置。 当串口收到消息时,HAL库会触发中断,将接收到的数据存储到缓冲区中。然后,通过FreeRTOS的任务和队列机制,将数据从缓冲区移动到消息队列中。任务可以从消息队列中获取消息,并进行处理,例如解析数据、执行相应的操作等。 在使用STM32 HAL与FreeRTOS的串口消息队列时,需要注意以下几点: 1. 配置串口参数:使用HAL库的API函数配置串口的通信参数,如波特率、数据位、停止位等。 2. 创建任务:使用FreeRTOS的任务创建函数创建串口任务,指定任务优先级和任务堆栈大小。 3. 创建消息队列:使用FreeRTOS的消息队列创建函数创建串口消息队列,指定消息队列的长度和消息大小。 4. 中断处理:在串口中断中,通过HAL库提供的函数将接收到的数据存储到缓冲区中。 5. 任务调度:使用FreeRTOS的任务调度机制,从缓冲区中将数据移动到消息队列中,并让任务从消息队列中获取消息进行处理。 6. 数据处理:任务从消息队列中获取消息后,根据消息进行相应的数据处理和操作。 通过STM32 HAL与FreeRTOS的串口消息队列,可以实现高效的串口通信和任务调度。这样可以将不同的串口任务分配给不同的任务进行处理,提高系统的实时性和并行处理能力。同时,通过消息队列机制,可以避免数据的丢失和冲突,提高系统的稳定性和可靠性。 ### 回答2: stm32是一款基于ARM Cortex-M处理器的微控制器系列,具有强大的性能和丰富的外设功能。HAL (Hardware Abstraction Layer)是一种抽象硬件层,提供了简化外设驱动程序编写的API接口。FreeRTOS是一款开源的实时操作系统,能够提供任务调度和同步机制。 在使用stm32 hal freertos时,可以通过串口消息队列来实现任务间的通信。消息队列是一种先进先出的数据结构,用于在任务之间传递数据。 首先,需要初始化一个消息队列,通过调用FreeRTOS提供的API函数来创建。可以指定队列的大小和每个消息的大小。 然后,可以在发送任务中使用hal库提供的串口发送函数,将需要传递的消息发送给消息队列。发送任务可以使用FreeRTOS提供的队列发送函数来向消息队列发送消息。 接收任务可以使用hal库提供的串口接收函数,然后调用FreeRTOS提供的队列接收函数来从消息队列中接收消息。 通过消息队列,发送任务可以将消息放入队列,接收任务可以从队列中获取消息。这样就实现了任务间的通信。 需要注意的是,由于消息队列是有限大小的,当队列已满时,发送任务可能会被阻塞。类似地,当队列为空时,接收任务可能会被阻塞。因此,在使用串口消息队列时,需要合理地设计队列的大小,以免造成问题。 总之,通过使用stm32 hal freertos和串口消息队列,可以方便地实现任务间的通信,实现数据的传递和处理。这对于复杂的嵌入式系统开发是非常有用的。 ### 回答3: stm32是一种微控制器品牌,HAL(Hardware Abstraction Layer)是一种硬件抽象层,FreeRTOS是一款开源的实时操作系统。 在使用stm32微控制器时,我们可以使用HAL库来简化与硬件的交互。针对串口通信,HAL库提供了一系列的函数,使我们可以方便地通过串口与其他设备进行数据传输。 而FreeRTOS则可以帮助我们实现任务调度和管理。在使用串口进行数据传输时,我们可以利用FreeRTOS提供的消息队列特性来实现线程间的高效通信。 消息队列是一种用来在不同任务之间传递数据的机制。当一个任务需要发送数据时,它可以将数据封装成一个消息并将其发送到消息队列中。其他任务可以从消息队列中接收消息,并根据接收到的消息做出相应的处理。 使用HAL库和FreeRTOS时,我们可以通过以下步骤来实现串口消息队列: 1. 初始化串口:使用HAL库中的函数初始化串口,设置好串口的参数和通信速率。 2. 创建消息队列:使用FreeRTOS中的函数创建一个消息队列,设置好队列的长度和消息的大小。 3. 发送消息:在需要发送数据的任务中,使用FreeRTOS提供的函数将数据封装成消息,并将消息发送到消息队列中。 4. 接收消息:在需要接收数据的任务中,使用FreeRTOS提供的函数从消息队列中接收消息,并根据接收到的消息进行处理。 通过以上步骤,我们可以实现多个任务之间的串口通信,并且保证数据的安全传输和高效处理。 总结起来,使用stm32的HAL库和FreeRTOS的消息队列特性,可以方便地实现串口通信,并且保证多任务间的高效通信。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值