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

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

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

哪些關(guān)閉了Linux搶占?搶占又關(guān)閉了誰?

Linux閱碼場(chǎng) ? 來源:未知 ? 作者:胡薇 ? 2018-05-04 08:57 ? 次閱讀

本文對(duì)比分析:

preempt_disable()

local_irq_disable()/local_irq_save(flags)

spin_lock()

spin_lock_irq()/spin_lock_irqsave(lock, flags)

哪些關(guān)閉了搶占?另外,再說清楚,搶占又關(guān)閉了誰。

首先,把這幾個(gè)API的關(guān)系圖再勾勒一下。

我們理解,spin_lock()會(huì)調(diào)用preempt_disable() 導(dǎo)致本核的搶占調(diào)度被關(guān)閉(preempt_disable函數(shù)實(shí)際增加preempt_count來達(dá)到此效果),其次我們理解spin_lock_irq()是local_irq_disable()+preempt_disable()的合體。

下面一幅圖描述了幾個(gè)函數(shù)之間的包含關(guān)系如下:

local_irq_disable()/local_irq_save()的disable和save版的唯一區(qū)別是,要不要保存CPU對(duì)中斷的屏蔽狀態(tài)。

spin_lock_irq()/spin_lock_irqsave(lock, flags)的唯一區(qū)別是,要不要保存CPU對(duì)中斷的屏蔽狀態(tài)。

是誰殺了搶占?

Kernel的代碼明確顯示,執(zhí)行搶占調(diào)度的時(shí)候,會(huì)同時(shí)檢測(cè)“non-zero preempt_count or interrupts are disabled”:

我們可以進(jìn)一步展開preemptible():

對(duì)于ARM處理器而言,判斷irqs_disabled(),其實(shí)就是判斷CPSR中的IRQMASK_I_BIT是否被設(shè)置。

所以,我們得出一個(gè)結(jié)論,前言這一節(jié)里面,列出的所有函數(shù),都能關(guān)閉本核的搶占調(diào)度。因?yàn)?,無論是preempt_count計(jì)數(shù)狀態(tài),還是中斷被關(guān)閉,都會(huì)導(dǎo)致kernel認(rèn)為無法搶占!

通殺邏輯如下:

殺手的差異在哪里?

既然都關(guān)閉了搶占,那么區(qū)別在哪里呢?

我們看兩段代碼,假設(shè)下面的代碼都發(fā)生在NICE為0的普通進(jìn)程:

preempt_disable()

xxx(1)

preempt_enable()

local_irq_disable()

xxx(2)

local_irq_enable()

首先,xxx(1)和xxx(2)里面,都是不可以搶占的。一個(gè)搞定了preempt_count,一個(gè)搞定了中斷。

但是假設(shè)xxx(1)內(nèi)喚醒了一個(gè)高優(yōu)先級(jí)的RT任務(wù),那么在preempt_enable()的時(shí)刻,直接就是一個(gè)搶占點(diǎn),這個(gè)時(shí)候,發(fā)生schedule,高優(yōu)先級(jí)RT任務(wù)進(jìn)來跑;假設(shè)xxx(2)內(nèi)喚醒了一個(gè)高優(yōu)先級(jí)的RT任務(wù),那么在local_irq_enable()的時(shí)刻,不是一個(gè)搶占點(diǎn),高優(yōu)先級(jí)RT的任務(wù)必須等待下一個(gè)搶占點(diǎn)。下一個(gè)搶占點(diǎn),可能是時(shí)鐘tick處理返回、中斷返回、軟中斷結(jié)束、yield()等等多種情況。

在preempt_enable()中,會(huì)執(zhí)行一次preempt_schedule():

而local_irq_enable()只是單純的開啟CPU對(duì)中斷的響應(yīng),對(duì)于ARM而言,它就是:

再來看大boss,spin_lock_irq是同時(shí)調(diào)用了preempt_disable和local_irq_disable:

而對(duì)應(yīng)的spin_unlock_irq()則同時(shí)調(diào)用了local_irq_enable()和preempt_enable():

大家想一想,為何preempt_enable()比local_irq_enable()后發(fā)生呢?如果代碼順序是這樣的:

preempt_enable()

local_irq_enable()

第一句preempt_enable()想執(zhí)行搶占調(diào)度的話,即便調(diào)用了preempt_schedule(),但是由于IRQ還是關(guān)門的,preempt_schedule()函數(shù)會(huì)立即返回(詳見《是誰殺了搶占?》一節(jié)),所以無法搶占;后一句local_irq_enable()不會(huì)執(zhí)行搶占調(diào)度。所以,如果這么干的話,

