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

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

3天內不再提示

Linux進程調度時機概念分析

汽車玩家 ? 來源:今日頭條 ? 作者:余生做酒長醉不憂 ? 2020-01-23 17:14 ? 次閱讀

Linux在眾多進程中是怎么進行調度的,這個牽涉到Linux進程調度時機的概念,由Linux內核中Schedule()的函數來決定是否要進行進程的切換,如果要切換的話,切換到哪個進程等等。

Linux進程調度時機主要有:

1、進程狀態轉換的時刻:進程終止、進程睡眠;

2、當前進程的時間片用完時(current->counter=0);

3、設備驅動程序

4、進程從中斷、異常及系統調用返回到用戶態時;

時機1,進程要調用sleep()或exit()等函數進行狀態轉換,這些函數會主動調用調度程序進行進程調度;

時機2,由于進程的時間片是由時鐘中斷來更新的,因此,這種情況和時機4是一樣的。

時機3,當設備驅動程序執行長而重復的任務時,直接調用調度程序。在每次反復循環中,驅動程序都檢查need_resched的值,如果必要,則調用調度程序schedule()主動放棄CPU。

時機4,如前所述,不管是從中斷、異常還是系統調用返回,最終都調用ret_from_sys_call(),由這個函數進行調度標志的檢測,如果必要,則調用調用調度程序。那么,為什么從系統調用返回時要調用調度程序呢?這當然是從效率考慮。從系統調用返回意味著要離開內核態而返回到用戶態,而狀態的轉換要花費一定的時間,因此,在返回到用戶態前,系統把在內核態該處理的事全部做完。

對于直接執行調度程序的時機,我們不討論,因為后面我們將會描述調度程序的工作過程。前面我們討論了時鐘中斷,知道了時鐘中斷的重要作用,下面我們就簡單看一下每個時鐘中斷發生時內核要做的工作,首先對這個最頻繁的調度時機有一個大體了解,然后再詳細討論調度程序的具體工作過程。

每個時鐘中斷(timer interrupt)發生時,由三個函數協同工作,共同完成進程的選擇和切換,它們是:schedule()、do_timer()及ret_form_sys_call()。我們先來解釋一下這三個函數:

schedule():進程調度函數,由它來完成進程的選擇(調度);

do_timer():暫且稱之為時鐘函數,該函數在時鐘中斷服務程序中被調用,是時鐘中斷服務程序的主要組成部分,該函數被調用的頻率就是時鐘中斷的頻率即每秒鐘100次(簡稱100赫茲或100Hz);

ret_from_sys_call():系統調用返回函數。當一個系統調用或中斷完成時,該函數被調用,用于處理一些收尾工作,例如信號處理、核心任務等等。

這三個函數是如何協調工作的呢?

前面我們看到,時鐘中斷是一個中斷服務程序,它的主要組成部分就是時鐘函數do_timer(),由這個函數完成系統時間的更新、進程時間片的更新等工作,更新后的進程時間片counter作為調度的主要依據。

在時鐘中斷返回時,要調用函數ret_from_sys_call(),前面我們已經討論過這個函數,在這個函數中有如下幾行:

cmpl $0, _need_resched

jne reschedule

……

restore_all:

RESTORE_ALL


reschedule:

call SYMBOL_NAME(schedule)

jmp ret_from_sys_call

這幾行的意思很明顯:檢測 need_resched 標志,如果此標志為非0,那么就轉到reschedule處調用調度程序schedule()進行進程的選擇。調度程序schedule()會根據具體的標準在運行隊列中選擇下一個應該運行的進程。當從調度程序返回時,如果發現又有調度標志被設置,則又調用調度程序,直到調度標志為0,這時,從調度程序返回時由RESTORE_ALL恢復被選定進程的環境,返回到被選定進程的用戶空間,使之得到運行。

以上就是時鐘中斷這個最頻繁的調度時機。討論這個的主要目的使讀者對時機4有個大致的了解。

另外,TIF_NEED_RESCHED的設置時機 :

設置這個標志的函數主要有兩個: resched_task(),set_tsk_need_resched().主要是resched_task,而resched_task的調用者 check_preempt_curr更是通過:try_to_wake_up/wake_up_new_task/pull_task /__migrate_task 這些被廣泛使用的函數, 從而分布在內核中大量的檢查點有機會搶占進程.

最后要說明的是,系統調用返回函數ret_from_sys_call()是從系統調用、異常及中斷返回函數通常要調用的函數,但并不是非得調用,對于那些要經常被響應的和要被盡快處理的中斷請求信號,為了減少系統開銷,處理完成后并不調用 ret_from_sys_call()(因為很顯然的,從這些中斷處理程序返回到的用戶空間肯定是那個被中斷的進程,無需重新選擇),并且,它們作的工作要盡可能少,因為響應的頻率太高了。

