欧美在线专区-欧美在线伊人-欧美在线一区二区三区欧美-欧美在线一区二区三区-pornodoxxx中国妞-pornodoldoo欧美另类

position>home>Basketball

從創建進程到進入main函數,發生了什么?

[導讀]這篇文章主要討論的從創程是操作系統層面上對于進程、線程的建進進入創建初始化等行為,而像Python、函數Java等基于解釋器、發生虛擬機的從創程語言,如何進入到main函數執行,建進進入這背后的函數路徑則更長(包含了解釋器和虛擬機內部的執行流程),以后有機會再討論。發生所以這里就重點關注C/C++這類native語言的從創程main函數是如何進入的。

關注+星標公眾,建進進入不錯過精彩內容
作者 | 軒轅之風
轉自 | 編程技術宇宙

首先先劃定一下這個問題的函數討論范圍:C/C++語言

從創建進程到進入main函數,發生了什么?

這篇文章主要討論的是操作系統層面上對于進程、線程的發生創建初始化等行為,而像Python、從創程Java等基于解釋器、建進進入虛擬機的函數語言,如何進入到main函數執行,這背后的路徑則更長(包含了解釋器和虛擬機內部的執行流程),以后有機會再討論。所以這里就重點關注C/C++這類native語言的main函數是如何進入的。

本文會兼顧敘述LinuxWindows兩個主要平臺上的詳細流程。

創建進程

第一步,創建進程。

在Linux上,我們要啟動一個新的進程,一般通過fork+ exec系列函數來實現,前者將當前進程“分叉”出一個孿生子進程,后者負責替換這個子進程的執行文件,來執行子進程的新程序文件。

這里的forkexec系列函數,是操作系統提供給應用程序的API函數,在其內部最終都會通過系統調用,進入操作系統內核,通過內核中的進程管理機制,來完成一個進程的創建。

操作系統內核將負責進程的創建,主要有下面幾個工作要做:

  • 創建內核中用于描述進程的數據結構,在Linux上是 task_struct
  • 創建新進程的頁目錄、頁表,用于構建新進程的內存地址空間

在Linux內核中,由于歷史原因,Linux內核早期并沒有線程的概念,而是用任務:task_struct來描述一個程序的執行實例:進程

在內核中,一個任務對應就是一個task_struct,也就是一個進程,內核的調度單元也是一個個的個task_struct

后來,多線程的概念興起,Linux內核為了支持多線程技術,task_struct實際上表示的變成了一個線程,通過將多個task_struct合并為一組(通過該結構內部的組id字段)再來描述一個進程。因此,Linux上的線程,也稱為輕量級進程

系統調用fork的一個重要使命就是要去創建新進程的task_struct結構,創建完成后,進程就擁有了調度單元。隨后將開始可以參與調度并有機會獲得執行。

加載可執行文件

通過fork成功創建進程后,此時的子進程和父進程相當于一個細胞進行了有絲分裂,兩個進程“幾乎”是一模一樣的。

而要想子進程執行新的程序,在子進程中還需要用到exec系列函數來實現對進程可執行程序的替換。

exec系列函數同樣是系統調用的封裝,通過調用它們,將進入內核sys_execve來執行真正的工作。

這個工作細節比較多,其中有一個重要的工作就是加載可執行文件到進程空間并對其進行分析,提取出可執行文件的入口地址

我們使用C、C++等高級語言編寫的代碼,最終通過編譯器會編譯生成可執行文件,在Linux上,是ELF格式,在Windows上,稱之為PE文件。

無論是ELF文件還是PE文件,在各自的文件頭中,都記錄了這個可執行文件的指令入口地址,它指示了程序該從哪里開始執行。

這個入口指向哪里,是我們的main函數嗎?這里賣一個關子,先來解決在這之前的一個問題:進程創建后,是如何來到這個入口地址的?

不管在Windows還是Linux上,應用線程都會經常在用戶空間和內核空間來回穿梭,這可能出現在以下幾種情況發生時:

  • 系統調用
  • 中斷
  • 異常

從內核返回時,線程是如何知道自己從哪里進來的,該回到應用空間的哪里去繼續執行呢?

答案是,在進入內核空間時,線程將自動保存上下文(其實就是一些寄存器的內容,比如指令寄存器EIP)到線程的堆棧上,記錄自己從哪里來的,等到從內核返回時,再從堆棧上加載這些信息,回到原來的地方繼續執行。

前面提到,子進程是通過sys_execve系統調用進入到內核中的,在后面完成可執行文件的分析后,拿到了ELF文件的入口地址,將會去修改原來保存在堆棧上的上下文信息,將EIP指向ELF文件的入口地址。這樣等sys_execve系統調用結束時,返回到用戶空間后,就能夠直接轉到新的程序入口開始執行代碼。

所以,一個非常重要的特點是:exec系列函數正常情況下是不會返回的,一旦進入,完成使命后,執行流程就會轉向新的可執行文件入口

另外需要提一下的是,在Linux上,除了ELF文件,還支持一些其他格式的可執行文件,如MS-DOS、COFF

除了二進制的可執行文件,還支持shell腳本,這個情況下將會將腳本解釋器程序作為入口來啟動

從ELF入口到main函數

