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

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

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

3天內不再提示

Cortex-M裸機環境下臨界區保護的三種實現

strongerHuang ? 來源:痞子衡嵌入式 ? 作者:痞子衡 ? 2021-09-08 09:23 ? 次閱讀

今天給大家分享的是Cortex-M裸機環境下臨界區保護的三種實現。

嵌入式玩過 RTOS 的朋友想必都對 OS_ENTER_CRITICAL()、OS_EXIT_CRITICAL() 這個功能代碼對特別眼熟,在 RTOS 里常常會有多任務(進程)處理,有些情況下一些特殊操作(比如 XIP 下 Flash 擦寫、低功耗模式切換)不能被隨意打斷,或者一些共享數據區不能被無序訪問(A 任務正在讀,B 任務卻要寫),這時候就要用到臨界區保護策略了。

所謂臨界區保護策略,簡單說就是系統中硬件臨界資源或者軟件臨界資源,多個任務必須互斥地對它們進行訪問。RTOS 環境下有現成的臨界區保護接口函數,而裸機系統里其實也有這種需求。在裸機系統里,臨界區保護主要就是跟系統全局中斷控制有關。痞子衡之前寫過一篇 《嵌入式MCU中通用的三重中斷控制設計》,文中介紹的第三重也是最頂層的中斷控制是系統全局中斷控制,今天痞子衡就從這個系統全局中斷控制使用入手給大家介紹三種臨界區保護做法:

一、臨界區保護測試場景

關于臨界區保護的測試場景無非兩種。第一種場景是受保護的多個任務間并無關聯,也不會互相嵌套,如下面的代碼所示,task1 和 task2 是按序被保護的,因此 enter_critical() 和 exit_critical() 這兩個臨界區保護函數總是嚴格地成對執行:

void critical_section_test(void)

{

// 進入臨界區

enter_critical();

// 做受保護的任務1

do_task1();

// 退出臨界區

exit_critical();

// 進入臨界區

enter_critical();

// 做受保護的任務2,與任務1無關聯

do_task2();

// 退出臨界區

exit_critical();

}

第二種場景就是多個任務間可能有關聯,會存在嵌套情況,如下面的代碼所示,task2 是 task1 的一個子任務,這種情況下,你會發現實際上是先執行兩次 enter_critical(),然后再執行兩次 exit_critical()。需要注意的是 task1 里面的子任務 task3 雖然沒有像子任務 task2 那樣被主動加一層保護,但由于主任務 task1 整體是受保護的,因此子任務 task3 也應該是受保護的。

void do_task1(void)

{

// 進入臨界區

enter_critical();

// 做受保護的任務2,是任務1中的子任務

do_task2();

// 退出臨界區

exit_critical();

// 做任務3

do_task3();

}

void critical_section_test(void)

{

// 進入臨界區

enter_critical();

// 做受保護的任務1

do_task1();

// 退出臨界區

exit_critical();

}

二、臨界區保護三種實現

上面的臨界區保護測試場景很清楚了,現在到 enter_critical()、exit_critical() 這對臨界區保護函數的實現環節了:

2.1 入門做法

首先是非常入門的做法,直接就是對系統全局中斷控制函數 __disable_irq()、__enable_irq() 的封裝。回到上一節的測試場景里,這種實現可以很好地應對非嵌套型任務的保護,但是對于互相嵌套的任務保護就失效了。上一節測試代碼里,task3 應該也要受到保護的,但實際上并沒有被保護,因為緊接著 task2 后面的 exit_critical() 直接就打開了系統全局中斷。

void enter_critical(void)

{

// 關閉系統全局中斷

__disable_irq();

}

void exit_critical(void)

{

// 打開系統全局中斷

__enable_irq();

}

2.2 改進做法