spin_lock_irq()

xxx(3)

spin_unlock_irq()

如果xxx(3)喚醒了高優(yōu)先級(jí)的RT,在spin_unlock_irq()的時(shí)刻,將無法直接搶占!

還好,真正的順序是:

local_irq_enable()

preempt_enable()

所以,在spin_unlock_irq()的時(shí)刻,RT進(jìn)程就換入執(zhí)行了。

看小一點(diǎn)的boss,spin_lock():

spin_lock()

xxx(4)

spin_unlock()

如果xxx(4)喚醒了RT進(jìn)程,在spin_unlock()的時(shí)刻,會(huì)立即搶入。因?yàn)閟pin_unlock()會(huì)調(diào)用preempt_enable()。

而搶占又殺了誰?

理論上,關(guān)搶占,并沒有徹底的關(guān)閉調(diào)度器,因?yàn)檫M(jìn)程還是可以主動(dòng)地sleep:

上述代碼,在spin_lock的區(qū)間里面,調(diào)用了msleep(),這個(gè)時(shí)候,不屬于搶占,Linux還是會(huì)pick下一個(gè)task來跑。

不過這樣的代碼,一般在后期蘊(yùn)藏著巨大的風(fēng)險(xiǎn),導(dǎo)致后期的莫名崩潰。所以呢,實(shí)際的工程里面,我們是嚴(yán)格地禁止的。

建議大家打開Kernel里面的config里面的DEBUG_ATOMIC_SLEEP,一旦出現(xiàn)這種情況,讓kernel去匯報(bào)錯(cuò)誤。

這種情況下,kernel檢測(cè)到有人在atomic上下文里面執(zhí)行可能睡眠的行為,會(huì)直接報(bào)執(zhí)行的棧回溯。

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

    關(guān)注

    68

    文章

    10899

    瀏覽量

    212617
  • Linux
    +關(guān)注

    關(guān)注

    87

    文章

    11336

    瀏覽量

    210100
  • API
    API
    +關(guān)注

    關(guān)注

    2

    文章

    1509

    瀏覽量

    62263

原文標(biāo)題:宋寶華: 是誰關(guān)閉了Linux搶占,而搶占又關(guān)閉了誰?

文章出處:【微信號(hào):LinuxDev,微信公眾號(hào):Linux閱碼場(chǎng)】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。

