{eval=Array;=+count(Array);}

国产xxxx99真实实拍_久久不雅视频_高清韩国a级特黄毛片_嗯老师别我我受不了了小说

問(wèn)答專(zhuān)欄Q & A COLUMN

Linux系統(tǒng)是如何創(chuàng)建進(jìn)程的?聽(tīng)說(shuō)這一過(guò)程是用C語(yǔ)言clone函數(shù)實(shí)現(xiàn)的?

linkFlylinkFly 回答0 收藏1
收藏問(wèn)題

2條回答

suosuopuo

suosuopuo

回答于2022-06-27 17:25

謝邀。

我之前兩節(jié)文章簡(jiǎn)要地從C語(yǔ)言源代碼層面討論了Linux系統(tǒng)中進(jìn)程的基本概念,我們知道了Linux內(nèi)核如何描述和記錄進(jìn)程的資源,以及進(jìn)程的五種基本狀態(tài)和進(jìn)程的家族樹(shù)。事實(shí)上,就進(jìn)程管理而言,Linux還是有一些獨(dú)特之處的。

Linux 系統(tǒng)中的進(jìn)程創(chuàng)建

許多操作系統(tǒng)都提供了專(zhuān)門(mén)的進(jìn)程產(chǎn)生機(jī)制,比較典型的過(guò)程是:首先在內(nèi)存新的地址空間里創(chuàng)建進(jìn)程,然后讀取可執(zhí)行程序,裝載到內(nèi)存中執(zhí)行。

Linux 系統(tǒng)創(chuàng)建線(xiàn)程并未使用上述經(jīng)典過(guò)程,而是將創(chuàng)建過(guò)程拆分到兩組獨(dú)立的函數(shù)中執(zhí)行:fork() 函數(shù)和 exec() 函數(shù)族。

基本流程是這樣的:首先,fork() 函數(shù)拷貝當(dāng)前進(jìn)程創(chuàng)建子進(jìn)程。產(chǎn)生的子進(jìn)程與父進(jìn)程的區(qū)別僅在與 PID 與 PPID 以及某些資源和統(tǒng)計(jì)量,例如掛起的信號(hào)等。準(zhǔn)備好進(jìn)程運(yùn)行的地址空間后,exec() 函數(shù)族負(fù)責(zé)讀取可執(zhí)行程序,并將其加載到相應(yīng)的位置開(kāi)始執(zhí)行。

Linux 系統(tǒng)創(chuàng)建進(jìn)程使用的這兩組函數(shù)效果與其他操作系統(tǒng)的經(jīng)典進(jìn)程創(chuàng)建方式效果是相似的,可能有讀者會(huì)覺(jué)得這么做會(huì)讓進(jìn)程創(chuàng)建過(guò)于繁瑣,其實(shí)不是的,Linux 這么做的其中一個(gè)原因是為了提高代碼的復(fù)用率,這得益于 Linux 高度概括的抽象,無(wú)需再額外設(shè)計(jì)一套機(jī)制用于創(chuàng)建進(jìn)程。

“寫(xiě)時(shí)拷貝”

早期 Linux 中的 fork() 函數(shù)直接把父進(jìn)程的所有資源賦值給創(chuàng)建出的子進(jìn)程,這樣的機(jī)制自然是簡(jiǎn)單的,但是效率卻比較低下。

原因是顯而易見(jiàn)的:子進(jìn)程并不一定要使用父進(jìn)程的資源,或者子進(jìn)程可能僅需以只讀的方式訪(fǎng)問(wèn)父進(jìn)程的資源,這時(shí)“拷貝一份資源”就純屬多余的開(kāi)銷(xiāo)了。

針對(duì)這樣的問(wèn)題,Linux 后續(xù)版本中的 fork() 函數(shù)開(kāi)始采用“寫(xiě)時(shí)拷貝”機(jī)制。寫(xiě)時(shí)拷貝技術(shù)可以將拷貝需求延遲,甚至免除拷貝,減小開(kāi)銷(xiāo)。

具體來(lái)說(shuō)就是,Linux 在調(diào)用 fork() 創(chuàng)建子進(jìn)程時(shí),并不著急拷貝整個(gè)進(jìn)程地址空間,而是暫時(shí)讓父子進(jìn)程以只讀的方式共享同一個(gè)拷貝。拷貝動(dòng)作只在子進(jìn)程需要寫(xiě)入時(shí)才會(huì)發(fā)生,以確保各個(gè)進(jìn)程有自己獨(dú)立的內(nèi)存空間。

