色哟哟视频在线观看-色哟哟视频在线-色哟哟欧美15最新在线-色哟哟免费在线观看-国产l精品国产亚洲区在线观看-国产l精品国产亚洲区久久

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

什么是RTOS臨界段

汽車電子技術 ? 來源:物聯網IoT開發 ? 作者: 杰杰mcu ? 2023-02-14 09:48 ? 次閱讀

什么是臨界段

代碼的臨界段也稱為臨界區,指處理時不可分割的代碼區域,一旦這部分代碼開始執行,則不允許任何中斷打斷。為確保臨界段代碼的執行不被中斷,在進入臨界段之前須關中斷,而臨界段代碼執行完畢后,要立即打開中斷。

臨界段的作用

其實在RTOS中,使用最多的臨界段是OS本身的調用,但是我們用戶也是需要對臨界資源進行保護的(臨界資源是一次僅允許一個線程使用的共享資源),特別是一些全局變量,當線程正在使用的時候不希望有人來打斷我的操作,就行很多時候我們寫代碼時,需要集中精力,不希望別人打斷我們的思路一樣。這樣子使得系統的運行更加穩定健壯。

什么時候會打斷代碼的執行?

顧名思義,代碼正在正常運行的時候,基本不會被打斷,能被打斷的都是系統發生了異常(中斷也是異常),在OS中,除了外部中斷能將正在運行的代碼打斷,還有線程的調度——PendSV,系統產生 PendSV中斷,在 PendSV Handler 里面實現線程的切換。我們要將這項東西屏蔽掉,保證當前只有一個線程在使用臨界資源。

如何關閉中斷?

其實,在我們常用的MCU中,一般為Cortex-M內核的,M內核是有一些指令能快速關閉中斷,一起來看看Cortex-M權威指南吧(以Cortex-M3為例)。

圖片

簡單來說,快速屏蔽中斷就是處理這些內核寄存器,在Cortex-M中有相應的操作指令,一般我們無需關注,因為OS已經給我們寫好了這些底層的東西。不過如果你是想自己寫一個OS的話,可以了解一下,要訪問 PRIMASK, FAULTMASK 以及 BASEPRI,同樣要使用 MRS/MSR 指令,如:
1MRS R0, BASEPRI ;讀取 BASEPRI 到 R02MRS R0, FAULTMASK ;似上
3MRS R0, PRIMASK ;似上
4MSR BASEPRI, R0 ;寫入 R0 到 BASEPRI 中
5MSR FAULTMASK, R0 ;似上
6MSR PRIMASK, R0 ;似上

只有在特權級下,才允許訪問這 3 個寄存器。

其實,為了快速地開關中斷, CM3 還專門設置了一條 CPS 指令,有 4 種用法:

圖片

1CPSID I ;PRIMASK=1, ;關中斷
2CPSIE I ;PRIMASK=0, ;開中斷
3CPSID F ;FAULTMASK=1, ;關異常
4CPSIE F ;FAULTMASK=0 ;開異常
上面的代碼中的PRIMASKFAULTMASTCortex-M 內核 里面三個中斷屏蔽寄存器中的兩個,還有一個是 BASEPRI,這些寄存器都用于屏蔽中斷。具體的作用見表格(表格出自《【野火】RT-Thread 內核實現與應用開發實戰指南》)
名字 功能描述
PRIMASK 這是個只有單一比特的寄存器。 在它被置 1 后,就關掉所有可屏蔽的異常,只剩下 NMI 和硬 FAULT 可以響應。它的缺省值是 0,表示沒有關中斷。
FAULTMASK 這是個只有 1 個位的寄存器。當它置 1 時,只有 NMI 才能響應,所有其它的異常,甚至是硬 FAULT,也通通閉嘴。它的缺省值也是 0,表示沒有關異常。
BASEPRI 這個寄存器最多有 9 位(由表達優先級的位數決定)。它定義了被屏蔽優先級的閾值。當它被設成某個值后,所有優先級號大于等于此值的中斷都被關(優先級號越大,優先級越低)。但若被設成 0,則不關閉任何中斷, 0 也是缺省值。

不同OS的處理臨界段的區別

FreeRTOS

FreeRTOS對中斷的開和關是通過操作 BASEPRI 寄存器來實現的,即大于等于 BASEPRI 的值的中斷會被屏蔽,小于 BASEPRI 的值的中斷則不會被屏蔽。這樣子的好處就是用戶可以設置 BASEPRI 的值來選擇性的給一些非常緊急的中斷留一條后路。比如飛控的防撞處理。代碼在portmacro.h 中實現:

屏蔽中斷:

1static portFORCE_INLINE void vPortRaiseBASEPRI( void )
 2{
 3uint32_t ulNewBASEPRI = configMAX_SYSCALL_INTERRUPT_PRIORITY;
 4
 5    __asm
 6    {
 7        msr basepri, ulNewBASEPRI
 8        dsb
 9        isb
10    }
11}

打開中斷:

1static portFORCE_INLINE void vPortSetBASEPRI( uint32_t ulBASEPRI )
2{
3    __asm
4    {
5        msr basepri, ulBASEPRI
6    }
7}

RT-Thread

FreeRTOS不同的是,RT-Thread 對臨界段的保護處理的很干脆,不管三七二十一直接把中斷全部關了(直接操作PRIMASK內核寄存器), 只有NMI FAULT 和硬 FAULT能被相應。 這種方法簡單粗暴,是很不錯的選擇。一般我們臨界段的處理時間是比較短的,關了再開其實并沒有太大的影響。

現在要看看RT-Thread的關中斷的代碼實現:
1rt_hw_interrupt_disable    PROC
2    EXPORT  rt_hw_interrupt_disable
3    MRS     r0, PRIMASK
4    CPSID   I
5    BX      LR
6    ENDP

開中斷:

1rt_hw_interrupt_enable    PROC
2    EXPORT  rt_hw_interrupt_enable
3    MSR     PRIMASK, r0
4    BX      LR
5    ENDP

這短短的幾句代碼其實還是很有意思的,我就引用火哥的話來解釋一下這些處理操作(我個人是不會匯編的,但是跟著書來解讀這些代碼還是很輕而易舉的)

可能有人懂匯編的話,就會看出來,關中斷,不就是直接使用 CPSID I 指令就行了嘛~開中斷,不就是使用 CPSIE I 指令就行了嘛,為啥跟我等凡人想的不一樣?

RT-Thread的處理好像是多此一舉了,實則不然,“所有東西的存在必然有其存在的意義”這句話應該沒人反駁吧~~因為RT-Thread要防止用戶錯誤地退出了中斷臨界段,因為這樣子可能會產生巨大的危害,所以RT-Thread將當前的PRIMASK的狀態保存起來,這樣子就必須要關多少次中斷就得開多少次中斷。

怎么說呢,用例子來證明吧:

1/* 臨界段 1 開始 */
 2rt_hw_interrupt_disable(); /* 關中斷,PRIMASK = 1 */
 3{
 4  /* 臨界段 2 */
 5  rt_hw_interrupt_disable(); /* 關中斷,PRIMASK = 1 */
 6  {
 7  }
 8  rt_hw_interrupt_enable(); /* 開中斷,PRIMASK = 0 */ (注意)
 9}
10/* 臨界段 1 結束 */
11rt_hw_interrupt_enable(); /* 開中斷,PRIMASK = 0 */
如果直接操作PRIMASK,而不保存PRIMASK的狀態,這樣子當臨界段2結束后調用一次打開中斷,那么連臨界段1的后半部分就無效了。而RT-Thread的實現就能很好避免這種問題,也用代碼來說明吧:
1/* 臨界段 1 開始 */
 2level1 = rt_hw_interrupt_disable(); /* 關中斷,level1=0,PRIMASK=1 */
 3{
 4  /* 臨界段 2 */
 5  level2 = rt_hw_interrupt_disable(); /* 關中斷,level2=1,PRIMASK=1 */ 
 6  {
 7  }
 8  rt_hw_interrupt_enable(level2); /* 開中斷,level2=1,PRIMASK=1 */ 
 9}
10/* 臨界段 1 結束 */
11rt_hw_interrupt_enable(level1); /* 開中斷,level1=0,PRIMASK=0 */

這樣子就完全避免了對吧!

