一区二区三区三上|欧美在线视频五区|国产午夜无码在线观看视频|亚洲国产裸体网站|无码成年人影视|亚洲AV亚洲AV|成人开心激情五月|欧美性爱内射视频|超碰人人干人人上|一区二区无码三区亚洲人区久久精品

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

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

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

Free RTOS的優(yōu)先級翻轉(zhuǎn)

汽車電子技術(shù) ? 來源:玩轉(zhuǎn)單片機 ? 作者:Julian ? 2023-02-10 15:31 ? 次閱讀
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

優(yōu)先級翻轉(zhuǎn)簡介:就是高優(yōu)先級的任務(wù)運行起來的效果好像成了低優(yōu)先級,而低優(yōu)先級比高優(yōu)先級先運行;

舉個栗子:假如有三個高、中、低優(yōu)先級不同的任務(wù),中優(yōu)先級任務(wù)正常跑,假如高、低優(yōu)先級任務(wù)它們兩個都在等待同一個二值信號量,但是較低優(yōu)先級的那個任務(wù)有點特別,就是它在獲取到信號量后有很長一段很長的延遲(如delay_100s),再釋放信號量,因為較低優(yōu)先級任務(wù)沒有釋放信號量,這就導(dǎo)致高優(yōu)先級的任務(wù)在這段時間是在死等,正常都是高優(yōu)先級的任務(wù)會搶占低優(yōu)先級任務(wù),而這個剛好違背了常理導(dǎo)致出現(xiàn)優(yōu)先級翻轉(zhuǎn);如何解決這個問題呢,請看下一篇推文!

用個小實驗看看具體效果

#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;        //定義結(jié)構(gòu)體變量
  
  RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC,ENABLE);  //開啟時鐘
  
  GPIO_InitStructure.GPIO_Pin=GPIO_Pin_0;            //選擇你要設(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);             //將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的時鐘
  
  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;
}


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

#define Low_TASK_PRIO 2       //任務(wù)優(yōu)先級
#define Low_STK_SIZE 50       //任務(wù)堆棧大小
TaskHandle_t LowTask_Handler;     //任務(wù)句柄
void Low_Task(void *p_arg);     //任務(wù)函數(shù)

#define Med_TASK_PRIO 3       //任務(wù)優(yōu)先級
#define Med_STK_SIZE 50       //任務(wù)堆棧大小
TaskHandle_t MedTask_Handler;     //任務(wù)句柄
void Med_Task(void *p_arg);     //任務(wù)函數(shù)

#define High_TASK_PRIO 4       //任務(wù)優(yōu)先級
#define High_STK_SIZE 50       //任務(wù)堆棧大小
TaskHandle_t HighTask_Handler;     //任務(wù)句柄
void High_Task(void *p_arg);     //任務(wù)函數(shù)

SemaphoreHandle_t Binary_Handle =NULL;  //二值信號量句柄

int main( void ) 
{
  NVIC_PriorityGroupConfig(NVIC_PriorityGroup_4);//設(shè)置系統(tǒng)中斷優(yōu)先級分組 4
  
  LED_Init(); //初始化 LED
  KEY_Init();
  USART_init(9600);
  
  //創(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)度
}

//開始任務(wù)函數(shù)
void Start_Task(void *pvParameters)
{
  taskENTER_CRITICAL();   //進(jìn)入臨界區(qū)
   /* 創(chuàng)建Test_Queue */
  Binary_Handle = xSemaphoreCreateBinary();
  if(Binary_Handle != NULL)
  {
    xSemaphoreGive(Binary_Handle);//釋放信號量
  }
  //創(chuàng)建 Low 任務(wù)
  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
  );
  //創(chuàng)建 Med 任務(wù)
  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
  );
  //創(chuàng)建 High 任務(wù)
  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); //刪除開始任務(wù)
  taskEXIT_CRITICAL();   //退出臨界區(qū)
}


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);
  }
}

實驗效果