如果子進(jìn)程用不到或者只需要讀取共享空間數(shù)據(jù),那么拷貝動(dòng)作就被省去了,Linux 就減小了開(kāi)銷(xiāo)。例如,系統(tǒng)調(diào)用 fork() 后立即調(diào)用 exec(),此時(shí) exec() 會(huì)加載新的映像覆蓋 fork() 的地址空間,拷貝動(dòng)作完全可以省去。

事實(shí)上,fork() 函數(shù)的實(shí)際開(kāi)銷(xiāo)就是復(fù)制父進(jìn)程的頁(yè)表以及給子進(jìn)程創(chuàng)建唯一的進(jìn)程描述符。在大多數(shù)情況下,Linux 創(chuàng)建進(jìn)程后都會(huì)馬上運(yùn)行新的可執(zhí)行程序,因此“寫(xiě)時(shí)拷貝”機(jī)制可以避免相當(dāng)多的數(shù)據(jù)拷貝。創(chuàng)建進(jìn)程速度快是 Linux 系統(tǒng)的一個(gè)特征,因此“寫(xiě)時(shí)拷貝”是一種相當(dāng)重要的優(yōu)化。

創(chuàng)建進(jìn)程時(shí),內(nèi)存地址空間里常常包含數(shù)十 MB 的數(shù)據(jù),如果每創(chuàng)建一次進(jìn)程,就拷貝一次數(shù)據(jù),開(kāi)銷(xiāo)顯然是非常大的。

fork() 函數(shù)

Linux 中的 fork() 函數(shù)其實(shí)是基于 clone() 實(shí)現(xiàn)的,clone() 函數(shù)可以通過(guò)一系列參數(shù)標(biāo)志指定父子進(jìn)程需要共享的資源,在 Linux 中輸入 man 命令可以查看 clone() 函數(shù)的C語(yǔ)言原型,以及相關(guān)的參數(shù)標(biāo)志:在Linux中,fork() 函數(shù)最終調(diào)用了 do_fork() 函數(shù),它的C語(yǔ)言代碼如下,請(qǐng)看(do_fork() 函數(shù)的C語(yǔ)言代碼比較長(zhǎng),下面面只列出了一部分):

do_fork() 函數(shù)完成了進(jìn)程創(chuàng)建的大部分工作,從相關(guān)的C語(yǔ)言源代碼可以看出,它調(diào)用了 copy_process() 函數(shù),copy_process() 函數(shù)的C語(yǔ)言源代碼如下,請(qǐng)看:copy_process() 函數(shù)的代碼也是比較長(zhǎng)的,在我手上的Linux系統(tǒng)中,達(dá)到了近 400 行,不過(guò)代碼的整體邏輯是清晰的:

(1)copy_process() 函數(shù)首先檢查了一些標(biāo)志位,接著調(diào)用 dup_task_struct() 函數(shù)為新進(jìn)程創(chuàng)建內(nèi)核棧,以及上一節(jié)提到的 thread_info 和 task_struct 結(jié)構(gòu):

創(chuàng)建后,接下來(lái)的 arch_dup_task_struct() 函數(shù)會(huì)將 orig 結(jié)構(gòu)拷貝給新創(chuàng)建的結(jié)構(gòu),查看相關(guān)C語(yǔ)言代碼,這一過(guò)程是清晰的:此時(shí)子進(jìn)程和父進(jìn)程的描述符是完全相同的。

(2)接下來(lái),需要檢查一些標(biāo)志位和統(tǒng)計(jì)信息,相關(guān)的C語(yǔ)言代碼如下,請(qǐng)看:

(3)將一些統(tǒng)計(jì)量清零,以及初始化一些區(qū)別成員,此時(shí)雖然新進(jìn)程的 task_struct 結(jié)構(gòu)體大多成員未被修改,但是父子進(jìn)程已經(jīng)有所區(qū)別。這一過(guò)程的相關(guān)C語(yǔ)言代碼片段如下,請(qǐng)看:

