FreeRTOS消息邮箱

#include "FreeRTOS.h"
#include "task.h"
#include "queue.h"
#include "os_start.h"

#define START_TASK_PRIO		1
#define START_STK_SIZE 		128  
TaskHandle_t StartTask_Handler;
void start_task(void *pvParameters);

#define LED1_TASK_PRIO		2
#define LED1_STK_SIZE 		50  
TaskHandle_t LED1Task_Handler;
void led1_task(void *pvParameters);


#define KEYPROCESS_TASK_PRIO		3
#define KEYPROCESS_STK_SIZE 		50  
TaskHandle_t KeyprocessTask_Handler;
void Keyprocess_task(void *pvParameters);

#define KEYMSG_Q_LENGTH	1   //队列长度
#define RXMSG_Q_LENGTH	4

#define RX_BUF_NUM          30
uint8_t Rx_Buf[RX_BUF_NUM];
uint8_t Rx_HAL_Buff_Temp[2];	//HAL函数接收临时缓存

QueueHandle_t Key_Queue;			//按键队列
QueueHandle_t UartRX_Queue;			//消息队列


void os_start(void)
{
	User_SysTick_ReConfig(configTICK_RATE_HZ);

	//创建开始任务
    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();           //进入临界区
	
	//创建队列
	Key_Queue = xQueueCreate(KEYMSG_Q_LENGTH,1);
	UartRX_Queue = xQueueCreate(RXMSG_Q_LENGTH,1);
  HAL_UART_Receive_IT(&huart6,Rx_HAL_Buff_Temp,1);
		
  //创建LED1任务
  xTaskCreate((TaskFunction_t )led1_task,     
              (const char*    )"led1_task",   
              (uint16_t       )LED1_STK_SIZE, 
              (void*          )NULL,
              (UBaseType_t    )LED1_TASK_PRIO,
              (TaskHandle_t*  )&LED1Task_Handler);        
  //创建key任务
  xTaskCreate((TaskFunction_t )Keyprocess_task,     	
              (const char*    )"keyprocess_task",   	
              (uint16_t       )KEYPROCESS_STK_SIZE, 
              (void*          )NULL,				
              (UBaseType_t    )KEYPROCESS_TASK_PRIO,	
              (TaskHandle_t*  )&KeyprocessTask_Handler);   

  vTaskDelete(StartTask_Handler); //删除开始任务
  taskEXIT_CRITICAL();            //退出临界区
}  

//LED1任务函数
void led1_task(void *pvParameters)
{
  BaseType_t err;
  
  while(1)
  {
    //按键检测
    if(KeyScan((uint8_t *)&KeyValue) == 0)
    {
      if((KeyValue > 0) && (KeyValue < 17))
      {
        err = xQueueSend(Key_Queue,(uint8_t *)&KeyValue,10);
        if(err != pdTRUE)
        {
          printf("Queue Send Fail!\r\n");
        }
        KeyValue = 0;
      }
    }
    vTaskDelay(10);
  }
}

//key任务函数
void Keyprocess_task(void *pvParameters)
{
  BaseType_t err;
  uint8_t keyVal;
  
  while(1)
  {
    if(Key_Queue != NULL)
    {
      err = xQueueReceive(Key_Queue,&keyVal,portMAX_DELAY);
      if(err == pdTRUE)
      {
        printf("KeyVal = %d\r\n",keyVal);
      }
    }
    else
    {
      vTaskDelay(1000);
    }
  }
}

//发生溢出中断或DMA中断时
void HAL_UART_ErrorCallback(UART_HandleTypeDef *huart)
{
  huart->ErrorCode = HAL_UART_ERROR_NONE;
  HAL_UART_Receive_IT(&huart6,Rx_HAL_Buff_Temp,1);
}
//接收到1字节数据,回调函数如下
void HAL_UART_RxCpltCallback(UART_HandleTypeDef *huart)
{
  BaseType_t err;
  BaseType_t pxHigherPriorityTaskWoken;
  
  if(huart->Instance==USART6)
  {
    //加上自己的代码
    if(UartRX_Queue != NULL)
    {
      err = xQueueSendFromISR(UartRX_Queue,Rx_Buf,&pxHigherPriorityTaskWoken);
      if(err == pdTRUE)
      {
        HAL_UART_Receive_IT(&huart6,Rx_HAL_Buff_Temp,1);
      }
      else
      {
        printf("Queue Send ISR Fail!\r\n");
      }
      portYIELD_FROM_ISR(pxHigherPriorityTaskWoken);
    }
  	HAL_UART_Receive_IT(&huart6,Rx_HAL_Buff_Temp,1);
  } 
}

