linux代碼導(dǎo)讀-進程管理_第1頁
linux代碼導(dǎo)讀-進程管理_第2頁
linux代碼導(dǎo)讀-進程管理_第3頁
linux代碼導(dǎo)讀-進程管理_第4頁
linux代碼導(dǎo)讀-進程管理_第5頁
已閱讀5頁,還剩58頁未讀, 繼續(xù)免費閱讀

下載本文檔

版權(quán)說明:本文檔由用戶提供并上傳,收益歸屬內(nèi)容提供方,若內(nèi)容存在侵權(quán),請進行舉報或認(rèn)領(lǐng)

文檔簡介

Linux內(nèi)核源代碼導(dǎo)讀中國科學(xué)技術(shù)大學(xué)計算機系陳香蘭(0551-3606864)xlanchen@/~xlanchen

Spring2009進程(任務(wù)管理)主要內(nèi)容進程描述符進程切換進程的創(chuàng)建和刪除2/2/20233Linux內(nèi)核源代碼導(dǎo)讀進程的概念進程是執(zhí)行程序的一個實例進程和程序的區(qū)別 幾個進程可以并發(fā)的執(zhí)行一個程序一個進程可以順序的執(zhí)行幾個程序2/2/20234Linux內(nèi)核源代碼導(dǎo)讀進程描述符為了管理進程,內(nèi)核必須對每個進程進行清晰的描述。進程描述符提供了內(nèi)核所需了解的進程信息源碼include/linux/sched.h定義

structtask_struct數(shù)據(jù)結(jié)構(gòu)很龐大2/2/20235Linux內(nèi)核源代碼導(dǎo)讀stack2/2/20236Linux內(nèi)核源代碼導(dǎo)讀Linux2.6進程的狀態(tài)簡單過一下,與狀態(tài)相關(guān)的一些宏1)組合狀態(tài)2)狀態(tài)判斷3)狀態(tài)設(shè)置2/2/20237Linux內(nèi)核源代碼導(dǎo)讀進程狀態(tài)轉(zhuǎn)換圖EXIT_ZOMBIE或者EXIT_DEAD或者TASK_DEAD2/2/20238Linux內(nèi)核源代碼導(dǎo)讀標(biāo)識一個進程使用進程描述符地址進程和進程描述符之間有非常嚴(yán)格的一一對應(yīng)關(guān)系,使得用32位進程描述符地址標(biāo)識進程非常方便使用PID(ProcessID,PID)每個進程的PID都存放在進程描述符的pid域中2/2/20239Linux內(nèi)核源代碼導(dǎo)讀進程和進程的內(nèi)核堆棧Linux為每個進程分配一個8KB大小的內(nèi)存區(qū)域,用于存放該進程兩個不同的數(shù)據(jù)結(jié)構(gòu):Thread_info進程的內(nèi)核堆棧進程處于內(nèi)核態(tài)時使用

不同于用戶態(tài)堆棧內(nèi)核控制路徑所用的堆棧

很少,因此對棧和描述符