針對入門做法,可不可以改進呢?當然可以,我們只需要加一個全局變量 s_lockObject 來實時記錄當前已進入的臨界區保護的次數,即如下代碼所示。每調用一次 enter_critical() 都會直接關閉系統全局中斷(保證臨界區一定是受保護的),并記錄次數,而調用 exit_critical() 時僅當當前次數是 1 時(即當前不是臨界區保護嵌套情況),才會打開系統全局中斷,否則只是抵消一次進入臨界區次數而已。改進后的實現顯然可以保護上一節測試代碼里的 task3 了。

static uint32_t s_lockObject;

void init_critical(void)

{

__disable_irq();

// 清零計數器

s_lockObject = 0;

__enable_irq();

}

void enter_critical(void)

{

// 關閉系統全局中斷

__disable_irq();

// 計數器加 1

++s_lockObject;

}

void exit_critical(void)

{

if (s_lockObject 《= 1)

{

// 僅當計數器不大于 1 時,才打開系統全局中斷,并清零計數器

s_lockObject = 0;

__enable_irq();

}

else

{

// 當計數器大于 1 時,直接計數器減 1 即可

--s_lockObject;

}

}

2.3 終極做法

上面的改進做法雖然解決了臨界區任務嵌套保護的問題,但是增加了一個全局變量和一個初始化函數,實現不夠優雅,并且嵌入式系統里全局變量極容易被篡改,存在一定風險,有沒有更好的實現呢?當然有,這要借助 Cortex-M 處理器內核的特殊屏蔽寄存器 PRIMASK,下面是 PRIMASK 寄存器位定義(取自 ARMv7-M 手冊),其僅有最低位 PM 是有效的,當 PRIMASK[PM] 為 1 時,系統全局中斷是關閉的(將執行優先級提高到 0x0/0x80);當 PRIMASK[PM] 為 0 時,系統全局中斷是打開的(對執行優先級無影響)。

345fd67a-1018-11ec-8fb8-12bb97331649.png

看到這,你應該明白了 __disable_irq()、__enable_irq() 功能其實就是操作 PRIMASK 寄存器實現的。既然 PRIMASK 寄存器控制也保存了系統全局中斷的開關狀態,我們可以通過獲取 PRIMASK 值來替代上面改進做法里的全局變量 s_lockObject 的功能,代碼實現如下:

uint32_t enter_critical(void)

{

// 保存當前 PRIMASK 值

uint32_t regPrimask = __get_PRIMASK();

// 關閉系統全局中斷(其實就是將 PRIMASK 設為 1)

__disable_irq();

return regPrimask;

}

void exit_critical(uint32_t primask)

{

// 恢復 PRIMASK

__set_PRIMASK(primask);

}

因為 enter_critical()、exit_critical() 函數原型有所變化,因此使用上也要相應改變下:

void critical_section_test(void)

{

// 進入臨界區

uint32_t primask = enter_critical();

// 做受保護的任務

do_task();

// 退出臨界區

exit_critical(primask);

// 。..

}

附錄、PRIMASK寄存器設置函數在各 IDE 下實現

//////////////////////////////////////////////////////// IAR 環境下實現(見 cmsis_iccarm.h 文件)#define __set_PRIMASK(VALUE) (__arm_wsr(“PRIMASK”, (VALUE)))#define __get_PRIMASK() (__arm_rsr(“PRIMASK”))//////////////////////////////////////////////////////// Keil 環境下實現(見 cmsis_armclang.h 文件)

__STATIC_FORCEINLINE void __set_PRIMASK(uint32_t priMask)

{

__ASM volatile (“MSR primask, %0” : : “r” (priMask) : “memory”);

}

__STATIC_FORCEINLINE uint32_t __get_PRIMASK(void)

{

uint32_t result;

__ASM volatile (“MRS %0, primask” : “=r” (result) );

return(result);

}

至此,Cortex-M裸機環境下臨界區保護的三種實現已經講述完畢,你學廢了嗎?

