0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

Free RTOS的信息隊(duì)列

汽車電子技術(shù) ? 來(lái)源:玩轉(zhuǎn)單片機(jī) ? 作者: Julian ? 2023-02-10 16:01 ? 次閱讀

隊(duì)列用于在任務(wù)之間以及任務(wù)與中斷之間傳遞數(shù)據(jù),可以在調(diào)度程序啟動(dòng)之前或之后創(chuàng)建隊(duì)列。

配置相關(guān)資源

#define configSUPPORT_DYNAMIC_ALLOCATION 1

創(chuàng)建信息隊(duì)列

QueueHandle_t xQueueCreate( UBaseType_t uxQueueLength, 
                              UBaseType_t uxItemSize );

參數(shù)

uxQueueLength:正在創(chuàng)建的隊(duì)列在任何時(shí)候可以容納的最大項(xiàng)數(shù)

uxItemSize:可以存儲(chǔ)在隊(duì)列中的每個(gè)數(shù)據(jù)項(xiàng)的大小(以字節(jié)為單位)

返回值

創(chuàng)建失敗返回NULL,創(chuàng)建成功返回句柄

發(fā)送信息

BaseType_t xQueueSend( QueueHandle_t xQueue, 
                       const void * pvItemToQueue, 
                       TickType_t xTicksToWait );

參數(shù)

xQueue:數(shù)據(jù)被發(fā)送(寫入)到的隊(duì)列的句柄

pvItemToQueue:一個(gè)指向要復(fù)制到隊(duì)列中的數(shù)據(jù)的指針

xTicksToWait:當(dāng)隊(duì)列已經(jīng)滿時(shí),任務(wù)保持阻塞狀態(tài)以等待隊(duì)列上的空間可用的最大時(shí)間

返回值

成功返回pdPASS,失敗返回errQUEUE_FULL

接收信息

BaseType_t xQueueReceive( QueueHandle_t xQueue, 
                          void*pvBuffer, 
                          TickType_txTicksToWait );

參數(shù)

xQueue:從其接收數(shù)據(jù)(讀)的隊(duì)列的句柄

pvBuffer:一個(gè)指向內(nèi)存的指針,接收到的數(shù)據(jù)將被復(fù)制到其中

xTicksToWait:任務(wù)保持阻塞狀態(tài)以等待隊(duì)列上的數(shù)據(jù)可用的最大時(shí)間,如果隊(duì)列已經(jīng)為空

返回值

成功返回pdPASS,失敗返回errQUEUE_FULL

注意:更多API函數(shù)請(qǐng)參閱官方相關(guān)文檔

簡(jiǎn)單程序

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

void LED_Init(void)
{
  GPIO_InitTypeDef GPIO_InitStructure;              //定義結(jié)構(gòu)體變量
  
  RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC,ENABLE);  //開(kāi)啟時(shí)鐘
  
  GPIO_InitStructure.GPIO_Pin=GPIO_Pin_0|GPIO_Pin_1;    //選擇你要設(shè)置的IO口
  GPIO_InitStructure.GPIO_Mode=GPIO_Mode_Out_PP;      //設(shè)置推挽輸出模式
  GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz;     //設(shè)置傳輸速率
  GPIO_Init(GPIOC,&GPIO_InitStructure);                //初始化GPIO
  
  GPIO_SetBits(GPIOC,GPIO_Pin_0|GPIO_Pin_1);            //將LED端口拉高,熄滅LED
}

void KEY_Init(void)
{
  GPIO_InitTypeDef GPIO_InitStructure; //定義結(jié)構(gòu)體變量  
  RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA|RCC_APB2Periph_GPIOE,ENABLE);
  
  GPIO_InitStructure.GPIO_Pin=GPIO_Pin_0;     //選擇你要設(shè)置的IO口
  GPIO_InitStructure.GPIO_Mode=GPIO_Mode_IPD;//下拉輸入  
  GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz;     //設(shè)置傳輸速率
  GPIO_Init(GPIOA,&GPIO_InitStructure);      /* 初始化GPIO */
  
  GPIO_InitStructure.GPIO_Pin=GPIO_Pin_3|GPIO_Pin_2|GPIO_Pin_4;
  GPIO_InitStructure.GPIO_Mode=GPIO_Mode_IPU;  //上拉輸入
  GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz;
  GPIO_Init(GPIOE,&GPIO_InitStructure);
}


