吴忠躺衫网络科技有限公司

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

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

3天內不再提示

RTOS多線程必須要MMU才行?

strongerHuang ? 來源:strongerHuang ? 作者:strongerHuang ? 2022-10-11 18:56 ? 次閱讀

前兩天看到有人說:計算機的多線程,必須要有MMU才行,否則系統不能正常運行。

MMU:Memory Management Unit,即內存管理單元,它是一個“好東西”,在大型的操作系統中,起到了關鍵的作用。

但是,并不是所有操作系統都需要MMU才行,我們嵌入式中很多常用的RTOS(實時操作系統)沒有MMU一樣可以實現多線程。

只是RTOS實現的多線程相對大型操作系統要簡單一點,其原理也要簡單的多。

下面就來圍繞RTOS給大家大家關于RTOS的多線程的內容。

多線程

1.單核“單線程”嚴格來說,單核處理器一次只能執行一條指令,也就是說只能“單線程”。(當然,多核處理器就不一樣)

為了在單核處理器上運行多個線程,我們實際上需要定期在線程之間進行快速切換,以便用戶感覺多個線程在并行運行。

比如處理器執行兩個線程,處理器實際在兩個線程之間來回切換,如下圖:9a4bc2c6-4854-11ed-a3b6-dac502259ad0.png

2.處理器在線程之間切換,它是如何做到的?

我們說的單核處理器是“單線程”的,它有一組寄存器,我們就叫這組寄存器屬于一個“線程”。

例如,計算兩個數字的總和時:

//假設我們有兩個整數:a和b

int c = a + b ;

實際發生的情況如下所示(當然,它取決于的MCU類型,但總體思路是相同):

# MIPS反匯編

LW V0, -32744(GP) # “a” 的值從RAM加載到寄存器V0

LW V1, -32740(GP) # 值“b” 從RAM加載到寄存器V1

ADDU V0, V1, V0 # a、b值相加,結果保存到寄存器V0中

SW V0, -32496(GP) # 寄存器V0的值存儲在RAM中(變量c所在的位置)

你會發現上面執行了4個動作,但是搶占式操作系統可以在任何時候搶占另一個線程,包括在這4個動作之間。

假如在這過程中有其他線程搶占了,其他線程同樣搶占了當前線程V0、 V1,如果不對V0、 V1進行保存,那么下次回來執行當前線程,結果就會出錯。

所以,針對當前這種問題,我們就需要在切換線程之前,對V0、 V1的數值進行保存,當下次切換到當前線程,再恢復V0、 V1的數值,大致流程如下:

9a61bd42-4854-11ed-a3b6-dac502259ad0.png

大概意思就是:當我們需要從一個線程切換到另一個線程時,內核獲得控制權,執行必要的內務處理(至少要保存和恢復寄存器值),然后將控制權轉移到下一個線程以運行。

線程的堆棧

上面說的搶占位置,到底在哪里,每個線程保存在哪個寄存器值中?這就是線程的堆棧的內容。

在有MMU的操作系統中,(用戶的)線程堆棧可以按需動態增長:線程需要的堆棧空間越多,線程堆棧就越多(如果內核允許)。

但是,我們一般的MCU卻沒有MMU這個“高端”的東西,所有RAM都靜態映射到地址空間。因此,每個線程都會有用于堆棧的RAM空間,如果線程使用的RAM超過堆棧的數量,則會導致內存溢出或細微的錯誤。(實際上,每個線程的堆??臻g只是一連續數組空間)。

因此,當我們決定為每個線程分配多少堆棧時,我們只是估計可能需要多少堆棧,但是具體多少可能不是很清楚。

比如,如果這是一個具有多層嵌套調用的GUI線程,則可能需要數個千字節,但如果它是一個流水燈的小線程,則可能幾十字節就足夠了。

假設我們有三個線程,它們的堆棧消耗如下:

9a7050b4-4854-11ed-a3b6-dac502259ad0.png

如上面所述,每個線程的寄存器值都保存在線程的堆棧中。線程的寄存器值集稱為線程的“上下文”。如下圖所示(線程A為在正在執行的“活動線程”):

9a895550-4854-11ed-a3b6-dac502259ad0.png

請注意,在正在執行的線程A的上下文沒有保存在堆棧中,堆棧指針指向線程A用戶數據的頂部,并且當前處理器的寄存器專用于線程A。

當內核決定將控制權切換到線程B時,它將執行以下操作:

將所有寄存器值保存到堆棧中(保存到線程A堆棧的頂部);

將堆棧指針切換到線程B的堆棧頂部;

