3.3 execve()函数
在打算启动另一个程序时,需要调execve()函数。首先,我们看下内核在运行进程时的流程。
1.读取可执行文件,病读取创建进程的内存映像所需的信息。
2.用新进程的数据覆盖当前进程的内存。
3.从最初的命令开始运行新的进程。
也就是说,在启动另一个程序时,并非新增一个进程,而是替换了当前进程。如下。
首先,读取可执行文件,以及创建进程的内存映像所需的信息。可执行文件不仅包含进程在运行过程中使用的代码与数据,还包含开始运行程序时所需的数据
1.包含代码的代码段在文件中的偏移量、大小,以及内存映像的其实地址
2.包含代码意外的变量等数据的数据段在文件中的便宜量、大小、以及内存映像的其实地址。
3. 程序执行的第一条指令的内存地址(入口点)
假设将要运行的程序的可执行文件的结构下所示:
与使用高级编程语言编写的源代码不同,在CPU上执行机器语言的指令时,必须提供需要操作的内存地址,因此在代码段和数据段中必须包含内存映像的起始地址,例如,使用一种虚拟的高级编程语言编写了如下一段源代码。
c = a + b
在机器语言层面,这段代码将转变成下面这样的直接对内存地址进行操作的指令。
load m100 r0 ;将内存地址100(变量a)的值读取到名为r0的寄存器中
load m200 r1 ;将内存地址200(变量b)的值读取到名为r1的寄存器中
add r0 r1 r2 ;将r0与r1相加,运算结果存到r2寄存器中
store r2 m300 ;将r2的值存到内存地址300(变量c)中
接下来基于读取的信息,将程序映射到内存上,如下:
最后,从入口点开始执行程序:
在打算创建一个别的进程时,通常采用被称为fork and exec的方式,即由父进程调用fork()创建子进程,再由子进程调用exec()。我们编写一个程序来了解一下这种方式,要求如下:
1.创建一个新的进程。
2.在创建echo hello程序后,父进程输出自身与子进程的进程ID,并结束运行,子进程输出自身的进程ID,然后结束进程。
#include <unistd.h>
#include <stdio.h>
#include <stdlib.h>
#include <err.h>
static void child()
{
char *args[] = {"/bin/echo","hello",NULL};
printf("I'm child! my pid is %d.\n",getpid());
fflush(stdout);
execve("/bin/echo",args,NULL);
err(EXIT_FAILURE,"exec() failed");
}
static void parent(__pid_t pid_c)
{
printf("I'm parent! my pid is %d and the pid of my child is %d.\n",getpid(),pid_c);
exit(EXIT_SUCCESS);
}
int main(void)
{
__pid_t ret;
ret = fork();
if(ret == -1)
{
err(EXIT_FAILURE,"fork failed\n");
}
if(ret ==0)
{
//fork()会返回0给子进程,因此这里调用child()
child();
}
else
{
//fork()会返回新创建的子进程的进程ID(大于1)给父进程,因此这里调用parent
parent(ret);
}
//在正常运行时,不可能运行到这里
err(EXIT_FAILURE, "shouldn't reach here");
}
运行结果如下:
Alex@ubuntu:test3_2$ cc -o fork-and-exec fork-and-exec.c
Alex@ubuntu:test3_2$ ./fork-and-exec
I'm parent! my pid is 4901 and the pid of my child is 4902.
I'm child! my pid is 4902.
Alex@ubuntu:test3_2$ hello
3.4 结束进程
可以使用_exit()函数(底层发起exit_group()系统调用)来结束进程,不过通常使用C标准库中的exit()函数来结束进程。这种情况下,C标准库会在调用完自身的终止处理后调用_exit()函数。