void USART_init(uint32_t bound)
{
  GPIO_InitTypeDef GPIO_InitStruct;   //定義GPIO結(jié)構(gòu)體變量
  USART_InitTypeDef USART_InitStruct;   //定義串口結(jié)構(gòu)體變量
  
  RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA|RCC_APB2Periph_USART1,ENABLE);   //使能GPIOC的時(shí)鐘
  
  GPIO_InitStruct.GPIO_Pin=GPIO_Pin_9;   //配置TX引腳
  GPIO_InitStruct.GPIO_Mode=GPIO_Mode_AF_PP;   //配置PA9為復(fù)用推挽輸出
  GPIO_InitStruct.GPIO_Speed=GPIO_Speed_50MHz;   //配置PA9速率
  GPIO_Init(GPIOA,&GPIO_InitStruct);   //GPIO初始化函數(shù)
  
  GPIO_InitStruct.GPIO_Pin=GPIO_Pin_10;   //配置RX引腳
  GPIO_InitStruct.GPIO_Mode=GPIO_Mode_IN_FLOATING;   //配置PA10為浮空輸入
  GPIO_InitStruct.GPIO_Speed=GPIO_Speed_50MHz;   //配置PA10速率
  GPIO_Init(GPIOA,&GPIO_InitStruct);   //GPIO初始化函數(shù)
  
  
  USART_InitStruct.USART_Mode=USART_Mode_Tx|USART_Mode_Rx;   //發(fā)送接收模式
  USART_InitStruct.USART_Parity=USART_Parity_No;   //無(wú)奇偶校驗(yàn)
  USART_InitStruct.USART_BaudRate=bound;   //波特率
  USART_InitStruct.USART_StopBits=USART_StopBits_1;   //停止位1位
  USART_InitStruct.USART_WordLength=USART_WordLength_8b;   //字長(zhǎng)8位
  USART_InitStruct.USART_HardwareFlowControl=USART_HardwareFlowControl_None;   //無(wú)硬件數(shù)據(jù)流控制
  USART_Init(USART1,&USART_InitStruct);   //串口初始化函數(shù)
  
  USART_Cmd(USART1,ENABLE);   //使能USART1
}

int fputc(int ch,FILE *f)   //printf重定向函數(shù)
{
  USART_SendData(USART1,(uint8_t)ch);   //發(fā)送一字節(jié)數(shù)據(jù)
  while(USART_GetFlagStatus(USART1,USART_FLAG_TXE) == RESET);   //等待發(fā)送完成
  return ch;
}


#define START_TASK_PRIO 1      //任務(wù)優(yōu)先級(jí)
#define START_STK_SIZE 128      //任務(wù)堆棧大小
TaskHandle_t StartTask_Handler;   //任務(wù)句柄
void Start_Task(void *pvParameters);//任務(wù)函數(shù)

#define Sen_TASK_PRIO 2       //任務(wù)優(yōu)先級(jí)
#define Sen_STK_SIZE 50       //任務(wù)堆棧大小
TaskHandle_t SenTask_Handler;     //任務(wù)句柄
void Sen_Task(void *p_arg);     //任務(wù)函數(shù)

#define Queue_TASK_PRIO 3       //任務(wù)優(yōu)先級(jí)
#define Queue_STK_SIZE 50       //任務(wù)堆棧大小
TaskHandle_t QueueTask_Handler;   //任務(wù)句柄
void Queue_Task(void *p_arg);     //任務(wù)函數(shù)

QueueHandle_t xQueue = NULL;

int main( void ) 
{
  NVIC_PriorityGroupConfig(NVIC_PriorityGroup_4);    //設(shè)置系統(tǒng)中斷優(yōu)先級(jí)分組 4
  LED_Init(); //初始化 LED
  KEY_Init();
  USART_init(9600);
  //創(chuàng)建開(kāi)始任務(wù)
  xTaskCreate(
    (TaskFunction_t )Start_Task,     //任務(wù)函數(shù)
    (const char* )"Start_Task",     //任務(wù)名稱
    (uint16_t )START_STK_SIZE,       //任務(wù)堆棧大小
    (void* )NULL,             //傳遞給任務(wù)函數(shù)的參數(shù)
    (UBaseType_t )START_TASK_PRIO,     //任務(wù)優(yōu)先級(jí)
    (TaskHandle_t* )&StartTask_Handler  //任務(wù)句柄 
  );
  vTaskStartScheduler();  //開(kāi)啟調(diào)度
}
//開(kāi)始任務(wù)函數(shù)
void Start_Task(void *pvParameters)
{
  taskENTER_CRITICAL();   //進(jìn)入臨界區(qū)
  //創(chuàng)建一個(gè)隊(duì)列
  xQueue = xQueueCreate( 1, sizeof(u8)  );
  if( xQueue != NULL )
  {
    printf("創(chuàng)建成功n");
  }
  //創(chuàng)建 Sen 任務(wù)
  xTaskCreate(
    (TaskFunction_t )Sen_Task, 
    (const char* )"Sen_Task", 
    (uint16_t )Sen_STK_SIZE, 
    (void* )NULL,
    (UBaseType_t )Sen_TASK_PRIO,
    (TaskHandle_t* )&SenTask_Handler
  );
  //創(chuàng)建 Queue 任務(wù)
  xTaskCreate(
    (TaskFunction_t )Queue_Task, 
    (const char* )"Queue_Task", 
    (uint16_t )Queue_STK_SIZE, 
    (void* )NULL,
    (UBaseType_t )Queue_TASK_PRIO,
    (TaskHandle_t* )&QueueTask_Handler
  );
  vTaskDelete(StartTask_Handler); //刪除開(kāi)始任務(wù)
  taskEXIT_CRITICAL();   //退出臨界區(qū)
}

