上篇講解了二值信號量,二值信號量只能判斷有無,而不能確定事件發生的次數,因此我們為了確定事件的次數引入了計數型信號量!
使用計數型信號量時,需要在FreeRTOSConfig.h中加入一行配置代碼
//為1時使用計數信號量
#define configUSE_COUNTING_SEMAPHORES 1
仔細閱讀源碼的同學就會發現有很多類似的代碼(如下圖),這些代碼讓FreeRTO可以實現不同需要的裁剪,以減少系統開銷
創建計數型信號量
xSemaphoreCreateCounting( UBaseType_t uxMaxCount, UBaseType_t uxInitialCount )
參數:
uxMaxCount:計數信號量最大計數值,但信號量值等于此值的時候,釋放信號量就會失敗
uxInitialCount :計數信號量初始值
返回值:
NULL:計數信號量創建失敗
其他值:計數信號量創建成功,返回計數信號量句柄
釋放信號量
非中斷釋放
xSemaphoreGive( SemaphoreHandle_t xSemaphore )
參數:
xSemaphore :要釋放的信號量句柄
返回值:
pdPASS: 釋放信號量成功
errQUEU_FULL:釋放信號量失敗
中斷釋放
BaseType_t xSemaphoreGiveFromISR( SemaphoreHandle_t xSemaphore,
BaseType_t* pxHigherPriorityTaskWoken)
參數:
xSemaphore:要釋放的信號量句柄
pxHigherPriorityTaskWoken:標記退出此函數后是否需要進行任務切換,pxHigherPriorityTaskWoken是可選參數,可以設置為NULL。當該值為pdTRUE的時候在退出中斷服務函數之前一定要進行一次任務切換。
返回值:
釋放成功返回pdPASS,失敗返回errQUEUE_FULL
獲取信號量
非中斷獲取
BaseType_t xSemaphoreTake( SemaphoreHandle_t xSemaphore,
TickType_t xBlockTime)
參數:
xSemaphore:要釋放的信號量句柄
xBlockTime:阻塞時間
返回值:
獲取成功返回pdTRUE,失敗返回pdFALSE
中斷獲取
BaseType_t xSemaphoreTakeFromISR( SemaphoreHandle_t xSemaphore,
BaseType_t* pxHigherPriorityTaskWoken)
參數:
xSemaphore:要釋放的信號量句柄
pxHigherPriorityTaskWoken:標記退出此函數后是否需要進行任務切換
返回值:
獲取成功返回pdTRUE,失敗返回pdFALSE
注意:不管是二值信號量、計數型信號量還是互斥信號量,它們都使用上面的釋放信號量和獲取信號量API函數
源碼例程
#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); //開啟時鐘
GPIO_InitStructure.GPIO_Pin=GPIO_Pin_0; //選擇你要設置的IO口
GPIO_InitStructure.GPIO_Mode=GPIO_Mode_Out_PP; //設置推挽輸出模式
GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz; //設置傳輸速率
GPIO_Init(GPIOC,&GPIO_InitStructure); //初始化GPIO
GPIO_SetBits(GPIOC,GPIO_Pin_0); //將LED端口拉高,熄滅LED
}
void KEY_Init(void)
{
GPIO_InitTypeDef GPIO_InitStructure; //定義結構體變量
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA|RCC_APB2Periph_GPIOE,ENABLE);
GPIO_InitStructure.GPIO_Pin=GPIO_Pin_0; //選擇你要設置的IO口
GPIO_InitStructure.GPIO_Mode=GPIO_Mode_IPD;//下拉輸入
GPIO_InitStructure.GPIO_Speed=GPIO_Speed_50MHz; //設置傳輸速率
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結構體變量
USART_InitTypeDef USART_InitStruct; //定義串口結構體變量
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為復用推挽輸出
GPIO_InitStruct.GPIO_Speed=GPIO_Speed_50MHz; //配置PA9速率
GPIO_Init(GPIOA,&GPIO_InitStruct); //GPIO初始化函數
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初始化函數
USART_InitStruct.USART_Mode=USART_Mode_Tx|USART_Mode_Rx; //發送接收模式
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; //無硬件數據流控制
USART_Init(USART1,&USART_InitStruct); //串口初始化函數
USART_Cmd(USART1,ENABLE); //使能USART1
}
int fputc(int ch,FILE *f) //printf重定向函數
{
USART_SendData(USART1,(uint8_t)ch); //發送一字節數據
while(USART_GetFlagStatus(USART1,USART_FLAG_TXE) == RESET); //等待發送完成
return ch;
}
#define START_TASK_PRIO 1 //任務優先級
#define START_STK_SIZE 128 //任務堆棧大小
TaskHandle_t StartTask_Handler; //任務句柄
void Start_Task(void *pvParameters);//任務函數
#define Send_TASK_PRIO 2 //任務優先級
#define Send_STK_SIZE 50 //任務堆棧大小
TaskHandle_t SendTask_Handler; //任務句柄
void Send_Task(void *p_arg); //任務函數
#define Receive_TASK_PRIO 2 //任務優先級
#define Receive_STK_SIZE 50 //任務堆棧大小
TaskHandle_t ReceiveTask_Handler; //任務句柄
void Receive_Task(void *p_arg); //任務函數
SemaphoreHandle_t CountSem_Handle =NULL; //計數型信號量句柄
int main( void )
{
NVIC_PriorityGroupConfig(NVIC_PriorityGroup_4);//設置系統中斷優先級分組 4
LED_Init(); //初始化 LED
KEY_Init();
USART_init(9600);
//創建開始任務
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(); //進入臨界區
/* 創建Test_Queue */
CountSem_Handle = xSemaphoreCreateCounting(5,0);
//創建 發送 任務
xTaskCreate(
(TaskFunction_t )Send_Task,
(const char* )"Send_Task",
(uint16_t )Send_STK_SIZE,
(void* )NULL,
(UBaseType_t )Send_TASK_PRIO,
(TaskHandle_t* )&SendTask_Handler
);
//創建 接收 任務
xTaskCreate(
(TaskFunction_t )Receive_Task,
(const char* )"Receive_Task",
(uint16_t )Receive_STK_SIZE,
(void* )NULL,
(UBaseType_t )Receive_TASK_PRIO,
(TaskHandle_t* )&ReceiveTask_Handler
);
vTaskDelete(StartTask_Handler); //刪除開始任務
taskEXIT_CRITICAL(); //退出臨界區
}
//發送 任務函數
void Send_Task(void *pvParameters)
{
BaseType_t xReturn = NULL;
while(1)
{
if (CountSem_Handle != NULL)
{
xReturn = xSemaphoreGive(CountSem_Handle);
if (xReturn == pdTRUE){
printf("信號量釋放成功n");
}
else{
printf("信號量釋放失敗n");
}
}
vTaskDelay(1000);
}
}
//接收 任務函數
void Receive_Task(void *pvParameters)
{
BaseType_t xReturn = NULL;
uint16_t count = 0;
while(1)
{
// 等待獲取信號量
xReturn = xSemaphoreTake(CountSem_Handle, portMAX_DELAY);
if (xReturn == pdTRUE)
{
count = uxSemaphoreGetCount(CountSem_Handle);
printf("%dn",count);
}
else
{
printf("獲取信號量失敗n");
}
vTaskDelay(2000);
}
}
實驗現象
--END--
-
FreeRTOS
+關注
關注
12文章
483瀏覽量
62001 -
信號量
+關注
關注
0文章
53瀏覽量
8314
發布評論請先 登錄
相關推薦
評論