來說,8KB足夠了Thread_info2/2/202310Linux內(nèi)核源代碼導(dǎo)讀Thread_unionC語言允許用如下的一個union結(jié)構(gòu)來方便的表示這樣的一個混合體2/2/202311Linux內(nèi)核源代碼導(dǎo)讀進程描述符的分配/回收/訪問Thread_info的分配/回收/訪問alloc_thread_infofree_thread_info2/2/202312Linux內(nèi)核源代碼導(dǎo)讀2/2/202313Linux內(nèi)核源代碼導(dǎo)讀從當(dāng)前內(nèi)核堆棧獲得當(dāng)前thread_info根據(jù)thread_info描述符和內(nèi)核態(tài)堆棧之間的配對,內(nèi)核可以很容易的從esp寄存器的值獲得當(dāng)前在CPU上運行的進程的描述符指針因為這個內(nèi)存區(qū)是8KB=213大小,內(nèi)核必須做的就是讓esp有13位的有效位,以獲得thread_info的基地址2/2/202314Linux內(nèi)核源代碼導(dǎo)讀current宏進程描述符2/2/202315Linux內(nèi)核源代碼導(dǎo)讀可知,從每個cpu相關(guān)參數(shù)中,把per_cpu__current_task,取出返回需要找到該參數(shù)的賦值之處??!2/2/202316Linux內(nèi)核源代碼導(dǎo)讀考慮對應(yīng)的x86_write_percpu的使用情況在__switch_to中被調(diào)用2/2/202317Linux內(nèi)核源代碼導(dǎo)讀Current宏的使用Current宏可以看成當(dāng)前進程的進程描述符指針,在內(nèi)核中直接使用比如current->pid返回在CPU上正在執(zhí)行的進程的PID2/2/202318Linux內(nèi)核源代碼導(dǎo)讀進程的PID進程的pid字段Pid最大值2/2/202319Linux內(nèi)核源代碼導(dǎo)讀Pid的管理和分配創(chuàng)建一個進程時,Pid名字空間do_forkcopy_processalloc_pidStructpid的cache2/2/202320Linux內(nèi)核源代碼導(dǎo)讀Pid位圖Pid數(shù)據(jù)結(jié)構(gòu)對pid名字空間:2.6內(nèi)核為PID專門引入了一個數(shù)據(jù)結(jié)構(gòu),Why?

獨立的進程;進程組;sessions

使用pid數(shù)字的注意之處考慮進程的刪除和創(chuàng)建2/2/202321Linux內(nèi)核源代碼導(dǎo)讀最初的pid名字空間在kernel_init中,被修改為init進程在start_kernel中,調(diào)用pidmap_init進行合理的初始化2/2/202322Linux內(nèi)核源代碼導(dǎo)讀分配第一個位圖頁初始化structpid的cache2/2/202323Linux內(nèi)核源代碼導(dǎo)讀2.6內(nèi)核為PID專門引入了一個數(shù)據(jù)結(jié)構(gòu)Why?獨立的進程;進程組;sessions使用pid數(shù)字的注意之處考慮進程的刪除和創(chuàng)建2/2/202324Linux內(nèi)核源代碼導(dǎo)讀閱讀alloc_pid、alloc_pidmap函數(shù)2/2/202325Linux內(nèi)核源代碼導(dǎo)讀進程鏈表為了對給定類型的進程(比如所有在可運行狀態(tài)下的進程)進行有效的搜索,內(nèi)核維護了幾個進程鏈表所有進程鏈表在進程描述符中:2/2/202326Linux內(nèi)核源代碼導(dǎo)讀進程鏈表中的插入和刪除使用常規(guī)list數(shù)據(jù)結(jié)構(gòu)操作2/2/202327Linux內(nèi)核源代碼導(dǎo)讀list_addlist_add_taillist_dellist_movelist_emptylist_for_eachlist_for_each_prevlist_for_each_safelist_for_each_entry…2/2/202328Linux內(nèi)核源代碼導(dǎo)讀例如,在do_fork調(diào)用的copy_process中for_each_process宏掃描整個進程鏈表2/2/202329Linux內(nèi)核源代碼導(dǎo)讀TASK_RUNNING狀態(tài)的進程組織對可運行隊列的一些操作函數(shù)底層:常規(guī)的list數(shù)據(jù)結(jié)構(gòu)操作入列出列等操作:dequeue_taskenqueue_taskconststructsched_class,調(diào)度類rt_sched_classfair_sched_classidle_sched_class每個cpu有一個運行隊列關(guān)于調(diào)度的描述,參見sched_coding.txt和sched-design-CFS.txt2/2/202330Linux內(nèi)核源代碼導(dǎo)讀運行隊列數(shù)據(jù)結(jié)構(gòu)2/2/202331Linux內(nèi)核源代碼導(dǎo)讀structcfs_rq……紅黑樹2/2/202332Linux內(nèi)核源代碼導(dǎo)讀structrt_rq……基于優(yōu)先級的運行隊列2/2/202333Linux內(nèi)核源代碼導(dǎo)讀2/2/202334Linux內(nèi)核源代碼導(dǎo)讀2/2/202335Linux內(nèi)核源代碼導(dǎo)讀調(diào)度類閱讀調(diào)度類sched_class的定義源碼找到主要與運行隊列有關(guān)的enqueue_task、dequeue_taskIdle相關(guān):idle_sched_classnoenqueue/yield_taskforidletasksdequeue_task_idleFair相關(guān)enqueue_task_fairdequeue_task_fairRt相關(guān)enqueue_task_rtdequeue_task_rt2/2/202336Linux內(nèi)核源代碼導(dǎo)讀Idle類特殊2/2/202337Linux內(nèi)核源代碼導(dǎo)讀Fair類進而查看1)enqueue_entity2)__enqueue_entity