//Sen 任務(wù)函數(shù)
void Sen_Task(void *pvParameters)
{
  u8 Key = 0;
  while(1)
  {
    if( Key < 10 )
    {
      Key++;
    }
    else
    {
      Key = 0;
    }
    if( xQueue != NULL )
    {
      xQueueSend( ( QueueHandle_t ) xQueue,         //數(shù)據(jù)被發(fā)送(寫入)到的隊(duì)列的句柄
            ( void *        ) &Key,          //一個(gè)指向要復(fù)制到隊(duì)列中的數(shù)據(jù)的指針
            ( TickType_t    ) portMAX_DELAY   );  //當(dāng)隊(duì)列已經(jīng)滿時(shí),任務(wù)保持阻塞狀態(tài)以等待隊(duì)列上的空間可用的最大時(shí)間
    }
    vTaskDelay(1000);
  }
}

//Queue 任務(wù)函數(shù)
void Queue_Task(void *pvParameters)
{
  u8 Key_Receive = 0;
  while(1)
  {
    xQueueReceive( ( QueueHandle_t ) xQueue, 
             ( void *        ) &Key_Receive, 
             ( TickType_t    ) portMAX_DELAY );
    printf("%dn",Key_Receive);
    vTaskDelay(500);
  }
}

實(shí)驗(yàn)效果

poYBAGPl-cyAAdDAAABZZm7mpPM637.png

--END--


聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問(wèn)題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • 程序
    +關(guān)注

    關(guān)注

    116

    文章

    3762

    瀏覽量

    80755
  • 隊(duì)列
    +關(guān)注

    關(guān)注

    1

    文章

    46

    瀏覽量

    10882
  • 傳遞數(shù)據(jù)
    +關(guān)注

    關(guān)注

    0

    文章

    3

    瀏覽量

    6208