從堆棧(從線程B的堆棧頂部)恢復所有寄存器值;

此時,你會看到:

9aa60c04-4854-11ed-a3b6-dac502259ad0.png

中斷(ISR)搶占

上面在執行過程中,或進行上下文切換時,還可能會涉及到一個非常重要的內容:中斷。

MCU通常具有外設:TIM、UART、 SPI、 CAN等,它們隨時都能發生重要事件以觸發中斷。

中斷條件是當當前正在執行的線程暫停時,處理器在一段時間內執行其他操作(Handles Interrupt),然后返回。中斷可能隨時觸發,我們應該做好處理的準備。

中斷處理程序稱為ISR(中斷服務程序):中斷可能具有不同的優先級,例如,如果觸發了一些低優先級的中斷,則當前正在執行的線程將暫停,并且ISR會獲得控制權。然后,如果觸發了某個高優先級中斷,則當前正在執行的ISR將再次暫停,并為該高優先級中斷運行一個新的ISR。

這樣一來,完成后,控制權將返回到第一個ISR,并且在完成時,也會恢復被中斷的線程。

重要的關鍵代碼:在線程活躍過程中,如果有重要的事情“關鍵的代碼”,在這過程中如果中斷發生,很容易導致意想不到的結果。

這部分關鍵的代碼,我們需要要保護起來,通常我們的做法就是:在之前“關鍵代碼”之前禁用全局中斷,執行完之后,開始全局中斷。

有點需要注意:關閉全局中斷,此時就不會相應中斷,所以,“關鍵代碼”不能太長。

中斷堆棧

在上面說到一點,高優先級中斷搶占低優先中斷,就會出現一個問題:低優先級的代碼需要和線程一樣,用于保存數據的堆棧。

一般有兩種方法:

使用被中斷的線程堆棧;

為中斷使用單獨的堆棧空間;

1.使用被中斷的線程堆棧如果使用被中斷的線程堆棧,就類似如下圖:

9ab70950-4854-11ed-a3b6-dac502259ad0.png

這種情況存在你一個嚴重的問題,你知道是什么嗎?

頻繁中斷,或者中斷較多,線程自身的堆棧空間就會很快被使用完。

每個線程的堆棧都應該包含以下內容:

線程自己的數據;

線程的上下文;

用于執行最壞情況的ISR的數據。

因此,我們就需要換一種方法,為為所有ISR中斷開辟單獨的堆??臻g。

2.為中斷使用單獨的堆棧空間

9ac71b92-4854-11ed-a3b6-dac502259ad0.png

為中斷使用單獨的堆??臻g大致如上圖所示。

好了,本文講述了上面幾種關于搶占,以及相關的內容,你學會了幾點?

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

    關注

    19

    文章

    7536

    瀏覽量

    88638
  • 多線程
    +關注

    關注

    0

    文章

    278

    瀏覽量

    20071
  • RTOS
    +關注

    關注

    22

    文章

    819

    瀏覽量

    119887
  • 內存管理
    +關注

    關注

    0

    文章

    168

    瀏覽量

    14188
  • MMU
    MMU
    +關注

    關注

    0

    文章

    91

    瀏覽量

    18366

原文標題:多線程必須要MMU才行?

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