void HAL_TIM_PeriodElapsedCallback(TIM_HandleTypeDef *htim)
{
  /* USER CODE BEGIN Callback 0 */
  BaseType_t err;
  BaseType_t pxHigherPriorityTaskWoken;
  /* USER CODE END Callback 0 */
  if (htim->Instance == TIM1) {
    HAL_IncTick();
  }
  /* USER CODE BEGIN Callback 1 */
  else if (htim->Instance == TIM2)
  {
    if(UartRX_Queue != NULL)
    {
      err = xQueueReceiveFromISR(UartRX_Queue,ReceiveBuf,&pxHigherPriorityTaskWoken);
      if(err == pdTRUE)
      {
        printf("Receive:%s\r\n",ReceiveBuf);
      }
      else
      {
        printf("Receive Failed!\r\n");
      }
      portYIELD_FROM_ISR(pxHigherPriorityTaskWoken);
    }
  }
  /* USER CODE END Callback 1 */
}
  • 0
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
FreeRTOS消息队列是一种用于任务间通信的机制,可以在任务之间传递消息。队列中可以存储有限的、大小固定的数据项目,任务与任务、任务与中断之间要交流的数据保存在队列中,叫做队列项目。队列所能保存的最大数据项目数量叫做队列的长度,创建队列的时候会指定数据项目的大小和队列的长度。FreeRTOS中的信号量也是依据队列实现的,所以有必要深入了解FreeRTOS的队列。 在FreeRTOS中,可以使用xQueueCreate()函数创建一个队列,该函数需要传入两个参数:队列的长度和每个队列项目的大小。创建队列后,可以使用xQueueSend()函数向队列中发送数据,使用xQueueReceive()函数从队列中接收数据。此外,还可以使用xQueuePeek()函数查看队列中的下一个数据项,而不将其从队列中删除。 下面是一个简单的示例,演示如何在FreeRTOS中使用消息队列: ```c #include "FreeRTOS.h" #include "task.h" #include "queue.h" #define QUEUE_LENGTH 5 #define ITEM_SIZE sizeof(int) void vSenderTask(void *pvParameters) { QueueHandle_t xQueue; int i = 0; // 创建一个长度为QUEUE_LENGTH,每个项目大小为ITEM_SIZE的队列 xQueue = xQueueCreate(QUEUE_LENGTH, ITEM_SIZE); while (1) { // 向队列中发送数据 xQueueSend(xQueue, &i, 0); // 延时一段时间 vTaskDelay(pdMS_TO_TICKS(500)); i++; } } void vReceiverTask(void *pvParameters) { QueueHandle_t xQueue; int iReceivedValue; // 获取发送任务创建的队列句柄 xQueue = (QueueHandle_t)pvParameters; while (1) { // 从队列中接收数据 xQueueReceive(xQueue, &iReceivedValue, portMAX_DELAY); // 处理接收到的数据 printf("Received value: %d\n", iReceivedValue); } } int main(void) { QueueHandle_t xQueue; // 创建一个长度为QUEUE_LENGTH,每个项目大小为ITEM_SIZE的队列 xQueue = xQueueCreate(QUEUE_LENGTH, ITEM_SIZE); // 创建发送任务 xTaskCreate(vSenderTask, "Sender", configMINIMAL_STACK_SIZE, NULL, tskIDLE_PRIORITY + 1, NULL); // 创建接收任务 xTaskCreate(vReceiverTask, "Receiver", configMINIMAL_STACK_SIZE, (void *)xQueue, tskIDLE_PRIORITY + 1, NULL); // 启动调度器 vTaskStartScheduler(); return 0; } ```

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值