信號量
FreeRTOS中的信號量是一種任務(wù)間通信的方式,信號量包括:二值信號量、互斥信號量、計數(shù)信號量,本次實驗只使用二值信號量。信號量用于任務(wù)間的同步,F(xiàn)reeRTOS是多任務(wù)系統(tǒng),不同任務(wù)間可能需要某種同步關(guān)系
二值信號量
可以通俗理解為0或1標(biāo)志位,比如串口中斷接收完數(shù)據(jù)是一種狀態(tài),此時就需要進(jìn)行串口數(shù)據(jù)處理又是一種狀態(tài),這時使用二值信號量就能很好達(dá)到任務(wù)同步效果
信號量的基本操作有獲取信號量和釋放信號量,例如:數(shù)據(jù)分析處理任務(wù)需要處理串口數(shù)據(jù)時,先嘗試獲取信號量,若獲取不到,也就是信號量是0,則先進(jìn)入阻塞等待,等待超時可先跳出,之后繼續(xù)嘗試獲取信號量。串口空閑中斷接受完一串?dāng)?shù)據(jù)后,可執(zhí)行釋放信號量操作,這時,數(shù)據(jù)分析處理任務(wù)就可以獲取到信號量,進(jìn)而可以處理串口數(shù)據(jù)了,實現(xiàn)了串口數(shù)據(jù)接收與數(shù)據(jù)處理的同步。
API函數(shù)
創(chuàng)建二值信號量
SemaphoreHandle_t xSemaphoreCreateBinary( void )
返回值:
NULL:創(chuàng)建信號量失敗,因為FreeRTOS堆棧不足。
其它值:信號量創(chuàng)建成功。這個返回值存儲著信號量句柄。
非中斷釋放二值信號量
BaseType_t xSemaphoreGive( SemaphoreHandle_t xSemaphore )
xSemaphore:要釋放的信號量句柄
返回值:
釋放成功返回pdPASS,失敗返回errQUEUE_FULL
中斷釋放二值信號量
BaseType_t xSemaphoreGiveFromISR( SemaphoreHandle_t xSemaphore,
BaseType_t* pxHigherPriorityTaskWoken)
參數(shù):
xSemaphore:要釋放的信號量句柄
pxHigherPriorityTaskWoken:標(biāo)記退出此函數(shù)后是否需要進(jìn)行任務(wù)切換
返回值:
釋放成功返回pdPASS,失敗返回errQUEUE_FULL
獲取信號量
BaseType_t xSemaphoreTake( SemaphoreHandle_t xSemaphore,
TickType_t xBlockTime)
參數(shù):
xSemaphore:要釋放的信號量句柄
xBlockTime:阻塞時間
返回值:
獲取成功返回pdTRUE,失敗返回pdFALSE
中斷獲取信號量
BaseType_t xSemaphoreTakeFromISR( SemaphoreHandle_t xSemaphore,
BaseType_t* pxHigherPriorityTaskWoken)
參數(shù):
xSemaphore:要釋放的信號量句柄
pxHigherPriorityTaskWoken:標(biāo)記退出此函數(shù)后是否需要進(jìn)行任務(wù)切換
返回值:
獲取成功返回pdTRUE,失敗返回pdFALSE
實現(xiàn)目的
通過按鍵觸發(fā)二值信號量的釋放,獲取任務(wù)一直在等待信號量的到來,再執(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端口時鐘
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的時鐘
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; //無奇偶校驗
USART_InitStruct.USART_BaudRate=bound; //波特率
USART_InitStruct.USART_StopBits=USART_StopBits_1; //停止位1位
USART_InitStruct.USART_WordLength=USART_WordLength_8b; //字長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)先級
#define START_STK_SIZE 128 //任務(wù)堆棧大小
TaskHandle_t StartTask_Handler; //任務(wù)句柄
void Start_Task(void *pvParameters); //任務(wù)函數(shù)
//釋放信號量
#define Release_TASK_PRIO 2 //任務(wù)優(yōu)先級
#define Release_STK_SIZE 50 //任務(wù)堆棧大小
TaskHandle_t ReleaseTask_Handler; //任務(wù)句柄
void Release_Task(void *p_arg); //任務(wù)函數(shù)
//獲取信號量
#define Gain_TASK_PRIO 3 //任務(wù)優(yōu)先級
#define Gain_STK_SIZE 50 //任務(wù)堆棧大小
TaskHandle_t GainTask_Handler; //任務(wù)句柄
void Gain_Task(void *p_arg); //任務(wù)函數(shù)
SemaphoreHandle_t KeySemaphore;//串口接收二值信號量句柄
int main( void )
{
NVIC_PriorityGroupConfig(NVIC_PriorityGroup_4);//設(shè)置系統(tǒng)中斷優(yōu)先級分組 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)先級
(TaskHandle_t* )&StartTask_Handler //任務(wù)句柄
);
vTaskStartScheduler(); //開啟調(diào)度
}
void Start_Task(void *pvParameters)
{
taskENTER_CRITICAL(); //進(jìn)入臨界區(qū)
//創(chuàng)建二值信號量
KeySemaphore = xSemaphoreCreateBinary();
//創(chuàng)建 釋放信號量 任務(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)建 獲取信號量 任務(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ū)
}
//釋放信號量 任務(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);
}
}
//獲取信號量 任務(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文章
484瀏覽量
62412 -
任務(wù)系統(tǒng)
+關(guān)注
關(guān)注
0文章
4瀏覽量
6135 -
信號量
+關(guān)注
關(guān)注
0文章
53瀏覽量
8376
發(fā)布評論請先 登錄
相關(guān)推薦
轉(zhuǎn):第25章 FreeRTOS任務(wù)二值信號量
FreeRTOS二值信號量卡住出不來怎么辦?
FreeRTOS信號量介紹
怎樣去使用FreeRTOS二值信號量呢
大部分國產(chǎn)低端MCU沒有空閑中斷怎么辦?
FreeRTOS信號量 & ESP32實戰(zhàn)
![<b class='flag-5'>FreeRTOS</b><b class='flag-5'>信號量</b> & ESP32實戰(zhàn)](https://file.elecfans.com/web1/M00/D9/4E/pIYBAF_1ac2Ac0EEAABDkS1IP1s689.png)
FreeRTOS 隊列 信號量 互斥量
![<b class='flag-5'>FreeRTOS</b> 隊列 <b class='flag-5'>信號量</b> 互斥<b class='flag-5'>量</b>](https://file.elecfans.com/web1/M00/D9/4E/pIYBAF_1ac2Ac0EEAABDkS1IP1s689.png)
STM32WB55XX freertos 二值信號量+dma+idle 不定長串口接收 + dma傳輸完成中斷
![STM32WB55XX <b class='flag-5'>freertos</b> <b class='flag-5'>二</b><b class='flag-5'>值</b><b class='flag-5'>信號量</b>+dma+idle 不定長串口接收 + dma傳輸完成中斷](https://file.elecfans.com/web1/M00/D9/4E/pIYBAF_1ac2Ac0EEAABDkS1IP1s689.png)
FreeRTOS高級篇6---FreeRTOS信號量分析
![<b class='flag-5'>FreeRTOS</b>高級篇6---<b class='flag-5'>FreeRTOS</b><b class='flag-5'>信號量</b>分析](https://file.elecfans.com/web1/M00/D9/4E/pIYBAF_1ac2Ac0EEAABDkS1IP1s689.png)
FreeRTOS系列第20篇---FreeRTOS信號量API函數(shù)
![<b class='flag-5'>FreeRTOS</b>系列第20篇---<b class='flag-5'>FreeRTOS</b><b class='flag-5'>信號量</b>API函數(shù)](https://file.elecfans.com/web1/M00/D9/4E/pIYBAF_1ac2Ac0EEAABDkS1IP1s689.png)
評論