網(wǎng)站首頁(yè) 編程語(yǔ)言 正文
FreeRTOS實(shí)時(shí)操作系統(tǒng)的多優(yōu)先級(jí)實(shí)現(xiàn)_操作系統(tǒng)
作者:jiang_2018 ? 更新時(shí)間: 2022-06-09 編程語(yǔ)言如何實(shí)現(xiàn)任務(wù)多優(yōu)先級(jí)
FreeRTOS中,數(shù)字優(yōu)先級(jí)越小,邏輯優(yōu)先級(jí)也越小,空閑任務(wù)優(yōu)先級(jí)為0.
List_t pxReadyTasksLists[configMAX_PRIORITIES]是數(shù)組,數(shù)組下標(biāo)代表任務(wù)優(yōu)先級(jí),任務(wù)創(chuàng)建是根據(jù)設(shè)置的任務(wù)優(yōu)先級(jí)插入到對(duì)應(yīng)下標(biāo)的列表根節(jié)點(diǎn)上,如下。
要支持多優(yōu)先級(jí),就是再任務(wù)切換時(shí)讓pxCurrentTCB
指向最高優(yōu)先級(jí)的TCB即可,之前時(shí)手動(dòng)再任務(wù)1、任務(wù)2來回切換,現(xiàn)在問題就是怎么找到優(yōu)先級(jí)最高的就緒任務(wù)TCB。有2套方法,軟件通用方法和硬件指令方法
軟件通用方法和硬件指令方法
通過configUSE_PORT_OPTIMISED_TASK_SELECTION指定使用軟件通用方法還是硬件指令方法,代碼再task.c
中
#if ( configUSE_PORT_OPTIMISED_TASK_SELECTION == 0 )//使用通用方法 /* uxTopReadyPriority 是全局變量,保存著最高優(yōu)先級(jí) */ #define taskRECORD_READY_PRIORITY( uxPriority ) \ { \ if( ( uxPriority ) > uxTopReadyPriority ) \ { \ uxTopReadyPriority = ( uxPriority ); \ } \ } /* taskRECORD_READY_PRIORITY */ /*-----------------------------------------------------------*/ #define taskSELECT_HIGHEST_PRIORITY_TASK() \ { \ /* 從高到底依次尋找非空的列表根節(jié)點(diǎn)下標(biāo) */ \ while( listLIST_IS_EMPTY( &( pxReadyTasksLists[ uxTopReadyPriority ] ) ) ) \ { \ configASSERT( uxTopReadyPriority ); \ --uxTopReadyPriority; \ } \ \ /* 更新pxCurrentTCB 和*/ \ listGET_OWNER_OF_NEXT_ENTRY( pxCurrentTCB, &( pxReadyTasksLists[ uxTopReadyPriority ] ) ); \ } /* taskSELECT_HIGHEST_PRIORITY_TASK */ /*-----------------------------------------------------------*/ /* 對(duì)于軟件方式這里做空*/ #define taskRESET_READY_PRIORITY( uxPriority ) #define portRESET_READY_PRIORITY( uxPriority, uxTopReadyPriority ) #else /* configUSE_PORT_OPTIMISED_TASK_SELECTION */ /* 硬件指令方式 */ /* 根據(jù)uxPriority來更新uxTopReadyPriority,記錄下最高優(yōu)先級(jí)*/ #define taskRECORD_READY_PRIORITY( uxPriority ) portRECORD_READY_PRIORITY( uxPriority, uxTopReadyPriority ) /*-----------------------------------------------------------*/ #define taskSELECT_HIGHEST_PRIORITY_TASK() \ { \ UBaseType_t uxTopPriority; \ \ /* 尋找優(yōu)先級(jí)最高的任務(wù)TCB來更新pxCurrentTCB */ \ portGET_HIGHEST_PRIORITY( uxTopPriority, uxTopReadyPriority ); \ configASSERT( listCURRENT_LIST_LENGTH( &( pxReadyTasksLists[ uxTopPriority ] ) ) > 0 ); \ listGET_OWNER_OF_NEXT_ENTRY( pxCurrentTCB, &( pxReadyTasksLists[ uxTopPriority ] ) ); \ } /* taskSELECT_HIGHEST_PRIORITY_TASK() */ /*-----------------------------------------------------------*/ /* 清除uxTopReadyPriority的uxPriority 位. */ #define taskRESET_READY_PRIORITY( uxPriority ) \ { \ if( listCURRENT_LIST_LENGTH( &( pxReadyTasksLists[ ( uxPriority ) ] ) ) == ( UBaseType_t ) 0 ) \ { \ portRESET_READY_PRIORITY( ( uxPriority ), ( uxTopReadyPriority ) ); \ } \ } #endif /* configUSE_PORT_OPTIMISED_TASK_SELECTION */
下面看著幾個(gè)port接口
#define portRECORD_READY_PRIORITY( uxPriority, uxReadyPriorities ) \ ( uxReadyPriorities ) |= ( 1UL << ( uxPriority ) ) #define portRESET_READY_PRIORITY( uxPriority, uxReadyPriorities ) \ ( uxReadyPriorities ) &= ~( 1UL << ( uxPriority ) )
可以看到硬件方式是把uxTopReadyPriority 看作一個(gè)位圖,每位代表一個(gè)優(yōu)先級(jí),一共32bit,任務(wù)就緒是就把對(duì)應(yīng)位置1,反之清0.
所以獲得最高就緒優(yōu)先級(jí)的硬件方法如下(利用clz指令,計(jì)算一個(gè)變量從高位開始第一次出現(xiàn)1的位前面0的個(gè)數(shù),上圖clz(uxReadyPriorities)=6)
#define portGET_HIGHEST_PRIORITY( uxTopPriority, uxReadyPriorities)\ uxTopPriority = ( 31 - __clz( ( uxReadyPriorities ) ) )
原文鏈接:https://blog.csdn.net/weixin_41572450/article/details/106036638
相關(guān)推薦
- 2023-02-12 Android?onbackpressed實(shí)現(xiàn)返回鍵的攔截和彈窗流程分析_Android
- 2024-04-05 springboot注冊(cè)攔截器與返回統(tǒng)一標(biāo)準(zhǔn)響應(yīng)格式
- 2022-07-08 python?動(dòng)態(tài)規(guī)劃問題解析(背包問題和最長(zhǎng)公共子串)_python
- 2022-07-02 Python使用struct庫(kù)的用法小結(jié)_python
- 2022-07-17 Go語(yǔ)言入門exec的基本使用示例_Golang
- 2022-05-02 Redis使用命令行與多數(shù)據(jù)庫(kù)配置_Redis
- 2024-03-08 學(xué)習(xí)基于ssm框架前后端分離實(shí)現(xiàn)注冊(cè)登錄MD5加密的心得體會(huì)
- 2022-04-18 python?commands模塊的適用方式_python
- 最近更新
-
- window11 系統(tǒng)安裝 yarn
- 超詳細(xì)win安裝深度學(xué)習(xí)環(huán)境2025年最新版(
- Linux 中運(yùn)行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲(chǔ)小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎(chǔ)操作-- 運(yùn)算符,流程控制 Flo
- 1. Int 和Integer 的區(qū)別,Jav
- spring @retryable不生效的一種
- Spring Security之認(rèn)證信息的處理
- Spring Security之認(rèn)證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權(quán)
- redisson分布式鎖中waittime的設(shè)
- maven:解決release錯(cuò)誤:Artif
- restTemplate使用總結(jié)
- Spring Security之安全異常處理
- MybatisPlus優(yōu)雅實(shí)現(xiàn)加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務(wù)發(fā)現(xiàn)-Nac
- Spring Security之基于HttpR
- Redis 底層數(shù)據(jù)結(jié)構(gòu)-簡(jiǎn)單動(dòng)態(tài)字符串(SD
- arthas操作spring被代理目標(biāo)對(duì)象命令
- Spring中的單例模式應(yīng)用詳解
- 聊聊消息隊(duì)列,發(fā)送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠(yuǎn)程分支