收藏 人收藏

    評論

    相關推薦

    Java多線程的用法

    本文將介紹一下Java多線程的用法。 基礎介紹 什么是多線程 指的是在一個進程中同時運行多個線程,每個線程都可以獨立執行不同的任務或操作。 與單線程
    的頭像 發表于 09-30 17:07 ?1004次閱讀

    多線程的過程程序

    1、多線程了解線程之前我們必須要先了解(程序—>進程—>線程)的過程程序:是一組計算機能識別和執行的指令,運行于電子計算機上,滿足人們某種需求的信息化工具。進程:進程指正在運行的程序
    發表于 08-24 08:28

    linux多線程編程技術

    (process)中只允許有一個線程,這樣多線程就意味著多進程。現在,多線程技術已經被許多操作系統所支持,包括Windows/NT,當然,也包括Linux。 為什么有了進程的概念后,還要再引入
    發表于 10-24 16:01 ?5次下載

    多線程好還是單線程好?單線程多線程的區別 優缺點分析

    摘要:如今單線程多線程已經得到普遍運用,那么到底多線程好還是單線程好呢?單線程多線程的區別又
    發表于 12-08 09:33 ?8.2w次閱讀

    mfc多線程編程實例及代碼,mfc多線程間通信介紹

    摘要:本文主要以MFC多線程為中心,分別對MFC多線程的實例、MFC多線程之間的通信展開的一系列研究,下面我們來看看原文。
    發表于 12-08 15:23 ?1.8w次閱讀
    mfc<b class='flag-5'>多線程</b>編程實例及代碼,mfc<b class='flag-5'>多線程</b>間通信介紹

    什么是多線程編程?多線程編程基礎知識

    摘要:多線程編程是現代軟件技術中很重要的一個環節。要弄懂多線程,這就要牽涉到多進程。本文主要以多線程編程以及多線程編程相關知識而做出的一些結論。
    發表于 12-08 16:30 ?1.3w次閱讀

    C#多線程技術

    C#和.NET類庫為開發多線程應用程序提供了很方便的支持,本章首先簡要介紹.NET類庫中的Thread類及各種線程支持,再通過示例說明線程使用中需要掌握的規則,最后論述線程同步時出現的
    發表于 04-23 11:32 ?15次下載

    Linux下的多線程編程

    線程呢?使用多線程到底有哪些好處?什么的系統應該選用多線程?我們首先必須回答這些問題?! ∈褂?b class='flag-5'>多線程的理由之一是和進程相比,它是一種非常"節
    發表于 04-02 14:43 ?635次閱讀

    RTOS多線程訪問同一硬件(如UART)的方法

    RTOS多線程(任務)訪問同一硬件(如UART)的方法
    的頭像 發表于 03-12 11:28 ?5522次閱讀

    通過多線程來實現ADC采集功能和OLED顯示功能

    RTOS編程和裸機編程最大的區別就是RTOS可實現多線程管理,這是RTOS的最大優勢。既然跑了操作系統,為何不用多線程實現ADC采集功能和O
    的頭像 發表于 11-20 09:30 ?6148次閱讀
    通過<b class='flag-5'>多線程</b>來實現ADC采集功能和OLED顯示功能

    RT-Thread學習筆記 --(4)RT-Thread多線程學習過程總結

    多線程是實時操作系統里面最重要的知識點之一,要學習RTOS多線程必須(沒錯,是必須)要熟練掌握的內容,只有熟練掌握
    發表于 01-25 18:54 ?0次下載
    RT-Thread學習筆記 --(4)RT-Thread<b class='flag-5'>多線程</b>學習過程總結

    SpringBoot實現多線程

    SpringBoot實現多線程
    的頭像 發表于 01-12 16:59 ?1883次閱讀
    SpringBoot實現<b class='flag-5'>多線程</b>

    什么是線程安全?如何理解線程安全?

    多線程編程中,線程安全是必須要考慮的因素。
    的頭像 發表于 05-30 14:33 ?2141次閱讀
    什么是<b class='flag-5'>線程</b>安全?如何理解<b class='flag-5'>線程</b>安全?

    labview AMC多線程

    labview_AMC多線程
    發表于 08-21 10:31 ?32次下載

    多線程如何保證數據的同步

    多線程編程是一種并發編程的方法,意味著程序中同時運行多個線程,每個線程可獨立執行不同的任務,共享同一份數據。由于多線程并發執行的特點,會引發數據同步的問題,即保證多個
    的頭像 發表于 11-17 14:22 ?1314次閱讀
    大发888娱乐城安装| 太阳城娱乐城88| 盈丰国际博彩网| 金盾百家乐官网网址| 百家乐桩闲落注点| 威尼斯人娱乐城现金开户| 拉斯维加斯娱乐城| 百家乐官网赌博技巧网| 百家乐庄闲的冷热| 大发888hanpa| 澳门百家乐官网的故事| 百家乐输一压二| 大发888心得| 百家乐官网游戏怎么刷钱| 百家乐制胜方法| 现金网游戏| 嬴澳门百家乐官网的公式| 挖掘百家乐赢钱秘籍| 顶级赌场连环夺宝下注有什么窍门| 百家乐官网娱乐城反水| 百家乐娱乐城反水| 棋牌室标语| 百家乐官网怎么做弊| 百家乐永利娱乐| 百家乐官网是骗人的| 罗盘24山度数| 大发888什么赢钱快| 百家乐官网游戏机价格| 百家乐出庄几率| 百家乐破解仪恒达| 乐宝百家乐官网游戏| 百家乐游戏机的玩法| 狮威国际娱乐| 百家乐视频挖坑| 188金宝博开户| 赌百家乐官网2号破解| 大发888大发888| 百家乐官网龙虎扑克牌游戏技巧打| 威尼斯人娱乐城博彩投注平台| 百家乐官网必胜法hk| 百家乐官网优博u2bet|