有人又會問了,FreeRTOS的臨界段能允許嵌套嗎,答案是肯定的,FreeRTOS中早已給我們想好調用的函數了,并且全部使用宏定義實現了:
1#define portDISABLE_INTERRUPTS()                vPortRaiseBASEPRI()
2#define portENABLE_INTERRUPTS()                 vPortSetBASEPRI( 0 )
3#define portENTER_CRITICAL()                    vPortEnterCritical()
4#define portEXIT_CRITICAL()                     vPortExitCritical()
5#define portSET_INTERRUPT_MASK_FROM_ISR()       ulPortRaiseBASEPRI()
6#define portCLEAR_INTERRUPT_MASK_FROM_ISR(x)    vPortSetBASEPRI(x)
其實原理都是差不多的,通過保存和恢復寄存器basepri的數值就可以實現嵌套使用。
1UBaseType_t uxSavedInterruptStatus;
 2
 3uxSavedInterruptStatus = portSET_INTERRUPT_MASK_FROM_ISR();
 4{
 5  uxSavedInterruptStatus = portSET_INTERRUPT_MASK_FROM_ISR();
 6  {
 7     //臨界區代碼
 8  }
 9  portCLEAR_INTERRUPT_MASK_FROM_ISR( uxSavedInterruptStatus );
10}
11portCLEAR_INTERRUPT_MASK_FROM_ISR( uxSavedInterruptStatus );
進入中斷源碼的實現:
1static portFORCE_INLINE uint32_t ulPortRaiseBASEPRI( void )
 2{
 3uint32_t ulReturn, ulNewBASEPRI = configMAX_SYSCALL_INTERRUPT_PRIORITY;
 4
 5    __asm
 6    {
 7        mrs ulReturn, basepri
 8        msr basepri, ulNewBASEPRI
 9        dsb
10        isb
11    }
12    return ulReturn;
13}
退出中斷源碼實現:(跟前面的函數一樣)
1static portFORCE_INLINE void vPortSetBASEPRI( uint32_t ulBASEPRI )
2{
3    __asm
4    {
5        msr basepri, ulBASEPRI
6    }
7}

總結

對于時間關鍵的任務而言,恰如其分地使用 PRIMASK 和 BASEPRI 來暫時關閉一些中斷是非常重要的。

FreeRTOS源碼中就有多處臨界段的處理,除了FreeRTOS操作系統源碼所帶的臨界段以外,用戶寫應用的時候也有臨界段的問題,比如以下兩種:
  • 讀取或者修改變量(特別是用于任務間通信的全局變量)的代碼,一般來說這是最常見的臨界代碼。
  • 調用公共函數的代碼,特別是不可重入的函數,如果多個任務都訪問這個函數,結果是可想而知的。
總之,對于臨界段要做到執行時間越短越好,否則會影響系統的實時性。

那假如我有一個線程,處理的時間較長,但是我又不想被其他線程打斷,關中斷可能影響系統的正常運行,怎么辦呢?其實很簡單,在OS中一般可以直接掛起調度器,系統正常運行,但是不會切換線程,當我處理完再把調度器解除即可。

RTOS使用得好,開發起來比裸機更簡單,使用得不好,那將是噩夢——杰杰

▲▲▲▲▲

聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • 中斷
    +關注

    關注

    5

    文章

    898

    瀏覽量

    41514
  • 代碼
    +關注

    關注

    30

    文章

    4790

    瀏覽量

    68650
  • 執行
    +關注

    關注

    0

    文章

    16

    瀏覽量

    12610