責任編輯:haq

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

    關注

    6043

    文章

    44617

    瀏覽量

    638060
  • 嵌入式
    +關注

    關注

    5091

    文章

    19176

    瀏覽量

    307264
  • RTOS
    +關注

    關注

    22

    文章

    819

    瀏覽量

    119863

原文標題:單片機非RTOS時,臨界區保護的實現辦法

文章出處:【微信號:strongerHuang,微信公眾號:strongerHuang】歡迎添加關注!文章轉載請注明出處。

收藏 人收藏

    評論

    相關推薦

    示波器的三種觸發模式

    示波器的觸發方式不僅影響波形捕捉的時機,還決定了顯示的波形是否穩定。 常見的觸發模式有三種: 單次觸發 (Single)、 正常觸發 (Normal)和 自動觸發 (Auto)。下面將對這三種觸發
    的頭像 發表于 01-07 11:04 ?327次閱讀
    示波器的<b class='flag-5'>三種</b>觸發模式

    如何使用Ozone分析Cortex-M異常

    Ozone可以幫助用戶快速分析和查找導致CPU故障的軟件bug。本文解釋如何使用Ozone的調試功能,深入了解Cortex-M架構上的這些錯誤。
    的頭像 發表于 11-29 11:14 ?904次閱讀
    如何使用Ozone分析<b class='flag-5'>Cortex-M</b>異常

    三種封裝形式的400G光模塊概述

    本文主要就三種封裝形式(QSFP-DD、OSFP、QSFP112)的400G光模塊做了簡單的梳理,從為什么會有400G光模塊問世?400G光模塊在三種封裝形式的各個具體型號(以短距離為主,最遠2km),
    的頭像 發表于 11-11 11:35 ?531次閱讀
    <b class='flag-5'>三種</b>封裝形式<b class='flag-5'>下</b>的400G光模塊概述

    mosfet的三種工作狀態及工作條件是什么

    )的不同,可以工作在三種主要狀態:截止狀態、線性和飽和。 1. 截止狀態 工作狀態描述 : 當VGS小于MOSFET的開啟電壓(VGS(TH))時,MOSF
    的頭像 發表于 10-06 16:51 ?2649次閱讀

    電流保護裝置的接線方式主要有哪三種

    三種主要的電流保護裝置接線方式的介紹: 1. 星形接線(Star Connection) 星形接線,也稱為Y型接線,是一常見的電流保護裝置接線方式。在這種接線方式中,電流
    的頭像 發表于 09-13 15:48 ?935次閱讀

    如何利用三種 SOT-563 封裝實現共同布局

    電子發燒友網站提供《如何利用三種 SOT-563 封裝實現共同布局.pdf》資料免費下載
    發表于 09-10 14:25 ?0次下載
    如何利用<b class='flag-5'>三種</b> SOT-563 封裝<b class='flag-5'>實現</b>共同布局

    單片機的三種總線結構

    單片機的三種總線結構包括地址總線(Address Bus, AB)、數據總線(Data Bus, DB)和控制總線(Control Bus, CB)。這三種總線在單片機內部及與外部設備之間的數據傳輸
    的頭像 發表于 09-10 11:32 ?3646次閱讀

    丙類諧振功率放大器有哪三種工作狀態

    丙類諧振功率放大器是一廣泛應用于通信、廣播、雷達等領域的高頻功率放大器。它利用非線性元件的非線性特性,通過調整工作狀態,實現高效率、高功率輸出。丙類諧振功率放大器有三種工作狀態:截止
    的頭像 發表于 08-01 11:05 ?1867次閱讀

    環境功能噪聲測量的方法

    環境功能噪聲測量是環境保護和城市規劃中的重要環節。本文將詳細介紹聲環境功能噪聲測量的方法,包括測量目的、測量標準、測量設備、測量步驟、
    的頭像 發表于 06-03 16:31 ?812次閱讀

    淺析FreeRTOS任務調度器的三種調度算法和應用

    FreeRTOS在MCU領域應用非常廣泛,今天就給大家講解一FreeRTOS調度器中的三種調度算法,以及在瑞薩RZ/T2L MPU中的應用。
    的頭像 發表于 05-10 14:02 ?7796次閱讀
    淺析FreeRTOS任務調度器的<b class='flag-5'>三種</b>調度算法和應用

    Cortex-M0+內核介紹

    和8位的價位實現32位性能。處理器的低門數使其能夠部署在需要簡單功能的應用中。 作為ARM Cortex-M處理器系列的最新成員,32位Cortex-M0+處理器采用了低成本90納米低功耗(LP)工藝,耗電量僅9μA/MHz,約
    的頭像 發表于 03-27 09:13 ?1142次閱讀
    <b class='flag-5'>Cortex-M</b>0+內核介紹

    三種實現光模塊更高傳輸速率的技術

    本文介紹三種提高光模塊傳輸速率的技術:波分復用技術、多路并行傳輸技術和高階調制技術。波分復用技術利用光的波長特性將不同波長的信號同時傳輸在同一條光纖上,實現光纖的并行傳輸。多路并行傳輸技術利用多個通道同時傳輸數據,從而提高傳輸帶寬和速度。
    的頭像 發表于 03-11 15:31 ?841次閱讀

    簡述斬波電路的三種控制方式

    斬波電路是一常見的電力電子器件,廣泛應用于直流電壓調節、電壓變換、電流變換等領域。它可以實現對電流和電壓的控制,以滿足不同的電氣設備的需求。斬波電路的控制方式主要有三種:脈寬調制控制、頻率調制控制
    的頭像 發表于 03-11 15:22 ?4556次閱讀

    Cortex-M85內核單片機如何快速上手

    2022年4月,Arm推出了全新的MCU級內核Cortex-M85。截止目前(2024年2月),Cortex-M85是最新、最強的Cortex-M內核。
    發表于 02-29 09:35 ?804次閱讀
    <b class='flag-5'>Cortex-M</b>85內核單片機如何快速上手

    M70 PLC增加注解的三種方法簡析

    M70 PLC增加注解的三種方法
    的頭像 發表于 02-26 09:59 ?1149次閱讀
    <b class='flag-5'>三</b>菱<b class='flag-5'>M</b>70 PLC增加注解的<b class='flag-5'>三種</b>方法簡析
    主站蜘蛛池模板: 国产福利视频一区二区 | 国产人妻人伦精品836700 | 日韩精品一卡二卡三卡四卡2021 | 免费果冻传媒2021在线观看 | 亚洲高清无在码在线无弹窗 | 免费麻豆国产黄网站在线观看 | 久久综合给合久久狠狠狠… | 久久综合老色鬼网站 | 男欢女爱免费视频 | 久久久久久免费高清电影 | 97成人免费视频 | 成年视频xxxxxx在线 | 在线观看中文字幕国产 | 成人毛片一区二区三区 | 美女动态图真人后进式 | 99亚偷拍自图区亚洲 | 色婷婷国产精品视频一区二区 | 大学生宿舍飞机china free | 国产精品久久久久婷婷五月色 | 在线欧美 精品 第1页 | 中文字幕亚洲无线码一区 | 凌晨三点免费WWW | 97碰成视频免费 | 亚洲AV无码专区国产乱码网站 | 一个人在线观看视频 | 久久综合给会久久狠狠狠 | 好硬好湿好爽再深一点视频 | 在线高清无码欧美久章草 | 国内视频在线精品一区 | 99精品在线看 | 玩弄放荡人妻一区二区三区 | 亚洲精品123区在线观看 | h版动漫在线播放的网站 | 日韩精品免费在线观看 | 天天狠狠弄夜夜狠狠躁·太爽了 | 亚洲一区综合图区 | 24小时日本在线观看片 | 欧美嫩freexxxhddvd | 亚洲日韩乱码人人爽人人澡人 | 久久综合给合久久狠狠狠… | 国产不卡免费 |