0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

RTOS多線程必須要MMU才行?

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

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

MMU:Memory Management Unit,即內(nèi)存管理單元,它是一個“好東西”,在大型的操作系統(tǒng)中,起到了關(guān)鍵的作用。

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

只是RTOS實現(xiàn)的多線程相對大型操作系統(tǒng)要簡單一點,其原理也要簡單的多。

下面就來圍繞RTOS給大家大家關(guān)于RTOS的多線程的內(nèi)容。

多線程

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

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

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

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

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

例如,計算兩個數(shù)字的總和時:

//假設(shè)我們有兩個整數(shù):a和b

int c = a + b ;

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

# MIPS反匯編

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

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

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

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

你會發(fā)現(xiàn)上面執(zhí)行了4個動作,但是搶占式操作系統(tǒng)可以在任何時候搶占另一個線程,包括在這4個動作之間。

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

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

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

大概意思就是:當(dāng)我們需要從一個線程切換到另一個線程時,內(nèi)核獲得控制權(quán),執(zhí)行必要的內(nèi)務(wù)處理(至少要保存和恢復(fù)寄存器值),然后將控制權(quán)轉(zhuǎn)移到下一個線程以運行。

線程的堆棧

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

在有MMU的操作系統(tǒng)中,(用戶的)線程堆??梢园葱鑴討B(tài)增長:線程需要的堆??臻g越多,線程堆棧就越多(如果內(nèi)核允許)。

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

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

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

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

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

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

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

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

當(dāng)內(nèi)核決定將控制權(quán)切換到線程B時,它將執(zhí)行以下操作:

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

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

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

此時,你會看到:

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

中斷(ISR)搶占

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

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

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

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

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

重要的關(guān)鍵代碼:在線程活躍過程中,如果有重要的事情“關(guān)鍵的代碼”,在這過程中如果中斷發(fā)生,很容易導(dǎo)致意想不到的結(jié)果。

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

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

中斷堆棧

在上面說到一點,高優(yōu)先級中斷搶占低優(yōu)先中斷,就會出現(xiàn)一個問題:低優(yōu)先級的代碼需要和線程一樣,用于保存數(shù)據(jù)的堆棧。

一般有兩種方法:

使用被中斷的線程堆棧;

為中斷使用單獨的堆??臻g;

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

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

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

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

每個線程的堆棧都應(yīng)該包含以下內(nèi)容:

線程自己的數(shù)據(jù);

線程的上下文;

用于執(zhí)行最壞情況的ISR的數(shù)據(jù)。

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

2.為中斷使用單獨的堆??臻g

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

為中斷使用單獨的堆棧空間大致如上圖所示。

好了,本文講述了上面幾種關(guān)于搶占,以及相關(guān)的內(nèi)容,你學(xué)會了幾點?

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報投訴
  • 計算機
    +關(guān)注

    關(guān)注

    19

    文章

    7494

    瀏覽量

    87961
  • 多線程
    +關(guān)注

    關(guān)注

    0

    文章

    278

    瀏覽量

    19961
  • RTOS
    +關(guān)注

    關(guān)注

    22

    文章

    813

    瀏覽量

    119643
  • 內(nèi)存管理
    +關(guān)注

    關(guān)注

    0

    文章

    168

    瀏覽量

    14141
  • MMU
    MMU
    +關(guān)注

    關(guān)注

    0

    文章

    91

    瀏覽量

    18291

原文標(biāo)題:多線程必須要MMU才行?

文章出處:【微信號:strongerHuang,微信公眾號:strongerHuang】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。

收藏 人收藏

    評論

    相關(guān)推薦

    Java多線程的用法

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

    多線程的過程程序

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

    linux多線程編程技術(shù)

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

    多線程好還是單線程好?單線程多線程的區(qū)別 優(yōu)缺點分析

    摘要:如今單線程多線程已經(jīng)得到普遍運用,那么到底多線程好還是單線程好呢?單線程多線程的區(qū)別又
    發(fā)表于 12-08 09:33 ?8.1w次閱讀

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

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

    什么是多線程編程?多線程編程基礎(chǔ)知識

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

    C#多線程技術(shù)

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

    Linux下的多線程編程

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

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

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

    通過多線程來實現(xiàn)ADC采集功能和OLED顯示功能

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

    RT-Thread學(xué)習(xí)筆記 --(4)RT-Thread多線程學(xué)習(xí)過程總結(jié)

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

    SpringBoot實現(xiàn)多線程

    SpringBoot實現(xiàn)多線程
    的頭像 發(fā)表于 01-12 16:59 ?1831次閱讀
    SpringBoot實現(xiàn)<b class='flag-5'>多線程</b>

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

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

    labview AMC多線程

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

    多線程如何保證數(shù)據(jù)的同步

    多線程編程是一種并發(fā)編程的方法,意味著程序中同時運行多個線程,每個線程可獨立執(zhí)行不同的任務(wù),共享同一份數(shù)據(jù)。由于多線程并發(fā)執(zhí)行的特點,會引發(fā)數(shù)據(jù)同步的問題,即保證多個
    的頭像 發(fā)表于 11-17 14:22 ?1237次閱讀