Linux進程調度和其他的UNIX進程調度不同,尤其是在“nice level”優先級的處理上,與優先權調度(priority高的進程最先運行)不同,Linux用的是時間片輪轉調度(Round Robing),但同時又保證了高優先級的進程運行的既快、時間又長(both sooner and longer)。而標準的UNIX調度程序都用到了多級進程隊列。大多數的實現都用到了二級優先隊列:一個標準隊列和一個實時(“real time”)隊列。一般情況下,如果實時隊列中的進程未被阻塞,它們都要在標準隊列中的進程之前被執行,并且,每個隊列中,“nice level”高的進程先被執行。

總體上,Linux 調度序程在交互性方面表現很出色,當然了,這是以犧牲一部分“吞吐量”為代價的。

Linux schedule框架(調度的時刻)

Linux進程調度時機概念分析

1.1、中心是rq(runqueue)

rq其實是runnable queue,即本cpu上所有可運行進程的隊列集合。每個cpu每種類型的rq(cfs/rt)只有一個,一個rq包含多個runnable的task,但是rq當前正在運行的進程(current running task)只有一個。

既然rq是中心,那么以下幾點就是關鍵路徑:

1、什么時候task入rq?

2、什么時候task出rq?

3、rq怎么樣從多個可運行的進程(runnable tasks)中選取一個進程作為當前的運行進程(current running task)?

我們下面就逐一解答這些疑問,理解了這些關鍵路徑,你就對linux的進程調度框架有了一個清晰的認識。

1.2、入rq(enqueue)

只有task新創建/或者task從blocked狀態被喚醒(wakeup),task才會被壓入rq。涉及到進程調度相關的步驟如下:

1、把task壓入rq(enqueue),且把task->state設置為TASK_RUNNING;

2、判斷壓入新task以后rq的負載情況,當前task需不需要被調度出去,如果需要把當前task的thread_info->flags其中TIF_NEED_RESCHED bit置位。

重點在這里:如果當前進程需要重新調度的條件成立,這里只是會設置TIF_NEED_RESCHED標志,并不會馬上調用schedule()來進行調度。真正的調度時機發生在從中斷/異常返回時,會判斷當前進程有沒有被設置TIF_NEED_RESCHED,如果設置則調用schedule()來進行調度。

為什么喚醒涉及到調度不會馬上執行?而是只設置一個TIF_NEED_RESCHED,等到中斷/異常返回的時候才執行?

我理解有幾點:(1)喚醒操作經常在中斷上下文中執行,在這個環境中直接調用schedule()進行調度是不行的;(2)為了維護非搶占內核以來的一些傳統,不要輕易中斷進程的處理邏輯除非他主動放棄;(3)在普通上下文中,喚醒后接著調用schedule()也是可以的,我們看到一些特殊函數就是這么干的(調用smp_send_reschedule()、resched_curr()的函數)。

3、等待中斷/異常的發生、返回,在返回時判讀有TIF_NEED_RESCHED,則調用schedule()進行調度;

1.3、出rq(dequeue)

在當前進程調用系統函數進入blocked狀態是,task會出rq(dequeue)。具體的步驟如下:

1、當前進程把task->state設置為TASK_INTERRUPTIBLE/TASK_UNINTERRUPTIBLE;

2、立即調用schedule()進行調度;

這里block是和wakeup、scheduler_tick最大的不同,block是馬上調用schedule()進行調度,而wakeup、scheduler_tick是設置TIF_NEED_RESCHED標志,等待中斷/異常返回時才執行真正的schedule()操作;

3、調用schedule()后,判斷當前進程task->state已經非TASK_RUNNING,則進行dequeue操作,并且調度其他進程到rq->curr。

1.4、定時調度rq(scheduler_tick)

前面說了在rq的enqueue、dequeue時刻會計算rq負載,來決定把哪個runnable task放到current running task。除了enqueue/dequeue時候,系統還會周期性的計算rq負載來進行調度,確保多進程在1個cpu上都能得到服務。具體的步驟如下:

1、每1 tick,local timer產生一次中斷。中斷中調用scheduler_tick(),計算rq的負載重新調度;

2、如果當前進程需要被調度,則設置TIF_NEED_RESCHED標志;

3、在local timer中斷返回的時候,時判讀有TIF_NEED_RESCHED,則調用schedule()進行調度;

1.5、中斷/異常返回(Interrupt/Exception)