收藏 人收藏

    評(píng)論

    相關(guān)推薦

    gd32vf103根據(jù)例子配置systick和adc中斷,但是adc中斷無法搶占systick是為什么?

    根據(jù)例子配置systick和adc中斷,但是adc中斷無法搶占systick 配置為向量和非相量都沒有實(shí)現(xiàn)
    發(fā)表于 01-10 07:43

    MPLS LSP隧道搶占

    MPLS LSP隧道搶占
    發(fā)表于 05-30 15:26

    Linux內(nèi)核搶占和用戶搶占的概念和區(qū)別

    本文詳解了Linux內(nèi)核搶占實(shí)現(xiàn)機(jī)制。首先介紹內(nèi)核搶占和用戶搶占的概念和區(qū)別,接著分析不可
    發(fā)表于 08-05 08:18

    詳解Linux內(nèi)核搶占實(shí)現(xiàn)機(jī)制

    本文詳解了Linux內(nèi)核搶占實(shí)現(xiàn)機(jī)制。首先介紹內(nèi)核搶占和用戶搶占的概念和區(qū)別,接著分析不可
    發(fā)表于 08-06 06:16

    嵌入式Linux實(shí)時(shí)化技術(shù)詳談

    Linux內(nèi)核設(shè)計(jì)中,中斷可以搶占最高優(yōu)先級(jí)的任務(wù),使高優(yōu)先級(jí)任務(wù)被阻塞的最長(zhǎng)時(shí)間不確定。而且,由于內(nèi)核為保護(hù)臨界區(qū)需要關(guān)閉中斷,更加增長(zhǎng)了高優(yōu)先級(jí)任務(wù)阻塞時(shí)間?!  ?時(shí)鐘精度  Linux
    發(fā)表于 08-03 07:00

    蘋果更新,教你關(guān)閉iPhone手機(jī)系統(tǒng)更新

    近期,蘋果更新10.2的系統(tǒng),對(duì)部分粉絲來說是好事兒,但是有很多蘋果手機(jī)用戶并不想更新系統(tǒng),但是總是會(huì)有一個(gè)煩人的小紅點(diǎn)提醒。那么,如何關(guān)閉蘋果手機(jī)的系統(tǒng)更新呢?
    發(fā)表于 12-19 14:22 ?3.9w次閱讀

    谷歌針對(duì)位置記錄問題更新 可以徹底關(guān)閉

    。 對(duì)此谷歌近日針對(duì)這一問題進(jìn)行了更新,更新內(nèi)容顯示,如果位置記錄項(xiàng)目被關(guān)閉,那么位置信息就不再會(huì)得到保存。
    的頭像 發(fā)表于 08-25 10:04 ?4300次閱讀

    Linux開放端口和關(guān)閉端口的方法

    Linux開放端口和關(guān)閉端口的方法如下
    發(fā)表于 05-18 09:14 ?1w次閱讀
    <b class='flag-5'>Linux</b>開放端口和<b class='flag-5'>關(guān)閉</b>端口的方法

    linux的端口怎么關(guān)閉

    怎么關(guān)閉linux的端口?下面本篇文章給大家介紹一下linux下查看端口是否打開、關(guān)閉/打開端口的方法。有一定的參考價(jià)值,有需要的朋友可以參考一下,希望對(duì)大家有所幫助。
    發(fā)表于 05-29 09:27 ?4102次閱讀
    <b class='flag-5'>linux</b>的端口怎么<b class='flag-5'>關(guān)閉</b>

    Linux閱碼場(chǎng)原創(chuàng)精華文章分享,你值得擁有

    技術(shù)的前世今生之前世 郭健:Linux進(jìn)程調(diào)度技術(shù)的前世今生之今生 宋寶華:是關(guān)閉Linux搶占
    的頭像 發(fā)表于 10-10 16:28 ?2780次閱讀

    Linux中內(nèi)核搶占相關(guān)的基礎(chǔ)知識(shí)

    今天要分享的是搶占相關(guān)的基礎(chǔ)知識(shí)。本文以內(nèi)核搶占為引子,概述一下 Linux 搶占的圖景。我盡量避開細(xì)節(jié)問題和源碼分析。 什么是內(nèi)核搶占?
    的頭像 發(fā)表于 11-09 16:48 ?2075次閱讀

    關(guān)閉Linux搶占?

    我們理解,spin_lock()會(huì)調(diào)用preempt_disable() 導(dǎo)致本核的搶占調(diào)度被關(guān)閉(preempt_disable函數(shù)實(shí)際增加preempt_count來達(dá)到此效果),其次我們理解spin_lock_irq()是local_irq_disable()+pr
    發(fā)表于 08-07 17:19 ?901次閱讀
    是<b class='flag-5'>誰</b><b class='flag-5'>關(guān)閉</b><b class='flag-5'>了</b><b class='flag-5'>Linux</b><b class='flag-5'>搶占</b>?

    Linux下如何開啟、關(guān)閉端口

    查看防火墻狀態(tài) 在Linux控制臺(tái)輸入:firewall-cmd --state 此時(shí)控制臺(tái)返回:not running表示防火墻處于關(guān)閉狀態(tài)
    的頭像 發(fā)表于 05-12 14:54 ?5688次閱讀
    <b class='flag-5'>Linux</b>下如何開啟、<b class='flag-5'>關(guān)閉</b>端口

    操作系統(tǒng)中搶占式和非搶占式內(nèi)核的區(qū)別

    操作系統(tǒng)一般分為搶占式內(nèi)核和非搶占式內(nèi)核,通常RTOS都是搶占式內(nèi)核。你知道搶占式內(nèi)核和非搶占式內(nèi)核的區(qū)別嗎?
    的頭像 發(fā)表于 05-29 10:47 ?1998次閱讀
    操作系統(tǒng)中<b class='flag-5'>搶占</b>式和非<b class='flag-5'>搶占</b>式內(nèi)核的區(qū)別

    linux關(guān)閉docker的命令

    Linux 系統(tǒng)中,關(guān)閉 Docker 的操作可以通過以下多種方式進(jìn)行。本文將詳細(xì)講解每一種方式,并提供示例代碼和命令,以幫助讀者更好地理解和實(shí)踐。 使用 docker 命令 最常用的方法
    的頭像 發(fā)表于 11-23 09:39 ?2908次閱讀