做者 | 轩辕之风O
php
来源 | 编程技术宇宙(ID:xuanyuancoding)程序员
头图 | CSDN 下载自东方IC
前几天,读者群里有小伙伴提问:从进程建立后,究竟是怎么进入我写的 main 函数的?
shell
今天这篇文章就来聊聊这个话题。编程
首先先划定一下这个问题的讨论范围:C/C++语言。数据结构
这篇文章主要讨论的是操做系统层面上对于进程、线程的建立初始化等行为,而像 Python、Java 等基于解释器、虚拟机的语言,如何进入到 main 函数执行,这背后的路径则更长(包含了解释器和虚拟机内部的执行流程),之后有机会再讨论。因此这里就重点关注 C/C++这类 native 语言的 main 函数是如何进入的。多线程
本文会兼顾叙述 Linux 和 Windows 两个主要平台上的详细流程。异步
建立进程
第一步,建立进程。函数
在 Linux 上,咱们要启动一个新的进程,通常经过 fork + exec 系列函数来实现,前者将当前进程“分叉”出一个孪生子进程,后者负责替换这个子进程的执行文件,来执行子进程的新程序文件。spa
这里的 fork、exec 系列函数,是操做系统提供给应用程序的 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 <stdio.h>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函数的了吗?有疑惑和不解的地方,欢迎留言交流。
更多精彩推荐
☞Unity “出圈”:游戏引擎的技术革新和跨界商机 ☞大写的服!用耳朵也能写代码?盲人程序员自学编程成为全栈工程师 ☞小霸王被申请破产重整;虎牙员工自曝被HR抬出公司;Office 2010被微软终止服务|极客头条 ☞有了图分析,可解释的AI还远吗? ☞移动云11.11,钜惠High不停! ☞提升警戒!国内虚拟货币犯罪形势日渐严峻
点分享点点赞点在看