收藏 人收藏

    評(píng)論

    相關(guān)推薦

    RTOS信號(hào)量、隊(duì)列通信原理

    有深入理解RTOS原理,或閱讀過(guò)RTOS源碼的同學(xué)應(yīng)該知道:RTOS實(shí)現(xiàn)任務(wù)間通信通常是由一系列指針進(jìn)行操作實(shí)現(xiàn)的。
    發(fā)表于 08-16 10:07 ?1615次閱讀

    【CW32移植Free-RTOS】CW32開(kāi)發(fā)者扶持計(jì)劃

    CW32配置Free-RTOS全過(guò)程,CW32開(kāi)發(fā)者扶持計(jì)劃
    的頭像 發(fā)表于 04-18 09:38 ?6070次閱讀
    【CW32移植<b class='flag-5'>Free-RTOS</b>】CW32開(kāi)發(fā)者扶持計(jì)劃

    【MiCOKit試用體驗(yàn)】慶科MiCO系統(tǒng)篇(4)MiCO RTOS消息隊(duì)列

    本帖最后由 gjianw217 于 2015-10-25 15:50 編輯 在本帖子中,主要分析一下慶科MiCO RTOS的消息隊(duì)列,具體包括:OS消息隊(duì)列MiCO消息隊(duì)列關(guān)鍵A
    發(fā)表于 10-24 17:03

    STM32 + F103VE + Free RTOS 操作系統(tǒng)移植

    STM32 + F103VE + Free RTOS 操作系統(tǒng)移植
    發(fā)表于 03-04 20:03

    在energia 編譯器中使用free_rtos或者ti_rtos

    請(qǐng)問(wèn)在energia編譯器里,有使用free_rtos或者ti_rtos的案例程序嗎?
    發(fā)表于 05-14 11:11

    CC3200 可以使用 ccs TI-RTOS 和 iar FREE-RTOS,為什么要有兩個(gè)OS,他們有什么區(qū)別?

    本帖最后由 一只耳朵怪 于 2018-6-7 15:32 編輯 請(qǐng)教一下TI的工程師, CC3200 可以使用ccs TI-RTOS 和 iar FREE-RTOS。1、為什么要有兩個(gè)OS
    發(fā)表于 06-07 02:34

    FREE-RTOS中怎樣使用simplelink api函數(shù)

    您好: ??? 我現(xiàn)在正在學(xué)習(xí)cc3200?? FREE_RTOS系統(tǒng)demo。請(qǐng)問(wèn)我想在啟動(dòng)線程函數(shù)之前使用??? “文件操作系統(tǒng)的API函數(shù)”? ,讀取flash的一些參數(shù),請(qǐng)問(wèn)我應(yīng)該怎么操作?
    發(fā)表于 06-21 08:04

    RTT有類似UCOS,或者FREE RTOS這類的移植方法嗎?

    的操作把難度為3的操作,降低為難度1. 網(wǎng)上一堆工具使用教程,也都是基于自身已經(jīng)了解很多基礎(chǔ)知識(shí)上再寫的. RTT難道就沒(méi)有類似UCOS,或者FREE RTOS這類的移植方法?
    發(fā)表于 01-21 06:20

    求F4探索者板+FREE RTOS +LWIP的工程模板!

    好煩好煩,心里很挫敗,都不想搞嵌入式了。。另求一個(gè)F4探索者板+FREE RTOS +LWIP的工程模板,有做過(guò)這方面的朋友幫幫忙。。感激不盡! 已被LWIP虐得不要不要的了,還有FREE R
    發(fā)表于 07-25 02:23

    Free RTOS移植問(wèn)題的解決辦法?

    按照原子哥的free rtos抑制說(shuō)明文檔做第一個(gè)工程,改完代碼編譯出現(xiàn)xPortSysTickHandler()函數(shù)未定義,頭文件中加入task.h和FreeRTOS.h。實(shí)在找不出問(wèn)題所在,,,there must be anther problem???
    發(fā)表于 06-11 07:57

    Small RTOS 的設(shè)計(jì)思想及消息隊(duì)列通信機(jī)制的應(yīng)用

    將Small RTOS 多任務(wù)的思想應(yīng)用在一個(gè)具體的單片機(jī)控制的電子式存包柜系統(tǒng)的軟件設(shè)計(jì)中,介紹了基于消息隊(duì)列的任務(wù)通訊的編程方法。
    發(fā)表于 04-23 06:22

    了解一下RTOS消息隊(duì)列的應(yīng)用

    基于RTOS的應(yīng)用中,通常使用隊(duì)列機(jī)制實(shí)現(xiàn)任務(wù)間的數(shù)據(jù)交互,一個(gè)應(yīng)用程序可以有任意數(shù)量的消息隊(duì)列,每個(gè)消息隊(duì)列都有自己的用途。 什么是消息隊(duì)列
    發(fā)表于 06-12 14:17

    實(shí)時(shí)操作系統(tǒng)Free RTOS的詳細(xì)介紹

    實(shí)時(shí)操作系統(tǒng)Free RTOS 簡(jiǎn)介 FreeRTOS是一個(gè)迷你的實(shí)時(shí)操作系統(tǒng)內(nèi)核。作為一個(gè)輕量級(jí)的操作系統(tǒng),功能包括:任務(wù)管理、時(shí)間管理、信號(hào)量、消息隊(duì)列、內(nèi)存管理、記錄功能、軟件定時(shí)器、協(xié)程等
    發(fā)表于 06-21 14:30 ?6127次閱讀

    RTOS消息隊(duì)列的多種用途

      消息隊(duì)列可以以多種不同的方式使用。事實(shí)上,您可以編寫可能只使用消息隊(duì)列的相當(dāng)復(fù)雜的應(yīng)用程序。僅使用消息隊(duì)列可以減少代碼的大?。凑加每臻g),因?yàn)榭梢阅M許多其他服務(wù)(信號(hào)量、時(shí)間延遲和事件標(biāo)志)。
    的頭像 發(fā)表于 06-29 14:57 ?2473次閱讀
    <b class='flag-5'>RTOS</b>消息<b class='flag-5'>隊(duì)列</b>的多種用途

    RTOS消息隊(duì)列的應(yīng)用

    基于RTOS的應(yīng)用中,通常使用隊(duì)列機(jī)制實(shí)現(xiàn)任務(wù)間的數(shù)據(jù)交互,一個(gè)應(yīng)用程序可以有任意數(shù)量的消息隊(duì)列,每個(gè)消息隊(duì)列都有自己的用途。
    發(fā)表于 05-29 10:49 ?602次閱讀
    <b class='flag-5'>RTOS</b>消息<b class='flag-5'>隊(duì)列</b>的應(yīng)用