FreeRTOS中的信號(hào)量是一種任務(wù)間通信的方式,信號(hào)量包括:二值信號(hào)量、互斥信號(hào)量、計(jì)數(shù)信號(hào)量,本次實(shí)驗(yàn)只使用二值信號(hào)量。信號(hào)量用于任務(wù)間的同步,F(xiàn)reeRTOS是多任務(wù)系統(tǒng),不同任務(wù)間可能需要某種同步關(guān)系
二值信號(hào)量
可以通俗理解為0或1標(biāo)志位,比如串口中斷接收完數(shù)據(jù)是一種狀態(tài),此時(shí)就需要進(jìn)行串口數(shù)據(jù)處理又是一種狀態(tài),這時(shí)使用二值信號(hào)量就能很好達(dá)到任務(wù)同步效果
信號(hào)量的基本操作有獲取信號(hào)量和釋放信號(hào)量,例如:數(shù)據(jù)分析處理任務(wù)需要處理串口數(shù)據(jù)時(shí),先嘗試獲取信號(hào)量,若獲取不到,也就是信號(hào)量是0,則先進(jìn)入阻塞等待,等待超時(shí)可先跳出,之后繼續(xù)嘗試獲取信號(hào)量。串口空閑中斷接受完一串?dāng)?shù)據(jù)后,可執(zhí)行釋放信號(hào)量操作,這時(shí),數(shù)據(jù)分析處理任務(wù)就可以獲取到信號(hào)量,進(jìn)而可以處理串口數(shù)據(jù)了,實(shí)現(xiàn)了串口數(shù)據(jù)接收與數(shù)據(jù)處理的同步。
API函數(shù)
創(chuàng)建二值信號(hào)量
SemaphoreHandle_t xSemaphoreCreateBinary( void )
返回值:
NULL:創(chuàng)建信號(hào)量失敗,因?yàn)镕reeRTOS堆棧不足。
其它值:信號(hào)量創(chuàng)建成功。這個(gè)返回值存儲(chǔ)著信號(hào)量句柄。
非中斷釋放二值信號(hào)量
BaseType_t xSemaphoreGive( SemaphoreHandle_t xSemaphore )
參數(shù):
xSemaphore:要釋放的信號(hào)量句柄
返回值:
釋放成功返回pdPASS,失敗返回errQUEUE_FULL
中斷釋放二值信號(hào)量
BaseType_t xSemaphoreGiveFromISR( SemaphoreHandle_t xSemaphore,
BaseType_t* pxHigherPriorityTaskWoken)
參數(shù):
xSemaphore:要釋放的信號(hào)量句柄
pxHigherPriorityTaskWoken:標(biāo)記退出此函數(shù)后是否需要進(jìn)行任務(wù)切換
返回值:
釋放成功返回pdPASS,失敗返回errQUEUE_FULL
獲取信號(hào)量
BaseType_t xSemaphoreTake( SemaphoreHandle_t xSemaphore,
TickType_t xBlockTime)
參數(shù):
xSemaphore:要釋放的信號(hào)量句柄
xBlockTime:阻塞時(shí)間
返回值:
獲取成功返回pdTRUE,失敗返回pdFALSE
中斷獲取信號(hào)量
BaseType_t xSemaphoreTakeFromISR( SemaphoreHandle_t xSemaphore,
BaseType_t* pxHigherPriorityTaskWoken)
參數(shù):
xSemaphore:要釋放的信號(hào)量句柄
pxHigherPriorityTaskWoken:標(biāo)記退出此函數(shù)后是否需要進(jìn)行任務(wù)切換
返回值:
獲取成功返回pdTRUE,失敗返回pdFALSE
實(shí)現(xiàn)目的
通過按鍵觸發(fā)二值信號(hào)量的釋放,獲取任務(wù)一直在等待信號(hào)量的到來,再執(zhí)行相應(yīng)的任務(wù)
上源碼
#include "stm32f10x.h"
#include "stm32f10x.h"
#include
#include "FreeRTOS.h"
#include "task.h"
#include "semphr.h"
void LED_Init(void)
{
GPIO_InitTypeDef GPIO_InitStructure;
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC, ENABLE); //使能PE端口時(shí)鐘
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_1|GPIO_Pin_5; //端口配置
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; //推挽輸出
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; //IO口速度為50MHz
GPIO_Init(GPIOC, &GPIO_InitStructure); //推挽輸出 ,IO口速度為50MHz
GPIO_SetBits(GPIOC,GPIO_Pin_1|GPIO_Pin_5); //輸出高
}
void KEY_Init(void)
{
GPIO_InitTypeDef GPIO_InitStructure; //定義結(jié)構(gòu)體變量
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,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
}
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; //無奇偶校驗(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; //無硬件數(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;
}
//創(chuàng)建開始任務(wù)
#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ù)
//釋放信號(hào)量
#define Release_TASK_PRIO 2 //任務(wù)優(yōu)先級(jí)
#define Release_STK_SIZE 50 //任務(wù)堆棧大小
TaskHandle_t ReleaseTask_Handler; //任務(wù)句柄
void Release_Task(void *p_arg); //任務(wù)函數(shù)
//獲取信號(hào)量
#define Gain_TASK_PRIO 3 //任務(wù)優(yōu)先級(jí)
#define Gain_STK_SIZE 50 //任務(wù)堆棧大小
TaskHandle_t GainTask_Handler; //任務(wù)句柄
void Gain_Task(void *p_arg); //任務(wù)函數(shù)
SemaphoreHandle_t KeySemaphore;//串口接收二值信號(hào)量句柄
int main( void )
{
NVIC_PriorityGroupConfig(NVIC_PriorityGroup_4);//設(shè)置系統(tǒng)中斷優(yōu)先級(jí)分組 4
LED_Init(); //初始化 LED
KEY_Init(); //按鍵初始化
USART_init(115200); //初始化串口
//創(chuàng)建開始任務(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(); //開啟調(diào)度
}
void Start_Task(void *pvParameters)
{
taskENTER_CRITICAL(); //進(jìn)入臨界區(qū)
//創(chuàng)建二值信號(hào)量
KeySemaphore = xSemaphoreCreateBinary();
//創(chuàng)建 釋放信號(hào)量 任務(wù)
xTaskCreate(
(TaskFunction_t )Release_Task,
(const char* )"Release_Task",
(uint16_t )Release_STK_SIZE,
(void* )NULL,
(UBaseType_t )Release_TASK_PRIO,
(TaskHandle_t* )&ReleaseTask_Handler
);
//創(chuàng)建 獲取信號(hào)量 任務(wù)
xTaskCreate(
(TaskFunction_t )Gain_Task,
(const char* )"Gain_Task",
(uint16_t )Gain_STK_SIZE,
(void* )NULL,
(UBaseType_t )Gain_TASK_PRIO,
(TaskHandle_t* )&GainTask_Handler
);
vTaskDelete(StartTask_Handler); //刪除開始任務(wù)
taskEXIT_CRITICAL(); //退出臨界區(qū)
}
//釋放信號(hào)量 任務(wù)函數(shù)
void Release_Task(void *pvParameters)
{
BaseType_t xReturn = NULL;
while(1)
{
if(GPIO_ReadInputDataBit( GPIOA, GPIO_Pin_0))
{
vTaskDelay(10);
if(GPIO_ReadInputDataBit( GPIOA, GPIO_Pin_0))
{
xReturn = xSemaphoreGive(KeySemaphore);
if(xReturn == pdPASS)
printf("釋放成功\\n");
}
}
vTaskDelay(10);
}
}
//獲取信號(hào)量 任務(wù)函數(shù)
void Gain_Task(void *pvParameters)
{
BaseType_t xReturn = NULL;
while(1)
{
xReturn = xSemaphoreTake(KeySemaphore,portMAX_DELAY);//一直阻塞獲取
if(xReturn == pdPASS)
printf("獲取成功\\n");
vTaskDelay(10);
}
}
-
FreeRTOS
+關(guān)注
關(guān)注
12文章
492瀏覽量
63861 -
任務(wù)系統(tǒng)
+關(guān)注
關(guān)注
0文章
4瀏覽量
6206 -
信號(hào)量
+關(guān)注
關(guān)注
0文章
53瀏覽量
8517
發(fā)布評(píng)論請(qǐng)先 登錄
FreeRTOS串口中斷接收不定長(zhǎng)的數(shù)據(jù)與二值信號(hào)量的使用

FreeRTOS信號(hào)量使用教程

轉(zhuǎn):第22章 FreeRTOS二值信號(hào)量
轉(zhuǎn):第25章 FreeRTOS任務(wù)二值信號(hào)量
FreeRTOS二值信號(hào)量卡住出不來怎么辦?
FreeRTOS信號(hào)量介紹
怎樣去使用FreeRTOS二值信號(hào)量呢
大部分國(guó)產(chǎn)低端MCU沒有空閑中斷怎么辦?
FreeRTOS信號(hào)量 & ESP32實(shí)戰(zhàn)

FreeRTOS 隊(duì)列 信號(hào)量 互斥量

STM32WB55XX freertos 二值信號(hào)量+dma+idle 不定長(zhǎng)串口接收 + dma傳輸完成中斷

FreeRTOS高級(jí)篇6---FreeRTOS信號(hào)量分析

FreeRTOS系列第20篇---FreeRTOS信號(hào)量API函數(shù)

評(píng)論