收藏 人收藏

    評論

    相關推薦

    【安富萊】【RTX操作系統教程】第11章 臨界,任務鎖和中斷鎖

    多個任務都訪問這個函數,結果是可想而知的。 總之,對于臨界要做到執行時間越短越好,要不會影響系統的實時性。11.2中斷鎖 中斷鎖就是RTOS提供的開關中斷函數,因為Cortex-M3/M4的RTX源碼
    發表于 01-25 16:52

    轉:第15章 FreeRTOS臨界和開關中斷

    第15章FreeRTOS臨界和開關中斷 本章教程為大家講解兩個重要的概念,FreeRTOS的臨界和開關中斷。 本章教程配套的例子含Cortex-M3內核的STM32F103和Cor
    發表于 08-31 10:15

    第11章 臨界,任務鎖和中斷鎖

    轉rtx操作系統 本章教程為大家講解幾個重要的概念,臨界,任務鎖和中斷鎖。本章教程配套的例子含Cortex-M3內核的STM32F103和Cortex-M4內核的STM32F407。11.1 臨界
    發表于 10-04 19:58

    RTOS臨界知識詳解

    執行完畢后,要立即打開中斷。 臨界的作用 其實在RTOS中,使用最多的臨界是OS本身的調用,但是我們用戶也是需要對
    發表于 10-09 11:02

    【設計技巧】從單片機到操作系統(8)-RTOS臨界知識詳解

    斷,而臨界代碼執行完畢后,要立即打開中斷。臨界的作用 其實在RTOS中,使用最多的臨界
    發表于 08-01 08:30

    FreeRTOS關于臨界的疑問如何解答

    FreeRTOS中關于臨界的函數有4個,taskENTER_CRITICAL():進入臨界,用于任務中taskEXIT_CRITICAL():退出
    發表于 06-19 04:36

    請問STM32的FreeRTOS怎么使用臨界

    STM32的FreeRTOS怎么使用臨界
    發表于 08-28 08:06

    怎樣去使用FreeRTOS的中斷配置和臨界

    STM32之FreeRTOS:(一) 中斷配置和臨界的使用文章目錄STM32之FreeRTOS:(一) 中斷配置和臨界的使用前言 一、stm32的NVIC 分組配置二、FreeRT
    發表于 01-14 09:28

    求大佬分享FreeRTOS臨界代碼

    求大佬分享FreeRTOS臨界代碼
    發表于 02-07 06:28

    freertos中的臨界是如何實現的?

    怎么實現對應臨界的保護
    發表于 10-20 07:25

    什么是臨界 RTOS臨界的作用是什么

    代碼的臨界也稱為臨界區,指處理時不可分割的代碼區域,一旦這部分代碼開始執行,則不允許任何中斷打斷。為確保臨界代碼的執行不被中斷,在進入
    的頭像 發表于 10-06 14:38 ?1.2w次閱讀
    什么是<b class='flag-5'>臨界</b><b class='flag-5'>段</b> <b class='flag-5'>RTOS</b><b class='flag-5'>臨界</b><b class='flag-5'>段</b>的作用是什么

    FreeRTOS學習筆記--臨界代碼處關閉中斷

    FreeRTOS學習筆記--臨界代碼處關閉中斷一、臨界代碼二、Cortex-M4中斷管理三、中斷屏蔽實驗四、結語一、臨界
    發表于 12-04 14:51 ?10次下載
    FreeRTOS學習筆記--<b class='flag-5'>臨界</b><b class='flag-5'>段</b>代碼處關閉中斷

    FreeRTOS臨界

    臨界斷代碼也叫做臨界區,是指那些必須完整運行,不能被打斷的代碼,FreeRTOS與臨界斷代碼保護有關的函數有4個:taskENTER_CRITICAL() ——任務級進入
    發表于 12-04 16:06 ?0次下載
    FreeRTOS<b class='flag-5'>臨界</b><b class='flag-5'>段</b>

    STM32之FreeRTOS:(一) 中斷配置和臨界的使用

    STM32之FreeRTOS:(一) 中斷配置和臨界的使用文章目錄STM32之FreeRTOS:(一) 中斷配置和臨界的使用前言 一、stm32的NVIC 分組配置二、FreeRT
    發表于 01-14 15:43 ?3次下載
    STM32之FreeRTOS:(一) 中斷配置和<b class='flag-5'>臨界</b><b class='flag-5'>段</b>的使用

    RTOS臨界知識詳解

    代碼的臨界也稱為臨界區,指處理時不可分割的代碼區域,一旦這部分代碼開始執行,則不允許任何中斷打斷。
    的頭像 發表于 06-13 14:07 ?1165次閱讀
    <b class='flag-5'>RTOS</b><b class='flag-5'>臨界</b><b class='flag-5'>段</b>知識詳解
    主站蜘蛛池模板: 大香网伊人久久综合观看| 电影果冻传媒在线播放| 超碰 无码 中文字幕| 久久国产加勒比精品无码| 日本久久网站| 变态露出野外调教| 欧美三级黄色大片| 99久久精品全部| 免费人成在线观看视频不卡| 中文字幕人妻无码系列第三区 | 亚欧视频在线观看| 一个人看的WWW高清电影| 22eee在线播放成人免费视频| 久久免费精品一区二区| 又黄又湿免费高清视频| 久久久久久久久人体| 60老妇性xxxxhd| 色婷婷五月综合中文字幕| 国产乱人偷精品视频A人人澡| 伊人影院综合| 熟女强奷系列中文字幕| 精品亚洲麻豆1区2区3区 | 国产日韩成人内射视频| 特污兔午夜影视院| 狠狠狠的在啪线香蕉| 在教室做啊好大用力| 美女脱三角裤| 大地影院日本韩国电影免费观看| 视频在线免费观看| 九九精品在线播放| 被六个男人躁到一夜同性| 亚洲 欧美 综合 高清 在线| 精品无码国产AV一区二区三区| 亚洲天堂一区二区三区| 女人被躁到高潮嗷嗷叫免费| 国产福利视频一区二区| 又色又爽又黄gif动态视频| 人妻中文字幕无码久久AV爆| 果冻传媒在线播放| QVOD在线播放| 又黄又湿免费高清视频|