pYYBAGPl8pqASUmmAAD4JaNeSLc099.png

--END--


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

    關(guān)注

    12

    文章

    492

    瀏覽量

    64021
  • 任務(wù)
    +關(guān)注

    關(guān)注

    1

    文章

    20

    瀏覽量

    8637
收藏 人收藏
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

    評論

    相關(guān)推薦
    熱點推薦

    RTOS應(yīng)用中的優(yōu)先級反轉(zhuǎn)問題

    在嵌入式系統(tǒng)中,如果使用基于優(yōu)先級調(diào)度算法的RTOS,系統(tǒng)中可能發(fā)生優(yōu)先級反轉(zhuǎn)現(xiàn)象。優(yōu)先級反轉(zhuǎn)用來描述系統(tǒng)中高優(yōu)先級任務(wù)由于等待低
    發(fā)表于 12-14 11:00 ?1534次閱讀

    CC1310rtos里面,優(yōu)先級定義時,是1的優(yōu)先級高還是2的優(yōu)先級高?

    CC1310rtos里面,優(yōu)先級定義時,是1的優(yōu)先級高還是2的優(yōu)先級高?
    發(fā)表于 06-21 10:12

    UCOS優(yōu)先級翻轉(zhuǎn)知多少?

    對于新手來說,優(yōu)先級翻轉(zhuǎn)看起來有點點困難?其實,你可能缺少一個有趣的故事故事主人公:LPT(Low PrioTask),MPT(Middle PrioTask),HPT(High PrioTask
    發(fā)表于 04-24 03:25

    干貨 | RTOS應(yīng)用中的優(yōu)先級反轉(zhuǎn)問題

    在嵌入式系統(tǒng)中,如果使用基于優(yōu)先級調(diào)度算法的RTOS,系統(tǒng)中可能發(fā)生優(yōu)先級反轉(zhuǎn)現(xiàn)象。優(yōu)先級反轉(zhuǎn)用來描述系統(tǒng)中高優(yōu)先級任務(wù)由于等待低
    發(fā)表于 03-09 15:00

    如何去解決uC/OS-II中優(yōu)先級翻轉(zhuǎn)問題?

    uC/OS-II的運行機制是什么?uC/OS-II中的優(yōu)先級翻轉(zhuǎn)問題有哪些?如何去解決uC/OS-II中優(yōu)先級翻轉(zhuǎn)問題?
    發(fā)表于 04-25 07:07

    如何正確設(shè)置中斷優(yōu)先級

    優(yōu)先級范圍從0x00~0xFF), 是絕大多數(shù)微控制器制造商只是使用其中的一部分優(yōu)先級NXP 1062 使用了其中的高4bits,所以中斷優(yōu)先級在0-15,共16個在Cortex-M內(nèi)核中,一個中斷的
    發(fā)表于 12-16 07:08

    uC/OS-II中優(yōu)先級翻轉(zhuǎn)問題

    本文著重分析優(yōu)先級翻轉(zhuǎn)問題的產(chǎn)生和影響,以及在uC/OS-II中的解決方案,在嵌入式系統(tǒng)的應(yīng)用中,實時性是一個重要的指標(biāo),而優(yōu)先級翻轉(zhuǎn)是影響系統(tǒng)實時性的重要問題
    發(fā)表于 01-06 16:53 ?2547次閱讀

    開發(fā)過程中是什么原因?qū)е?b class='flag-5'>優(yōu)先級翻轉(zhuǎn)的?

    最近在開發(fā)過程中,遇到一個問題線程優(yōu)先級翻轉(zhuǎn)的問題。那什么原因?qū)е?b class='flag-5'>優(yōu)先級翻轉(zhuǎn)呢? 在RTOS開發(fā)中,優(yōu)先
    的頭像 發(fā)表于 05-28 15:28 ?2558次閱讀
    開發(fā)過程中是什么原因?qū)е?b class='flag-5'>優(yōu)先級</b><b class='flag-5'>翻轉(zhuǎn)</b>的?

    STM32F103芯片中斷優(yōu)先級以及FreeRTOS優(yōu)先級設(shè)置

    STM32F103只用了4個位來表達(dá)優(yōu)先級,因此最多支持16的可編程優(yōu)先級(0~15),15為最低優(yōu)先級。
    發(fā)表于 01-25 18:59 ?1次下載
    STM32F103芯片中斷<b class='flag-5'>優(yōu)先級</b>以及FreeRTOS<b class='flag-5'>優(yōu)先級</b>設(shè)置

    中斷優(yōu)先級處理的原則及配置 搶占優(yōu)先級和響應(yīng)優(yōu)先級的區(qū)別

    首先我們需要知道什么是中斷優(yōu)先級:中斷優(yōu)先級是CPU響應(yīng)中斷的先后順序
    的頭像 發(fā)表于 05-18 15:10 ?3w次閱讀
    中斷<b class='flag-5'>優(yōu)先級</b>處理的原則及配置 搶占<b class='flag-5'>優(yōu)先級</b>和響應(yīng)<b class='flag-5'>優(yōu)先級</b>的區(qū)別

    具有固定優(yōu)先級調(diào)度程序RTOS優(yōu)先級倒置

      優(yōu)先級繼承并不能真正治愈優(yōu)先級倒置,它只是在某些情況下將其影響最小化。硬實時應(yīng)用程序仍應(yīng)仔細(xì)設(shè)計,以便一開始就不會發(fā)生優(yōu)先級反轉(zhuǎn)。
    的頭像 發(fā)表于 06-22 15:45 ?1564次閱讀
    具有固定<b class='flag-5'>優(yōu)先級</b>調(diào)度程序<b class='flag-5'>RTOS</b>的<b class='flag-5'>優(yōu)先級</b>倒置

    基于RTOS內(nèi)核的滴答處理程序優(yōu)先級設(shè)置

      RTOS 內(nèi)核用戶在配置滴答優(yōu)先級和頻率方面可能具有的靈活性肯定會給剛接觸內(nèi)核的開發(fā)人員帶來一些困難。
    的頭像 發(fā)表于 06-29 09:12 ?1573次閱讀
    基于<b class='flag-5'>RTOS</b>內(nèi)核的滴答處理程序<b class='flag-5'>優(yōu)先級</b>設(shè)置

    FreeRTOS任務(wù)的優(yōu)先級示例

    任務(wù)的優(yōu)先級:0~24之間。數(shù)字越大,任務(wù)優(yōu)先等級越高。高優(yōu)先級的任務(wù)優(yōu)先執(zhí)行。
    的頭像 發(fā)表于 09-15 11:13 ?3269次閱讀

    什么是優(yōu)先級反轉(zhuǎn)

    假設(shè)現(xiàn)在有三個任務(wù)TaskA(優(yōu)先級高)、TaskB(優(yōu)先級中)、TaskC(優(yōu)先級低),一個信號量(Semaphore),此信號量用于任務(wù)之間爭奪某個資源。在某一時刻,高優(yōu)先級的Ta
    的頭像 發(fā)表于 04-24 13:01 ?2758次閱讀
    什么是<b class='flag-5'>優(yōu)先級</b>反轉(zhuǎn)

    I2C子系統(tǒng)優(yōu)先級翻轉(zhuǎn)優(yōu)先級繼承

    優(yōu)先級翻轉(zhuǎn)優(yōu)先級繼承 優(yōu)先級翻轉(zhuǎn)在可剝奪內(nèi)核中是非常常見的,例子如下(H:High、M:Middle、L:Low) 任務(wù) H 和任務(wù) M
    的頭像 發(fā)表于 07-22 15:08 ?1148次閱讀
    I2C子系統(tǒng)<b class='flag-5'>優(yōu)先級</b><b class='flag-5'>翻轉(zhuǎn)</b>與<b class='flag-5'>優(yōu)先級</b>繼承