cubeMX+STM32+Freertos 读队列时阻塞

读队列时阻塞

本例内容是创建一个队列,由多个任务往队列中写数据,以及从队列中把数据读出。这个队列创建出来保存 uint16_t 型数据单元。往队列中写数据的任务没有设定阻塞超时时间,而读队列的任务设定了超时时间。往队列中写数据的任务的优先级低于读队列任务的优先级。这意味着队列中永远不会保持超过一个的数据单元。因为一旦有数据被写入队列,读队列任务立即解除阻塞,抢占写队列任务,并从队列中接收数据,同时数据从队列中删除—队列再一次变为空队列。

cubeMX的配置方法

任务01不停地往队列中写数值 100,而任务02不停地往队列中写入数值 200,且任务01与任务02两者优先级相同。任务03读队列任务设定了 100 毫秒的阻塞超时时间,所以会进入阻塞态以等待队列数据有效。一旦队列中数据单元有效,或者即使队列数据无效但等待时间超过 100 毫秒,此任务将会解除阻塞。在本例中,将永远不会出现 100 毫秒超时,因为任务01和任务02在不停地往队列中写数据。

cubemx:版本5.2.1
STM32:stm32f103zet6
系统时钟:72M
辅助工具:串口助手
配置方法如下图所示:
1、RCC配置
RCC配置
2、SYS配置SYS配置
3、串口配置USART配置
4、FreeRTOS Task和Queue的创建Task和Queue创建
5、FreeRTOS配置Freertos配置

代码实现

任务01具体代码

/**
  * @brief  Function implementing the myTask01 thread.
  * @param  argument: Not used 
  * @retval None
  */
/* USER CODE END Header_StartTask01 */
void StartTask01(void const * argument)
{

  /* USER CODE BEGIN StartTask01 */
  /* Infinite loop */
	osStatus putStatus1;
  for(;;)
  {
    //往队列中写入100
    putStatus1=osMessagePut (myQueue01Handle,100,0);
		if(putStatus1 != osOK)
		{
			printf("Could not send to the queue.\r\n");
		}
		
		taskYIELD();
  }
	
  /* USER CODE END StartTask01 */
}

任务02具体代码

/* USER CODE BEGIN Header_StartTask02 */
/**
* @brief Function implementing the myTask02 thread.
* @param argument: Not used
* @retval None
*/
/* USER CODE END Header_StartTask02 */
void StartTask02(void const * argument)
{
  /* USER CODE BEGIN StartTask02 */
  /* Infinite loop */
	osStatus putStatus2;
  for(;;)
  {
  //往队列中写入200
    putStatus2=osMessagePut (myQueue01Handle,200,0);
		if(putStatus2 != osOK)
		{
			printf("Could not send to the queue.\r\n");
		}
		
		taskYIELD();
  }
  /* USER CODE END StartTask02 */
}

任务03具体代码

/* USER CODE BEGIN Header_StartTask03 */
/**
* @brief Function implementing the myTask03 thread.
* @param argument: Not used
* @retval None
*/
/* USER CODE END Header_StartTask03 */
void StartTask03(void const * argument)
{
  /* USER CODE BEGIN StartTask03 */
  /* Infinite loop */
	osEvent event;

  for(;;)
  {
		if( osMessageWaiting( myQueue01Handle ) != 0 )
		{
			printf( "Queue should have been empty!\r\n" );
		}
		
		event = osMessageGet( myQueue01Handle,100);
		
		if( event.status == osEventMessage )
		{
			/* 成功读出数据,并打印出来 */
			printf( "Received = %d\r\n", event.value.v);
		}
		else
		{
			/* 等待100ms也没有接收到任何数据
			必然存在错误,因为发送任务不停地往队列中写入数据 */
			printf( "Could not receive from the queue.\r\n" );
		}
  }
  /* USER CODE END StartTask03 */
}

主程序具体代码

/**
  * @brief  The application entry point.
  * @retval int
  */
int main(void)
{
  /* USER CODE BEGIN 1 */

  /* USER CODE END 1 */
  

  /* MCU Configuration--------------------------------------------------------*/

  /* Reset of all peripherals, Initializes the Flash interface and the Systick. */
  HAL_Init();

  /* USER CODE BEGIN Init */

  /* USER CODE END Init */

  /* Configure the system clock */
  SystemClock_Config();

  /* USER CODE BEGIN SysInit */

  /* USER CODE END SysInit */

  /* Initialize all configured peripherals */
  MX_GPIO_Init();
  MX_USART1_UART_Init();
  /* USER CODE BEGIN 2 */


  /* USER CODE END 2 */

  /* Call init function for freertos objects (in freertos.c) */
  MX_FREERTOS_Init();

  /* Start scheduler */
  osKernelStart();
  
  /* We should never get here as control is now taken by the scheduler */

  /* Infinite loop */
  /* USER CODE BEGIN WHILE */
  while (1)
  {
    /* USER CODE END WHILE */

    /* USER CODE BEGIN 3 */
		
  }
  /* USER CODE END 3 */
}

任务01、02、03的程序均在freertos.c中添加即可

实现效果

结果分析:写队列任务在每次循环中都调用 taskYIELD()。 taskYIELD()通知调度器立即进行任务切换,而不必等到当前任务的时间片耗尽。某个任务调用 taskYIELD()等效于其自愿放弃运行态。由于本例中两个写队列任务具有相同的任务优先级,所以一旦其中一个任务调用了 taskYIELD(),另一个任务将会得到执行 — 调用 taskYIELD()的任务转移到就绪态,同时另一个任务进入运行态。这样就可以使得这两个任务轮翻地往队列发送数据。通过串口调试助手可以看出成功读出队列数据。
串口调试界面
不足之处请指出,有点帮助请鼓励!

  • 1
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

DHZFSYJ

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值