網(wǎng)站首頁(yè) 編程語(yǔ)言 正文
FreeRTOS提供了兩個(gè)系統(tǒng)延時(shí)函數(shù):相對(duì)延時(shí)函數(shù)vTaskDelay()和絕對(duì)延時(shí)函數(shù)
vTaskDelayUntil()。相對(duì)延時(shí)是指每次延時(shí)都是從任務(wù)執(zhí)行函數(shù)vTaskDelay()開(kāi)始,延時(shí)指定的時(shí)間結(jié)束;
絕對(duì)延時(shí)是指每隔指定的時(shí)間,執(zhí)行一次調(diào)用vTaskDelayUntil()函數(shù)的任務(wù)。換句話說(shuō):任務(wù)以固定的頻率執(zhí)行。
在《FreeRTOS任務(wù)控制》一文中,已經(jīng)介紹了這兩個(gè)API函數(shù)的原型和用法,本文將分析這兩個(gè)函數(shù)的實(shí)現(xiàn)原理。
1. 相對(duì)延時(shí)函數(shù)vTaskDelay()
考慮下面的任務(wù),任務(wù)A在執(zhí)行任務(wù)主體代碼后,調(diào)用相對(duì)延時(shí)函數(shù)vTaskDelay()進(jìn)入阻塞狀態(tài)。系統(tǒng)中除了任務(wù)A外,還有其它任務(wù),但是任務(wù)A的優(yōu)先級(jí)最高。
void vTaskA( void * pvParameters ) { /* 阻塞500ms. 注:宏pdMS_TO_TICKS用于將毫秒轉(zhuǎn)成節(jié)拍數(shù),FreeRTOS V8.1.0及 以上版本才有這個(gè)宏,如果使用低版本,可以使用 500 / portTICK_RATE_MS */ const portTickType xDelay = pdMS_TO_TICKS(500); for( ;; ) { // ... // 這里為任務(wù)主體代碼 // ... /* 調(diào)用系統(tǒng)延時(shí)函數(shù),阻塞500ms */ vTaskDelay( xDelay ); } }
對(duì)于這樣一個(gè)任務(wù),執(zhí)行過(guò)程如圖1-1所示。當(dāng)任務(wù)A獲取CPU使用權(quán)后,先執(zhí)行任務(wù)A的主體代碼,之后調(diào)用系統(tǒng)延時(shí)函數(shù)vTaskDelay()進(jìn)入阻塞狀態(tài)。任務(wù)A進(jìn)入阻塞后,其它任務(wù)得以執(zhí)行。
FreeRTOS內(nèi)核會(huì)周期性的檢查任務(wù)A的阻塞是否達(dá)到,如果阻塞時(shí)間達(dá)到,則將任務(wù)A設(shè)置為就緒狀態(tài)。由于任務(wù)A的優(yōu)先級(jí)最高,會(huì)搶占CPU,再次執(zhí)行任務(wù)主體代碼,不斷循環(huán)。
從圖1-1可以看出,任務(wù)A每次延時(shí)都是從調(diào)用延時(shí)函數(shù)vTaskDelay()開(kāi)始算起的,延時(shí)是相對(duì)于這一時(shí)刻開(kāi)始的,所以叫做相對(duì)延時(shí)函數(shù)。
從圖1-1還可以看出,如果執(zhí)行任務(wù)A的過(guò)程中發(fā)生中斷,那么任務(wù)A執(zhí)行的周期就會(huì)變長(zhǎng),所以使用相對(duì)延時(shí)函數(shù)vTaskDelay(),不能周期性的執(zhí)行任務(wù)A。
圖1-1:相對(duì)延時(shí)函數(shù)執(zhí)行示意圖
我們來(lái)看一下源碼。
void vTaskDelay( const TickType_t xTicksToDelay ) { BaseType_t xAlreadyYielded = pdFALSE; /* 如果延時(shí)時(shí)間為0,則不會(huì)將當(dāng)前任務(wù)加入延時(shí)列表 */ if( xTicksToDelay > ( TickType_t ) 0U ) { vTaskSuspendAll(); { /* 將當(dāng)前任務(wù)從就緒列表中移除,并根據(jù)當(dāng)前系統(tǒng)節(jié)拍計(jì)數(shù)器值計(jì)算喚醒時(shí)間,然后將任務(wù)加入延時(shí)列表 */ prvAddCurrentTaskToDelayedList( xTicksToDelay, pdFALSE ); } xAlreadyYielded = xTaskResumeAll(); } /* 強(qiáng)制執(zhí)行一次上下文切換*/ if( xAlreadyYielded == pdFALSE ) { portYIELD_WITHIN_API(); } }
如果延時(shí)大于0,則會(huì)將當(dāng)前任務(wù)從就緒列表刪除,然后加入到延時(shí)列表。是調(diào)用函數(shù)prvAddCurrentTaskToDelayedList()完成這一過(guò)程的。我們?cè)谇懊嬉幌盗胁┪闹卸啻翁岬剑瑃asks.c中定義了很多局部靜態(tài)變量,其中有一個(gè)變量xTickCount定義如下所示:
static volatile TickType_t xTickCount = ( TickType_t ) 0U;
這個(gè)變量用來(lái)計(jì)數(shù),記錄系統(tǒng)節(jié)拍中斷的次數(shù),它在啟動(dòng)調(diào)度器時(shí)被清零,在每次系統(tǒng)節(jié)拍時(shí)鐘發(fā)生中斷后加1。相對(duì)延時(shí)函數(shù)會(huì)使用到這個(gè)變量,xTickCount表示了當(dāng)前的系統(tǒng)節(jié)拍中斷次數(shù),這個(gè)值加上參數(shù)規(guī)定的延時(shí)時(shí)間(以系統(tǒng)節(jié)拍數(shù)表示)xTicksToDelay,就是下次喚醒任務(wù)的時(shí)間,xTickCount+ xTicksToDelay會(huì)被記錄到任務(wù)TCB中,隨著任務(wù)一起被掛接到延時(shí)列表。
我們知道變量xTickCount是TickType_t類型的,它也會(huì)溢出。在32位架構(gòu)中,當(dāng)xTicksToDelay達(dá)到4294967295后再增加,就會(huì)溢出變成0。為了解決xTickCount溢出問(wèn)題,F(xiàn)reeRTOS使用了兩個(gè)延時(shí)列表:xDelayedTaskList1和xDelayedTaskList2,并使用兩個(gè)列表指針類型變量pxDelayedTaskList和pxOverflowDelayedTaskList分別指向上面的延時(shí)列表1和延時(shí)列表2(在創(chuàng)建任務(wù)時(shí)將延時(shí)列表指針指向延時(shí)列表)。順便說(shuō)一下,上面的兩個(gè)延時(shí)列表指針變量和兩個(gè)延時(shí)列表變量都是在tasks.c中定義的靜態(tài)局部變量。
如果內(nèi)核判斷出xTickCount+ xTicksToDelay溢出,就將當(dāng)前任務(wù)掛接到列表指針pxOverflowDelayedTaskList指向的列表中,否則就掛接到列表指針pxDelayedTaskList指向的列表中。
每次系統(tǒng)節(jié)拍時(shí)鐘中斷,中斷服務(wù)函數(shù)都會(huì)檢查這兩個(gè)延時(shí)列表,查看延時(shí)的任務(wù)是否到期,如果時(shí)間到期,則將任務(wù)從延時(shí)列表中刪除,重新加入就緒列表。如果新加入就緒列表的任務(wù)優(yōu)先級(jí)大于當(dāng)前任務(wù),則會(huì)觸發(fā)一次上下文切換。
2. 絕對(duì)延時(shí)函數(shù)vTaskDelayUntil()
考慮下面的任務(wù),任務(wù)B首先調(diào)用絕對(duì)延時(shí)函數(shù)vTaskDelayUntil ()進(jìn)入阻塞狀態(tài),阻塞時(shí)間到達(dá)后,執(zhí)行任務(wù)主體代碼。系統(tǒng)中除了任務(wù)B外,還有其它任務(wù),但是任務(wù)B的優(yōu)先級(jí)最高。
void vTaskB( void * pvParameters ) { static portTickType xLastWakeTime; const portTickType xFrequency = pdMS_TO_TICKS(500); // 使用當(dāng)前時(shí)間初始化變量xLastWakeTime ,注意這和vTaskDelay()函數(shù)不同 xLastWakeTime = xTaskGetTickCount(); for( ;; ) { /* 調(diào)用系統(tǒng)延時(shí)函數(shù),周期性阻塞500ms */ vTaskDelayUntil( &xLastWakeTime,xFrequency ); // ... // 這里為任務(wù)主體代碼,周期性執(zhí)行.注意這和vTaskDelay()函數(shù)也不同 // ... } }
對(duì)于這樣一個(gè)任務(wù),執(zhí)行過(guò)程如圖2-1所示。當(dāng)任務(wù)B獲取CPU使用權(quán)后,先調(diào)用系統(tǒng)延時(shí)函數(shù)vTaskDelayUntil()使任務(wù)進(jìn)入阻塞狀態(tài)。任務(wù)B進(jìn)入阻塞后,其它任務(wù)得以執(zhí)行。FreeRTOS內(nèi)核會(huì)周期性的檢查任務(wù)A的阻塞是否達(dá)到,如果阻塞時(shí)間達(dá)到,則將任務(wù)A設(shè)置為就緒狀態(tài)。
由于任務(wù)B的優(yōu)先級(jí)最高,會(huì)搶占CPU,接下來(lái)執(zhí)行任務(wù)主體代碼。任務(wù)主體代碼執(zhí)行完畢后,會(huì)繼續(xù)調(diào)用系統(tǒng)延時(shí)函數(shù)vTaskDelayUntil()使任務(wù)進(jìn)入阻塞狀態(tài),周而復(fù)始。
從圖2-1可以看出,從調(diào)用函數(shù)vTaskDelayUntil()開(kāi)始,每隔固定+
-周期,任務(wù)B的主體代碼就會(huì)被執(zhí)行一次,即使任務(wù)B在執(zhí)行過(guò)程中發(fā)生中斷,也不會(huì)影響這個(gè)周期性,只是會(huì)縮短其它任務(wù)的執(zhí)行時(shí)間!所以這個(gè)函數(shù)被稱為絕對(duì)延時(shí)函數(shù),它可以用于周期性的執(zhí)行任務(wù)A的主體代碼。
圖2-1:絕對(duì)延時(shí)函數(shù)執(zhí)行示意圖
函數(shù)vTaskDelayUntil()是如何做到周期性的呢,我們來(lái)看一下源碼。
void vTaskDelayUntil( TickType_t * const pxPreviousWakeTime, const TickType_t xTimeIncrement ) { TickType_t xTimeToWake; BaseType_t xAlreadyYielded, xShouldDelay = pdFALSE; vTaskSuspendAll(); { /* 保存系統(tǒng)節(jié)拍中斷次數(shù)計(jì)數(shù)器 */ const TickType_t xConstTickCount = xTickCount; /* 計(jì)算任務(wù)下次喚醒時(shí)間(以系統(tǒng)節(jié)拍中斷次數(shù)表示) */ xTimeToWake = *pxPreviousWakeTime + xTimeIncrement; /* *pxPreviousWakeTime中保存的是上次喚醒時(shí)間,喚醒后需要一定時(shí)間執(zhí)行任務(wù)主體代碼,如果上次喚醒時(shí)間大于當(dāng)前時(shí)間,說(shuō)明節(jié)拍計(jì)數(shù)器溢出了 */ if( xConstTickCount < *pxPreviousWakeTime ) { /*只有當(dāng)周期性延時(shí)時(shí)間大于任務(wù)主體代碼執(zhí)行時(shí)間,才會(huì)將任務(wù)掛接到延時(shí)列表.*/ if( ( xTimeToWake < *pxPreviousWakeTime ) && ( xTimeToWake > xConstTickCount ) ) { xShouldDelay = pdTRUE; } } else { /* 也都是保證周期性延時(shí)時(shí)間大于任務(wù)主體代碼執(zhí)行時(shí)間 */ if( ( xTimeToWake < *pxPreviousWakeTime ) || ( xTimeToWake > xConstTickCount ) ) { xShouldDelay = pdTRUE; } } /* 更新喚醒時(shí)間,為下一次調(diào)用本函數(shù)做準(zhǔn)備. */ *pxPreviousWakeTime = xTimeToWake; if( xShouldDelay != pdFALSE ) { /* 將本任務(wù)加入延時(shí)列表,注意阻塞時(shí)間并不是以當(dāng)前時(shí)間為參考,因此減去了當(dāng)前系統(tǒng)節(jié)拍中斷計(jì)數(shù)器值*/ prvAddCurrentTaskToDelayedList( xTimeToWake - xConstTickCount, pdFALSE ); } } xAlreadyYielded = xTaskResumeAll(); /* 強(qiáng)制執(zhí)行一次上下文切換 */ if( xAlreadyYielded == pdFALSE ) { portYIELD_WITHIN_API(); } }
與相對(duì)延時(shí)函數(shù)vTaskDelay不同,本函數(shù)增加了一個(gè)參數(shù)pxPreviousWakeTime用于指向一個(gè)變量,變量保存上次任務(wù)解除阻塞的時(shí)間。這個(gè)變量在任務(wù)開(kāi)始時(shí)必須被設(shè)置成當(dāng)前系統(tǒng)節(jié)拍中斷次數(shù)(見(jiàn)上文的任務(wù)B舉例),此后函數(shù)vTaskDelayUntil()在內(nèi)部自動(dòng)更新這個(gè)變量。
由于變量xTickCount可能會(huì)溢出,所以程序必須檢測(cè)各種溢出情況,并且要保證延時(shí)周期不得小于任務(wù)主體代碼執(zhí)行時(shí)間。這很好理解,不可能出現(xiàn)每5毫秒執(zhí)行一個(gè)需要20毫秒才能執(zhí)行完的任務(wù)。
如果我們以橫坐標(biāo)表示變量xTickCount的范圍,則橫坐標(biāo)左端為0,右端為變量xTickCount所能表示的最大值。在如圖2-2所示的三種情況下,才可以將任務(wù)加入延時(shí)列表。
圖2-2中
*pxPreviousWakeTime和xTimeToWake之間表示任務(wù)周期性延時(shí)時(shí)間,
*pxPreviousWakeTime和xConstTickCount之間表示任務(wù)B主體代碼執(zhí)行時(shí)間。
圖2-2中
第一種情況處理系統(tǒng)節(jié)拍中斷計(jì)數(shù)器(xConstTickCount)和喚醒時(shí)間計(jì)數(shù)器(xTimeToWake)溢出情況;
第二種情況處理喚醒時(shí)間計(jì)數(shù)器(xTimeToWake)溢出情況
第三種情況處理常規(guī)無(wú)溢出的情況。
從圖中可以看出,不管是溢出還是無(wú)溢出,都要求在下次喚醒任務(wù)之前,當(dāng)前任務(wù)主體代碼必須被執(zhí)行完。表現(xiàn)在圖2-2中,就是變量xTimeToWake總是大于變量xConstTickCount(每溢出一次的話相當(dāng)于加上一次最大值Max)。
圖2-2:將任務(wù)加入延時(shí)列表的三種情況
計(jì)算的喚醒時(shí)間合法后,就將當(dāng)前任務(wù)加入延時(shí)列表,同樣延時(shí)列表也有兩個(gè)。每次系統(tǒng)節(jié)拍中斷,中斷服務(wù)函數(shù)都會(huì)檢查這兩個(gè)延時(shí)列表,查看延時(shí)的任務(wù)是否到期,如果時(shí)間到期,則將任務(wù)從延時(shí)列表中刪除,重新加入就緒列表。如果新加入就緒列表的任務(wù)優(yōu)先級(jí)大于當(dāng)前任務(wù),則會(huì)觸發(fā)一次上下文切換。
3.小結(jié)
上面的例子中,調(diào)用系統(tǒng)延時(shí)的任務(wù)都是最高優(yōu)先級(jí),這是為了便于分析而特意為之的,實(shí)際上的任務(wù)可不一定能設(shè)置為最高優(yōu)先級(jí)。對(duì)于相對(duì)延時(shí),如果任務(wù)不是最高優(yōu)先級(jí),則任務(wù)執(zhí)行周期更不可測(cè),這個(gè)問(wèn)題不大,我們本來(lái)也不會(huì)使用它作為精確延時(shí);
對(duì)于絕對(duì)延時(shí)函數(shù),如果任務(wù)不是最高優(yōu)先級(jí),則仍然能周期性的將任務(wù)解除阻塞,但是解除阻塞的任務(wù)不一定能獲得CPU權(quán)限,因此任務(wù)主體代碼也不會(huì)總是精確周期性執(zhí)行。
如果要想精確周期性執(zhí)行某個(gè)任務(wù),可以使用系統(tǒng)節(jié)拍鉤子函數(shù)vApplicationTickHook(),它在系統(tǒng)節(jié)拍中斷服務(wù)函數(shù)中被調(diào)用,因此這個(gè)函數(shù)中的代碼必須簡(jiǎn)潔。
原文鏈接:https://freertos.blog.csdn.net/article/details/51705148
相關(guān)推薦
- 2022-08-15 前端寫代碼的時(shí)候,不滿足條件程序停止執(zhí)行下面的程序,并彈窗提示
- 2022-07-03 C語(yǔ)言詳解strcmp函數(shù)的分析及實(shí)現(xiàn)_C 語(yǔ)言
- 2023-10-28 Nginx 出現(xiàn)403 Forbidden 的幾種解決方案
- 2022-09-02 useEffect支持async及await使用方式_React
- 2023-07-10 SpringBoot AOP+注解方式實(shí)現(xiàn)多數(shù)據(jù)源切換可能遇到的問(wèn)題
- 2023-12-06 Access denied for user root @ localhost (using p
- 2022-03-04 前端添加把數(shù)組轉(zhuǎn)為String,后端查詢顯示把String拆分為數(shù)組
- 2021-12-03 Android識(shí)別NFC芯片制造商的方法_Android
- 最近更新
-
- 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)證過(guò)濾器
- 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)程分支