在閱讀本文之前,有兩個定義在FreeRTOSConfig.h中的宏,你必須先明白它們是什么意思,《FreeRTOS內(nèi)核配置說明》一文中,講解了這兩個宏:
configKERNEL_INTERRUPT_PRIORITY
configMAX_SYSCALL_INTERRUPT_PRIORITY
FreeRTOS與Cortex-M內(nèi)核可謂是絕配,以至于讓移植和使用FreeRTOS都變得更簡單起來。根據(jù)FreeRTOS官方反饋,在Cortex-M內(nèi)核上使用FreeRTOS大多數(shù)的問題點是由不正確的優(yōu)先級設(shè)置引起的。這個問題也是在意料之中的,因為盡管Cortex-M內(nèi)核的中斷模式是非常強大的,但對于那些使用傳統(tǒng)中斷優(yōu)先級架構(gòu)的工程師來說,Cortex-M內(nèi)核中斷機制也有點笨拙(或者是說使用比較繁瑣),并且違反直覺(這個主要是因為Cortex-M中斷優(yōu)先級數(shù)值越大代表的優(yōu)先級反而越小)。本章打算描述Cortex-M的中斷優(yōu)先級機制,并描述怎樣結(jié)合RTOS內(nèi)核使用。
說明:雖然Cortex-M內(nèi)核的優(yōu)先級方案看上去比較復(fù)雜,但每一個官方發(fā)布的FreeRTOS 接口包(在FreeRTOSV7.2.0FreeRTOSSourceportable文件夾中,一般為port.c)內(nèi)都會有正確配置的演示例程,可以以此為參考。
1.有效優(yōu)先級
1.1Cortex-M 硬件詳述
首先需要清楚有效優(yōu)先級的總數(shù),這取決于微控制器制造商怎么使用Cortex內(nèi)核。所以,并不是所有的Cortex-M內(nèi)核微處理器都具有相同的中斷優(yōu)先級級別。 Cortex-M構(gòu)架自身最多允許256級可編程優(yōu)先級(優(yōu)先級配置寄存器最多8位,所以優(yōu)先級范圍從0x00~0xFF),但是絕大多數(shù)微控制器制造商只是使用其中的一部分優(yōu)先級。比如,TI Stellaris Cortex-M3和Cortex-M4微控制器使用優(yōu)先級配置寄存器的3個位,能提供8級優(yōu)先級。再比如,NXP LPC17xx Cortex-M3微控制器使用優(yōu)先級配置寄存器的5個位,能提供32級優(yōu)先級。
1.2應(yīng)用到RTOS
RTOS中斷嵌套方案將有效的中斷優(yōu)先級分成兩組:一組可以通過RTOS臨界區(qū)屏蔽,另一組不受RTOS影響,永遠都是使能的。宏configMAX_SYSCALL_INTERRUPT_PRIORITY在FreeRTOSConfig.h中配置,定義兩組中斷優(yōu)先級的邊界。邏輯優(yōu)先級高于此值的中斷不受RTOS影響。最優(yōu)值取決于微控制器使用的優(yōu)先級配置寄存器的位數(shù)。
2.與數(shù)值相反的優(yōu)先級值和邏輯優(yōu)先級設(shè)置
2.1Cortex-M 硬件詳述
有必要先解釋一下優(yōu)先級值和邏輯優(yōu)先級:在Cortex-M內(nèi)核中,假如有8級優(yōu)先級,我們說優(yōu)先級值是0~7,但數(shù)值最大的優(yōu)先級7卻代表著最低的邏輯優(yōu)先級。很多使用傳統(tǒng)傳統(tǒng)中斷優(yōu)先級架構(gòu)的工程師會覺得這樣比較繞,違反直覺。以下內(nèi)容提到的優(yōu)先級要仔細區(qū)分是優(yōu)先級數(shù)值還是邏輯優(yōu)先級。 接下來需要清楚的是,在Cortex-M內(nèi)核中,一個中斷的優(yōu)先級數(shù)值越低,邏輯優(yōu)先級卻越高。比如,中斷優(yōu)先級為2的中斷可以搶占中斷優(yōu)先級為5的中斷,但反過來就不行。換句話說,中斷優(yōu)先級2比中斷優(yōu)先級5的優(yōu)先級更高。 這是Cortex-M內(nèi)核最容易讓人犯錯之處,因為大多數(shù)的非Cortex-M內(nèi)核微控制器的中斷優(yōu)先級表述是與之相反的。
2.2應(yīng)用到 RTOS
以“FromISR”結(jié)尾的FreeRTOS函數(shù)是具有中斷調(diào)用保護的(執(zhí)行這些函數(shù)會進入臨界區(qū)),但是就算是這些函數(shù),也不可以被邏輯優(yōu)先級高于configMAX_SYSCALL_INTERRUPT_PRIORITY的中斷服務(wù)函數(shù)調(diào)用。(宏configMAX_SYSCALL_INTERRUPT_PRIORITY定義在頭文件FreeRTOSConfig.h中)。因此,任何使用RTOSAPI函數(shù)的中斷服務(wù)例程的中斷優(yōu)先級數(shù)值大于等于configMAX_SYSCALL_INTERRUPT_PRIORITY宏的值。這樣就能保證中斷的邏輯優(yōu)先級等于或低于configMAX_SYSCALL_INTERRUPT_PRIORITY。 Cortex中斷默認情況下有一個數(shù)值為0的優(yōu)先級。大多數(shù)情況下0代表最高級優(yōu)先級。因此,絕對不可以在優(yōu)先級為0的中斷服務(wù)例程中調(diào)用RTOSAPI函數(shù)。
3.Cortex-M 內(nèi)部優(yōu)先級概述
3.1Cortex-M 硬件詳述
Cortex-M內(nèi)核的中斷優(yōu)先級寄存器是以最高位(MSB)對齊的。比如,如果使用了3位來表達優(yōu)先級,則這3個位位于中斷優(yōu)先級寄存器的bit5、bit6、bit7位。剩余的bit0~bit4可以設(shè)置成任何值,但為了兼容,最好將他們設(shè)置成1. Cortex-M優(yōu)先級寄存器最多有8位,如果一個微控制器只使用了其中的3位,那么這3位是以最高位對齊的,見下圖:
某微控制器只使用了優(yōu)先級寄存器中的3位,下圖展示了優(yōu)先級數(shù)值5(二進制101B)是怎樣在優(yōu)先級寄存器中存儲的。如果優(yōu)先級寄存器中未使用的位置1,下圖也展示了為什么數(shù)值5(二進制0000 0101B)可以看成數(shù)值191(二進制1011 1111)的。
某微控制器只使用了優(yōu)先級寄存器中的4位,下圖展示了優(yōu)先級數(shù)值5(二進制101B)是怎樣在優(yōu)先級寄存器中存儲的。如果優(yōu)先級寄存器中未使用的位置1,下圖也展示了為什么數(shù)值5(二進制0000 0101B)可以看成數(shù)值95(二進制0101 1111)的。
3.2應(yīng)用到 RTOS
上文中已經(jīng)描述,那些在中斷服務(wù)例程中調(diào)用RTOS API函數(shù)的中斷邏輯優(yōu)先級必須低于或等于configMAX_SYSCALL_INTERRUPT_PRIORITY(低邏輯優(yōu)先級意味著高優(yōu)先級數(shù)值)。 CMSIS以及不同的微控制器供應(yīng)商提供了可以設(shè)置某個中斷優(yōu)先級的庫函數(shù)。一些庫函數(shù)的參數(shù)使用最低位對齊,另一些庫函數(shù)的參數(shù)可能使用最高位對齊,所以,使用時應(yīng)該查閱庫函數(shù)的應(yīng)用手冊進行正確設(shè)置。 可以在FreeRTOSConfig.h中設(shè)置宏configMAX_SYSCALL_INTERRUPT_PRIORITY和configKERNEL_INTERRUPT_PRIORITY的值。這兩個宏需要根據(jù)Cortex-M內(nèi)核自身的情況進行設(shè)置,要以最高有效位對齊。比如某微控制器使用中斷優(yōu)先級寄存器中的3位,設(shè)置configKERNEL_INTERRUPT_PRIORITY的值為5,則代碼為:
#defineconfigKERNEL_INTERRUPT_PRIORITY(5<<(8-3))? 宏configKERNEL_INTERRUPT_PRIORITY指定RTOS內(nèi)核使用的中斷優(yōu)先級,因為RTOS內(nèi)核不可以搶占用戶任務(wù),因此這個宏一般設(shè)置為硬件支持的最小優(yōu)先級。對于Cortex-M硬件,RTOS使用到硬件的PendSV和SysTick硬件中斷,在函數(shù)xPortStartScheduler()中(該函數(shù)在port.c中,由啟動調(diào)度器函數(shù)vTaskStartScheduler()調(diào)用),將PendSV和SysTick硬件中斷優(yōu)先級寄存器設(shè)置為宏configKERNEL_INTERRUPT_PRIORITY指定的值。 ? 有關(guān)代碼如下(位于port.c):?/*PendSV優(yōu)先級設(shè)置寄存器地址為0xe000ed22 ?SysTick優(yōu)先級設(shè)置寄存器地址為0xe000ed23*/ #define?portNVIC_SYSPRI2_REG?????(?*?(?(?volatile?uint32_t?*?)?0xe000ed20?)) ? #define?portNVIC_PENDSV_PRI?(?(?(uint32_t)configKERNEL_INTERRUPT_PRIORITY?)?<16UL?) #define?portNVIC_SYSTICK_PRI?(?(?(uint32_t)configKERNEL_INTERRUPT_PRIORITY?)?<24UL?) /*?….?*/ /*確保PendSV?和SysTick為最低優(yōu)先級中斷?*/ portNVIC_SYSPRI2_REG?|=?portNVIC_PENDSV_PRI; portNVIC_SYSPRI2_REG?|=portNVIC_SYSTICK_PRI;?
4.臨界區(qū)
4.1Cortex-M 硬件詳述
RTOS內(nèi)核使用Cortex-M內(nèi)核的BASEPRI寄存器來實現(xiàn)臨界區(qū)(注:BASEPRI為優(yōu)先級屏蔽寄存器,優(yōu)先級數(shù)值大于或等于該寄存器的中斷都會被屏蔽,優(yōu)先級數(shù)值越大,邏輯優(yōu)先級越低,但是為零時不屏蔽任何中斷)。這允許RTOS內(nèi)核可以只屏蔽一部分中斷,因此可以提供一個靈活的中斷嵌套模式。 那些需要在中斷調(diào)用時保護的API函數(shù),F(xiàn)reeRTOS使用寄存器BASEPRI實現(xiàn)中斷保護臨界區(qū)。當進入臨界區(qū)時,將寄存器BASEPRI的值設(shè)置成configMAX_SYSCALL_INTERRUPT_PRIORITY,當退出臨界區(qū)時,將寄存器BASEPRI的值設(shè)置成0。很多Bug反饋都提到,當退出臨界區(qū)時不應(yīng)該將寄存器設(shè)置成0,應(yīng)該恢復(fù)它之前的狀態(tài)(之前的狀態(tài)不一定是0)。但是Cortex-M NVIC決不會允許一個低優(yōu)先級中斷搶占當前正在執(zhí)行的高優(yōu)先級中斷,不管BASEPRI寄存器中是什么值。與進入臨界區(qū)前先保存BASEPRI的值,退出臨界區(qū)再恢復(fù)的方法相比,退出臨界區(qū)時將BASEPRI寄存器設(shè)置成0的方法可以獲得更快的執(zhí)行速度。
4.2應(yīng)用到RTOS kernel
RTOS內(nèi)核通過寫configMAX_SYSCALL_INTERRUPT_PRIORITY的值到BASEPRI寄存器的方法創(chuàng)建臨界區(qū)。中斷優(yōu)先級0(具有最高的邏輯優(yōu)先級)不能被BASEPRI寄存器屏蔽,因此,configMAX_SYSCALL_INTERRUPT_PRIORITY絕不可以設(shè)置成0。
原文標題:Cortex-M內(nèi)核使用FreeRTOS特別注意事項
文章出處:【微信公眾號:安芯教育科技】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。
責任編輯:haq
-
內(nèi)核
+關(guān)注
關(guān)注
3文章
1372瀏覽量
40293 -
RTOS
+關(guān)注
關(guān)注
22文章
813瀏覽量
119649 -
FreeRTOS
+關(guān)注
關(guān)注
12文章
484瀏覽量
62182
原文標題:Cortex-M內(nèi)核使用FreeRTOS特別注意事項
文章出處:【微信號:Ithingedu,微信公眾號:安芯教育科技】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。
發(fā)布評論請先 登錄
相關(guān)推薦
評論