(紅黑樹)3)sched_entity結(jié)構(gòu)4)structrq5)structcfs_rqCompletelyFairScheduler完全公平調(diào)度2/2/202338Linux內(nèi)核源代碼導(dǎo)讀Rt類進而查看:1)enqueue_rt_entity2)__enqueue_rt_entity 每個cpu有一個隊列3)sched_rt_entity4)structrq5)structrt_rq6)structrt_prio_array優(yōu)先級隊列2/2/202339Linux內(nèi)核源代碼導(dǎo)讀激活一個任務(wù)activate_task相對的:deactivate_task2/2/202340Linux內(nèi)核源代碼導(dǎo)讀pidhash表及鏈接表在一些情況下,內(nèi)核必須能從進程的PID得出對應(yīng)的進程描述符指針。例如kill系統(tǒng)調(diào)用為了加速查找,引入了pid_hash散列表初始化:pidhash_initTask_struct中:2/2/202341Linux內(nèi)核源代碼導(dǎo)讀pidhash表及鏈接表2/2/202342Linux內(nèi)核源代碼導(dǎo)讀進程之間的親屬關(guān)系程序創(chuàng)建的進程具有父子關(guān)系,在編程時往往需要引用這樣的父子關(guān)系。進程描述符中有幾個域用來表示這樣的關(guān)系2/2/202343Linux內(nèi)核源代碼導(dǎo)讀等待隊列當(dāng)要把除了TASK_RUNNING狀態(tài)之外的進程組織在一起時,linux使用了等待隊列TASK_INTERRUPTIBLE和TASK_UNINTERRUPTIBLE狀態(tài)的進程再分成很多類,每一類對應(yīng)一個特定的事件。在這種情況下,進程狀態(tài)提供的信息滿足不了快速檢索,因此,內(nèi)核引進了另外的進程鏈表,叫做等待隊列等待隊列在內(nèi)核中有很多用途,尤其是對中斷處理、進程同步和定時用處很大2/2/202344Linux內(nèi)核源代碼導(dǎo)讀等待隊列使得進程可以在事件上的條件等待,并且當(dāng)?shù)却臈l件為真時,由內(nèi)核喚醒它們等待隊列由循環(huán)鏈表實現(xiàn)閱讀相關(guān)的宏2/2/202345Linux內(nèi)核源代碼導(dǎo)讀在等待隊列上內(nèi)核實現(xiàn)了一些操作函數(shù)add_wait_queueadd_wait_queue_exclusiveremove_wait_queue2/2/202346Linux內(nèi)核源代碼導(dǎo)讀進程等待等待一個特定事件的進程能調(diào)用下面幾個函數(shù)中的任一個sleep_onsleep_on_timeoutinterruptible_sleep_oninterruptible_sleep_on_timeout進程等待由需要等待的進程自己進行(調(diào)用)2/2/202347Linux內(nèi)核源代碼導(dǎo)讀sleep_on相當(dāng)于閱讀實際的sleep_on代碼2/2/202348Linux內(nèi)核源代碼導(dǎo)讀此外,還可能按照如下方式進行sleep2/2/202349Linux內(nèi)核源代碼導(dǎo)讀例如事件等待wait_event__wait_event等待,直到事件發(fā)生(有效,或…)2/2/202350Linux內(nèi)核源代碼導(dǎo)讀進程的喚醒利用wake_up或者wake_up_interruptible等一系列的宏,都讓插入等待隊列中的進程進入TASK_RUNNING狀態(tài)2/2/202351Linux內(nèi)核源代碼導(dǎo)讀__wake_up__wake_up_common 間接default_wake_functionactivate_tasktry_to_wake_up2/2/202352Linux內(nèi)核源代碼導(dǎo)讀進程切換(processswitching)為了控制進程的執(zhí)行,內(nèi)核必須有能力掛起正在CPU上執(zhí)行的進程,并恢復(fù)以前掛起的某個進程的執(zhí)行,這叫做進程切換,任務(wù)切換,上下文切換2/2/202353Linux內(nèi)核源代碼導(dǎo)讀進程上下文包含了進程執(zhí)行需要的所有信息用戶地址空間