上面交代了,一個新的進程,是如何執行到可執行文件的入口地址的。

同時也留了一個問題,這個入口地址是什么?是我們的main函數嗎?

這里有一個簡單的C程序,運行起來后輸出經典的hello world:

#include?
int?main()?{
????printf("hello,?world!\n");
????return?0;
}

通過gcc編譯后,生成了一個ELF可執行文件,通過readelf指令,可以實現對ELF文件的分析,這里可以看到ELF文件的入口地址是0x400430:

隨后,我們通過反匯編神器,IDA打開分析這個文件,看一下位于0x400430入口的地方是什么函數?

可以看到,入口地方是一個叫做 _start的函數,并不是我們的main函數。

在_start的結尾,調用了 __libc_start_main函數,而這個函數,位于libc.so中。

你可能疑惑,這個函數是哪里冒出來的,我們的代碼中并沒有用到它呢?

其實,在進入main函數之前,還有一個重要的工作要做,這就是:C/C++運行時庫的初始化。上面的 __libc_start_main就是在完成這一工作。

在通過GCC進行編譯時,編譯器將自動完成運行時庫的鏈接,將我們的main函數封裝起來,由它來調用。

glibc是開源的,我們可以在GitHub上找到這個項目的libc-start.c文件,一窺 __libc_start_main的真面目,我們的main函數正是被它在調用。

完整流程

到這里,我們梳理了,從進程創建fork,到通過exec系列函數完成可執行文件的替換,再到執行流程進入到ELF文件的入口,再到我們的main函數的完整流程。

Windows上的一些區別

下面簡單介紹下Windows上這一流程的一些差異。

首先是創建進程的環節,Windows系統將fork+exec兩步合并了一步,通過CreateProcess系列函數一步到位,在其參數中指定子進程的可執行文件路徑。

不同于Linux上進程和線程的邊界模糊,在Windows操作系統上,內核是有明確的進程和線程概念定義,進程用EPROCESS結構表示,線程用ETHREAD結構表示。

所以在Windows上,進程相關的工作準備就緒后,還需要單獨創建一個參與內核調度的執行單元,也就是進程中的第一個線程:主線程。當然,這個工作也封裝在了CreateProcess系列函數中了。

新進程的主線程創建完成后,便開始參與系統調度了。主線程從哪里開始執行呢?內核在創建時就明確進行了指定:nt!KiThreadStartup,這是一個內核函數,線程啟動后就從這里開始執行。

線程從這里啟動后,再通過Windows的異步過程調用APC機制執行提前插入的APC,進而將執行流程引入應用層,去執行Windows進程應用程序的初始化工作,比如一些核心DLL文件的加載(Kernel32.dll、ntdll.dll)等等。

隨后,再次通過APC機制,再轉向去執行可執行文件的入口點。

這后面和Linux上的機制類似,同樣沒有直接到main函數,而是需要先進行C/C++運行時庫的初始化,這之后經過運行時函數的包裝,才最終來到我們的main函數。

下面是Windows上,從創建進程到我們的main函數的完整流程(高清大圖:https://bbs.pediy.com/upload/attach/201604/501306_qz5f5hi1n3107kt.png):

現在你清楚,從進程啟動是怎么一步步到你的main函數的了嗎?

------------?END?------------

推薦閱讀:

SEGGER的三款RTOS有什么特點?

Keil MDK如何將變量存儲在指定內存地址

幾款優秀的支持C、C++等多種語言的在線編譯器


關注 微信公眾號『strongerHuang』,后臺回復“1024”查看更多內容,回復“加群”按規則加入技術交流群。


長按前往圖中包含的公眾號關注


點擊“ 閱讀原文”查看更多分享,歡迎點分享、收藏、點贊、在看

免責聲明:本文內容由21ic獲得授權后發布,版權歸原作者所有,本平臺僅提供信息存儲服務。文章僅代表作者個人觀點,不代表本平臺立場,如有問題,請聯系我們,謝謝!

Popular articles

主站蜘蛛池模板: 阿v视频免费在线观看| 98av视频| 亚韩在线| 欧美aaaaaaaaa| 国内精品视频一区二区三区八戒 | 国产一区精品视频| 老子影院午夜| 啊好深好硬快点用力视频| 女人是男人的女未来1分49分 | 天天在线天天看成人免费视频| 亚洲精品动漫在线| 一道本免费视频| 天堂亚洲国产日韩在线看| 国漫永生第二季在线观看| 午夜羞羞影院| 欧美中文字幕无线码视频 | 大香伊人久久精品一区二区| 国产日韩欧美中文字幕| 在线日韩理论午夜中文电影| 福利久草| 大香线一本| 韩国一大片a毛片女同| 日本免费看片在线播放| 国产线路中文字幕| 成人免费福利电影| 三年片免费高清版| 免费国产va在线观看视频| 在公交车上弄到高c了公交车视频| 国产国语一级毛片全部| 亚洲欧美日韩在线不卡| 四虎影院国产| 无毒不卡在线观看| 日本大片免费一级| 永久免费无内鬼放心开车| 8av国产精品爽爽ⅴa在线观看| 最近中文字幕mv在线视频www| 波多野结衣作品大全| 国产va在线播放| 2021天天干| 最近免费中文字幕大全| 免费一级在线观|