優先級翻轉簡介:就是高優先級的任務運行起來的效果好像成了低優先級,而低優先級比高優先級先運行;
舉個栗子:假如有三個高、中、低優先級不同的任務,中優先級任務正常跑,假如高、低優先級任務它們兩個都在等待同一個二值信號量,但是較低優先級的那個任務有點特別,就是它在獲取到信號量后有很長一段很長的延遲(如delay_100s),再釋放信號量,因為較低優先級任務沒有釋放信號量,這就導致高優先級的任務在這段時間是在死等,正常都是高優先級的任務會搶占低優先級任務,而這個剛好違背了常理導致出現優先級翻轉;如何解決這個問題呢,請看下一篇推文!
用個小實驗看看具體效果
#include "stm32f10x.h"
#include
#include "FreeRTOS.h"
#include "task.h"
#include "semphr.h"
//毫秒級的延時
void Delay_Ms(u16 time)
{
u16 i=0;
while(time--)
{
i=12000; //自己定義
while(i--) ;
}
}
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 5 //任務優先級
#define START_STK_SIZE 128 //任務堆棧大小
TaskHandle_t StartTask_Handler; //任務句柄
void Start_Task(void *pvParameters);//任務函數
#define Low_TASK_PRIO 2 //任務優先級
#define Low_STK_SIZE 50 //任務堆棧大小
TaskHandle_t LowTask_Handler; //任務句柄
void Low_Task(void *p_arg); //任務函數
#define Med_TASK_PRIO 3 //任務優先級
#define Med_STK_SIZE 50 //任務堆棧大小
TaskHandle_t MedTask_Handler; //任務句柄
void Med_Task(void *p_arg); //任務函數
#define High_TASK_PRIO 4 //任務優先級
#define High_STK_SIZE 50 //任務堆棧大小
TaskHandle_t HighTask_Handler; //任務句柄
void High_Task(void *p_arg); //任務函數
SemaphoreHandle_t Binary_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 */
Binary_Handle = xSemaphoreCreateBinary();
if(Binary_Handle != NULL)
{
xSemaphoreGive(Binary_Handle);//釋放信號量
}
//創建 Low 任務
xTaskCreate(
(TaskFunction_t )Low_Task,
(const char* )"Low_Task",
(uint16_t )Low_STK_SIZE,
(void* )NULL,
(UBaseType_t )Low_TASK_PRIO,
(TaskHandle_t* )&LowTask_Handler
);
//創建 Med 任務
xTaskCreate(
(TaskFunction_t )Med_Task,
(const char* )"Med_Task",
(uint16_t )Med_STK_SIZE,
(void* )NULL,
(UBaseType_t )Med_TASK_PRIO,
(TaskHandle_t* )&MedTask_Handler
);
//創建 High 任務
xTaskCreate(
(TaskFunction_t )High_Task,
(const char* )"High_Task",
(uint16_t )High_STK_SIZE,
(void* )NULL,
(UBaseType_t )High_TASK_PRIO,
(TaskHandle_t* )&HighTask_Handler
);
vTaskDelete(StartTask_Handler); //刪除開始任務
taskEXIT_CRITICAL(); //退出臨界區
}
void Low_Task(void *pvParameters)
{
int count = 0;
while(1)
{
printf("Low正在等待n");
xSemaphoreTake(Binary_Handle,portMAX_DELAY);
printf("Low獲取成功%d次n",++count);
Delay_Ms(5000);
xSemaphoreGive(Binary_Handle);//釋放信號量
vTaskDelay(50);
}
}
void Med_Task(void *pvParameters)
{
//BaseType_t xReturn = NULL;
while(1)
{
printf("正在運行n");
vTaskDelay(1000);
}
}
void High_Task(void *pvParameters)
{
int count = 0;
while(1)
{
printf("High正在等待n");
xSemaphoreTake(Binary_Handle,portMAX_DELAY);
printf("High獲取成功%d次n",++count);
xSemaphoreGive(Binary_Handle);//釋放信號量
vTaskDelay(50);
}
}
實驗效果
--END--
聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。
舉報投訴
-
FreeRTOS
+關注
關注
12文章
484瀏覽量
62394 -
任務
+關注
關注
1文章
20瀏覽量
8560
發布評論請先 登錄
相關推薦
RTOS應用中的優先級反轉問題
在嵌入式系統中,如果使用基于優先級調度算法的RTOS,系統中可能發生優先級反轉現象。優先級反轉用來描述系統中高優先級任務由于等待低
發表于 12-14 11:00
?1263次閱讀
UCOS優先級翻轉知多少?
對于新手來說,優先級翻轉看起來有點點困難?其實,你可能缺少一個有趣的故事故事主人公:LPT(Low PrioTask),MPT(Middle PrioTask),HPT(High PrioTask
發表于 04-24 03:25
干貨 | RTOS應用中的優先級反轉問題
在嵌入式系統中,如果使用基于優先級調度算法的RTOS,系統中可能發生優先級反轉現象。優先級反轉用來描述系統中高優先級任務由于等待低
發表于 03-09 15:00
如何正確設置中斷優先級
優先級范圍從0x00~0xFF), 是絕大多數微控制器制造商只是使用其中的一部分優先級NXP 1062 使用了其中的高4bits,所以中斷優先級在0-15,共16個在Cortex-M內核中,一個中斷的
發表于 12-16 07:08
uC/OS-II中優先級翻轉問題
本文著重分析優先級翻轉問題的產生和影響,以及在uC/OS-II中的解決方案,在嵌入式系統的應用中,實時性是一個重要的指標,而優先級翻轉是影響系統實時性的重要問題
發表于 01-06 16:53
?2415次閱讀
2.FreeRTOS中斷優先級和任務優先級
FreeRTOS中斷優先級和任務優先級架構:Cortex-M3版本:FreeRTOS V9.0.0前言:最開始,我并沒有搞清楚什么是中斷優先級和任務優先級,但看了部分資料后發現這兩個并
發表于 12-04 20:21
?9次下載
STM32F103芯片中斷優先級以及FreeRTOS優先級設置
STM32F103只用了4個位來表達優先級,因此最多支持16級的可編程優先級(0~15),15為最低優先級。
發表于 01-25 18:59
?1次下載
什么是優先級反轉
假設現在有三個任務TaskA(優先級高)、TaskB(優先級中)、TaskC(優先級低),一個信號量(Semaphore),此信號量用于任務之間爭奪某個資源。在某一時刻,高優先級的Ta
評論