一起学《Linux是怎样工作的》第3章进程管理 3.3~3.4节(execve()函数/结束进程)

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()函数。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值