(4)將新創(chuàng)建的子進(jìn)程狀態(tài)設(shè)置為 TASK_UNINTERRUUPTIBLE,確保其暫時(shí)不會(huì)被投入運(yùn)行,這一過(guò)程的C語(yǔ)言代碼相對(duì)簡(jiǎn)單。(5)調(diào)用 alloc_pid() 函數(shù)為新進(jìn)程分配一個(gè)獨(dú)一無(wú)二的 pid,相關(guān)C語(yǔ)言代碼如下,請(qǐng)看:

(6)根據(jù) clone() 函數(shù)的參數(shù)標(biāo)志位,拷貝或共享已經(jīng)打開(kāi)的文件、文件系統(tǒng)、信號(hào)處理函數(shù)、進(jìn)程地址空間等資源,例如下面這段C語(yǔ)言代碼:(7)將為新進(jìn)程創(chuàng)建的 task_struct 結(jié)構(gòu)體的指針?lè)祷亟o調(diào)用者,也即 do_fork() 函數(shù)。此時(shí)新創(chuàng)建的進(jìn)程還沒(méi)有被投入運(yùn)行。

現(xiàn)在回到 do_fork() 函數(shù)。如果調(diào)用 clone() 函數(shù)時(shí),沒(méi)有傳遞 CLONE_STOPPED 參數(shù),新創(chuàng)建的進(jìn)程將被喚醒,并投入運(yùn)行,這一過(guò)程的C語(yǔ)言代碼如下:到這里,一個(gè)新的進(jìn)程就被 Linux 創(chuàng)建完畢了。

Linux 內(nèi)核有意讓新創(chuàng)建的子進(jìn)程先運(yùn)行,因?yàn)樽舆M(jìn)程常常會(huì)立即調(diào)用 exec() 函數(shù)加載新的程序到內(nèi)存中運(yùn)行,這樣就避免了寫(xiě)時(shí)拷貝的額外開(kāi)銷(xiāo)。如果父進(jìn)程首先執(zhí)行,顯然極有可能開(kāi)始往地址空間寫(xiě)入操作,導(dǎo)致拷貝動(dòng)作發(fā)生。

小結(jié)

本節(jié)詳細(xì)的從C語(yǔ)言代碼層面分析了Linux內(nèi)核創(chuàng)建進(jìn)程的過(guò)程,可見(jiàn),即使是復(fù)雜的操作系統(tǒng)代碼,也是通過(guò)一系列基本C語(yǔ)言語(yǔ)法和函數(shù)實(shí)現(xiàn)的。那么,Linux 是如何創(chuàng)建線(xiàn)程的呢?之前我們?cè)?jīng)提到,Linux 系統(tǒng)并不特別區(qū)分進(jìn)程和線(xiàn)程,線(xiàn)程其實(shí)是一種特殊的進(jìn)程,Linux 是如何實(shí)現(xiàn)這一“特殊”過(guò)程的呢?敬請(qǐng)關(guān)注。

評(píng)論0 贊同0
  •  加載中...
Chaz

Chaz

回答于2022-06-27 17:25

這跟execvp函數(shù)的實(shí)現(xiàn)方式有關(guān):

int execvp(const char *file ,char * const argv []);


execvp()會(huì)從PATH 環(huán)境變量所指的目錄中查找符合參數(shù)file的文件名,找到后便執(zhí)行該文件,然后將第二個(gè)參數(shù)argv傳給該欲執(zhí)行的文件。如果執(zhí)行成功則函數(shù)不會(huì)返回,執(zhí)行失敗則直接返回-1,失敗原因存于errno中。


之所以顯示“fail to exec”,是因?yàn)樵赑ATH環(huán)境變量所指的目錄中沒(méi)有名為“hello”的程序。建議進(jìn)行如下操作:

1、運(yùn)行“echo $PATH”,查看一下PATH環(huán)境變量指向那些目錄

2、編寫(xiě)一個(gè)輸出“hello world”的程序,并命名為hello,即執(zhí)行命令:

gcc -o hello

hello.c

3、把名為”hello“的程序拷貝到PATH變量所指的其中一個(gè)目錄中

評(píng)論0 贊同0
  •  加載中...

最新活動(dòng)

您已邀請(qǐng)0人回答 查看邀請(qǐng)

我的邀請(qǐng)列表

  • 擅長(zhǎng)該話(huà)題
  • 回答過(guò)該話(huà)題
  • 我關(guān)注的人
向幫助了您的網(wǎng)友說(shuō)句感謝的話(huà)吧!
付費(fèi)偷看金額在0.1-10元之間
<