在前面幾節中有一個重要的概念,wakeup、scheduler_tick操作后,如果需要調度只會設置TIF_NEED_RESCHED,在中斷/異常返回時才執行真正的調度schedule()操作;

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

    關注

    87

    文章

    11022

    瀏覽量

    207059
  • 進程
    +關注

    關注

    0

    文章

    195

    瀏覽量

    13902
收藏 人收藏

    評論

    相關推薦

    Linux內核進程管理與調度:策略優化與實踐分析

    今天給大家上點硬貨,關于Linux進程管理和調度是學習和理解Linux的必學知識。為協調多個進程 "同時" 運行,現代操作系統通常使用
    發表于 05-08 09:42 ?649次閱讀
    <b class='flag-5'>Linux</b>內核<b class='flag-5'>進程</b>管理與<b class='flag-5'>調度</b>:策略優化與實踐<b class='flag-5'>分析</b>

    Linux內核搶占和用戶搶占的概念和區別

    本文詳解了Linux內核搶占實現機制。首先介紹了內核搶占和用戶搶占的概念和區別,接著分析了不可搶占內核的特點及實時系統中實現內核搶占的必要性。然后分析了禁止內核搶占的情況和內核搶占的
    發表于 08-05 08:18

    干貨分享:基于嵌入式Linux進程調度實現方法

    調度策略,實現了高效、靈活的進程調度。 2.Linux 進程調度
    發表于 12-10 14:17

    Linux進程、線程以及調度

    報名:《Linux進程、線程以及調度》4節系列微課(522-25)
    發表于 05-15 14:44

    Linux進程管理

    Linux進程管理 本章主要介紹進程概念、狀態、構成以及Linux進程的相關知識。 掌握
    發表于 04-28 14:57 ?0次下載

    linux處理機調度與死鎖

    linux處理機調度與死鎖 掌握處理機的三級調度 掌握作業調度進程調度
    發表于 04-28 14:59 ?0次下載

    Linux 2.6進程調度

    分析了與Linux 2.6 進程調度密切相關的一些重要數據結構,詳細描述了進程調度
    發表于 06-13 10:13 ?11次下載

    Li nux與VxWorks任務調度機制分析

    分析Linux和VxWorks兩種多任務操作系統任務調度機制的異同,從任務控制塊、調度時機、調度
    發表于 11-13 17:54 ?10次下載

    Linux進程調度的原理解析

    進程調度依據 調度程序運行時,要在所有可運行狀態的進程中選擇最值得運行的進程投入運行。選擇進程
    發表于 11-02 11:01 ?1次下載

    uClinux進程調度器的實現分析

    分享到:標簽:uClinux 調度策略 進程調度器 摘要:針對操作系統中進程調度機制,依次對其調度
    發表于 11-06 14:30 ?0次下載

    Linux 進程調度淺析

    都比較低,但是linux作為一個通用操作系統,不能假設系統負載低,必須為應付高負載下的進程調度做精心的設計。當然,這些設計對于低負載(且沒有什么實時性要求)的環境,沒多大用。極端情況下,如果CPU
    發表于 04-02 14:40 ?267次閱讀

    嵌入式Linux實時進程調度算法改進

    ,具有實時性的同時又具有嵌入 式系統的特點。2 實時進程調度算法分析2.1 Linux進程調度
    發表于 04-02 14:43 ?329次閱讀

    linux進程調度淺析

    (如桌面系統、網絡服務器、等)負載都比較低,但是linux作為一個通用操作系統,不能假設系統負載低,必須為應付高負載下的進程調度做精心的設計。當然,這些設計對于低負載(且沒有什么實時性要求)的環境,沒多大
    發表于 04-02 14:45 ?282次閱讀

    Linux進程概念說明

    進程Linux 操作系統中最重要的基本概念之一,這一節我們將了解學習 Linux 進程的一些基礎知識。
    發表于 07-14 14:27 ?660次閱讀

    帶大家看看Linux內核如何調度進程

    部分,打開調度器的黑匣子,來看看Linux內核如何調度進程的。實際上,進程調度器主要做兩件事:選
    的頭像 發表于 07-26 15:14 ?1834次閱讀
    亚洲欧美日韩精品久久_久久精品AⅤ无码中文_日本中文字幕有码在线播放_亚洲视频高清不卡在线观看
    <acronym id="s8ci2"><small id="s8ci2"></small></acronym>
    <rt id="s8ci2"></rt><rt id="s8ci2"><optgroup id="s8ci2"></optgroup></rt>
    <acronym id="s8ci2"></acronym>
    <acronym id="s8ci2"><center id="s8ci2"></center></acronym>