包括程序代碼,數(shù)據(jù),用戶堆棧等控制信息

進程描述符,內(nèi)核堆棧等硬件上下文2/2/202354Linux內(nèi)核源代碼導(dǎo)讀硬件上下文盡管每個進程可以有自己的地址空間,但所有的進程只能共享CPU的寄存器。因此,在恢復(fù)一個進程執(zhí)行之前,內(nèi)核必須確保每個寄存器裝入了掛起進程時的值。這樣才能正確的恢復(fù)一個進程的執(zhí)行硬件上下文:

進程恢復(fù)執(zhí)行前必須裝入寄存器的一組數(shù)據(jù)包括通用寄存器的值以及一些系統(tǒng)寄存器通用寄存器系統(tǒng)寄存器2/2/202355Linux內(nèi)核源代碼導(dǎo)讀在linux中一個進程的上下文主要保存在thread_info,

task_struct的thread_struct中,其他信息放在內(nèi)核態(tài)堆棧中2/2/202356Linux內(nèi)核源代碼導(dǎo)讀thread_info2/2/202357Linux內(nèi)核源代碼導(dǎo)讀Thread_struct……2/2/202358Linux內(nèi)核源代碼導(dǎo)讀Pt_regs2/2/202359Linux內(nèi)核源代碼導(dǎo)讀上下文切換switch_to宏執(zhí)行進程切換,schedule()函數(shù)通過調(diào)用context_switch,間接調(diào)用這個宏一調(diào)度一個新的進程在CPU上運行switch_to利用了prev

溫馨提示

  • 1. 本站所有資源如無特殊說明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請下載最新的WinRAR軟件解壓。
  • 2. 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請聯(lián)系上傳者。文件的所有權(quán)益歸上傳用戶所有。
  • 3. 本站RAR壓縮包中若帶圖紙,網(wǎng)頁內(nèi)容里面會有圖紙預(yù)覽,若沒有圖紙預(yù)覽就沒有圖紙。
  • 4. 未經(jīng)權(quán)益所有人同意不得將文件中的內(nèi)容挪作商業(yè)或盈利用途。
  • 5. 人人文庫網(wǎng)僅提供信息存儲空間,僅對用戶上傳內(nèi)容的表現(xiàn)方式做保護處理,對用戶上傳分享的文檔內(nèi)容本身不做任何修改或編輯,并不能對任何下載內(nèi)容負(fù)責(zé)。
  • 6. 下載文件中如有侵權(quán)或不適當(dāng)內(nèi)容,請與我們聯(lián)系,我們立即糾正。
  • 7. 本站不保證下載資源的準(zhǔn)確性、安全性和完整性, 同時也不承擔(dān)用戶因使用這些下載資源對自己和他人造成任何形式的傷害或損失。

最新文檔

評論

0/150

提交評論