一、 进程创建:

  Unix 下的进程创建很特别,与许多其他操作系统不同,它分两步操作来创建和执行进程: fork() 和 exec() 。首先,fork() 通过拷贝当前进程创建一个子进程;然后,exec() 函数负责读取可执行文件并将其载入地址空间开始运行。

1、fork() :kernel/fork.c

  在Linux系统中,通过调用fork()来创建一个进程。调用 fork() 的进程称为父进程,新产生的进程称为子进程。在该调用结束时,在返回点这个相同的位子上,父进程恢复执行,子进程开始执行。fork()系统调用从内核返回两次:一次返回到父进程,另一次返回到新产生的子进程。使用fork()创建新进程的流程如下:

1)fork() 调用clone;

2)clone() 调用 do_fork();

3)do_fork() 调用 copy_process() 函数,copy_process() 函数将完成第 4-11 步;

4)调用 dup_task_struct() 为新进程创建一个内核栈、thread_info结构和task_struct,这些值与当前进程的值相同;

5)检查并确保新创建这个子进程后,当前用户所拥有的进程数目没有超出给它分配的资源的限制;

6)清理子进程进程描述符中的一些成员(清零或初始化,如PID),以使得子进程与父进程区别开来;

7)将子进程的状态设置为 TASK_UNINTERRUPTIBLE,保证它不会投入运行;

8)调用 copy_flags() 以更新 task_struct 的 flags 成员;

9)调用 alloc_pid() 为新进程分配一个有效的 PID;

10)根据传递给clone() 的参数标志,copy_process() 拷贝或共享打开的文件、文件系统信息、信号处理函数、进程地址空间和命名空间等;

11)做一些扫尾工作并返回一个指向子进程的指针。

12)回到 do_fork() 函数,如果 copy_process() 函数成功返回,新创建的子进程将被唤醒并让其投入运行。

  下面用一段简单的代码演示一下 fork() 函数:

   #include <unistd.h>
#include <stdio.h> int main(){
pid_t fpid;
int count= ;
fpid = fork(); // fpid 为fork()的返回值
if(fpid < ){ // 当fork()的返回值为负值时,表明调用 fork() 出错
printf("error in fork!");
}
else if(fpid == ){ // fork() 返回值为0,表明该进程是子进程
printf("this is a child process, the process id is %d\n",getpid());
count++;
}
else{ // fork() 返回值大于0,表明该进程是父进程,这时返回值其实是子进程的PID
printf("this is a father process, the process id is %d\n",getpid());
count++;
}
printf("计数 %d 次\n",count);
return ;
}

输出结果:

  可以看到,调用 fork() 函数后,原本只有一个进程,变成了两个进程。这两个进程除了 fpid 的值不同外几乎完全相同,它们都继续执行接下来的程序。由于 fpid 的值不同,因此会进入不同的判断语句,这也是为什么两个结果有不同之处的原因。另外,可以看到,父进程的 PID 刚好比子进程的 PID 小1。 fork()  的返回值有以下三种:

a)在父进程中,fork() 返回新创建子进程的 PID;

b)在子进程中,fork() 返回0;

c)如果 fork() 调用出错,则返回负值

 2、exec() :fs/exec.c (源程序 exec.c 实现对二进制可执行文件和 shell 脚本文件的加载与执行)

  通常,创建新的进程都是为了立即执行新的、不同的程序,而接着调用 exec() 这组函数就可以创建新的地址空间,并把新的程序载入其中

  exec() 并不是一个函数,而是一个函数簇,一共包含六个函数,分别为: execl、execlp、execle、execv、execvp、execve,定义如下:

#include <unistd.h>  

int execl(const char *path, const char *arg, ...);
int execlp(const char *file, const char *arg, ...);
int execle(const char *path, const char *arg, ..., char *const envp[]);
int execv(const char *path, char *const argv[]);
int execvp(const char *file, char *const argv[]);
int execve(const char *path, char *const argv[], char *const envp[]);

  这六个函数的功能其实差不多,只是接受的参数不同。exec() 函数的参数主要有3个部分:执行文件部分、命令参数部分和环境变量部分:

1)执行文件部分:也就是函数中的 path 部分,该部分指出了可执行文件的查找方式。其中 execl、execle、execv、execve的查找方式都是使用的绝对路径,而 execlp和execvp则可以只给出文件名进行查找,系统会从环境变量 "$PATH"中查找相应的路径;

2)命令参数部分:也就是函数中的 file 部分,该部分指出了参数的传递方式以及要传递哪些参数。这里,"l"结尾的函数表示使用逐个列举的方式传递参数;"v"结尾的表示将所有参数整体构造成一个指针数组进行传递,然后将该数组的首地址当做参数传递给它,数组中的最后一个指针要求为 NULL;

3)环境变量部分:exec() 函数簇使用了系统默认的环境变量,也可以传入指定的环境变量。其中 execle 和execve 这两个函数就可以在 envp[] 中指定当前进程所使用的环境变量。

·  当 exec() 执行成功时,exec() 函数会取代执行它的进程,此时,exec() 函数没有返回值,进程结束。当 exec() 函数执行失败时,将返回失败信息(返回 -1),进程继续执行后面的代码。

  通常,exec() 会放在 fork() 函数的子进程部分,来替代子进程继续执行,exec() 执行成功后子进程就会消失,但是执行失败的话,就必须要使用 exit() 函数来让子进程退出。下面用一段简单的代码来演示一下 exec() 函数簇中的一个函数的用法,其余的参考:https://www.cnblogs.com/dongguolei/p/8098181.html

   #include <unistd.h>
#include <stdio.h>
#include <errno.h>
#include <string.h> int main(){
int childpid;
pid_t fpid;
fpid = fork();
if(fpid == ){             // 子进程
char *execv_str[] = {"ps","aux",NULL}; // 指令:ps aux 查看系统中所有进程
if( execv("/usr/bin/ps",execv_str) < ){
perror("error on exec\n");
exit();
}
}
else{
wait(&childpid);
printf("execv done\n");
}
}

  在这个程序中,使用 fork() 创建了一个子进程,随后立即调用 exec() 函数簇中的 execv() 函数,execv() 函数执行了一条指令,显示当前系统中所有的进程,结果如下(进程有很多,这里只截了一部分):

  注意看最后两个进程,分别是父进程和调用 fork() 后创建的子进程。

二、进程终结

  进程被创建后,最终要终结。当一个进程终结时,内核必须释放它所占有的资源,并把这一消息告诉其父进程。系统通过 exit() 系统调用来处理终止和退出进程的相关工作,而大部分工作则由 do_exit() 来完成 (kernel/exit.c):

1)将task_struct 中的标志成员设置为 PF_EXITING;

2)调用 del_timer_sync() 删除任一内核定时器,以确保没有定时器在排队,也没有定时器处理程序在运行;

3)调用 exit_mm() 函数释放进程占用的 mm_struct,如果没有别的进程使用它们(地址空间被共享),就彻底释放它们;

4)调用 sem__exit() 函数,如果进程排队等候 IPC 信号,它则离开队列;

5)调用 exit_files() 和 exit_fs(),以分别递减文件描述符、文件系统数据的引用计数,若其中某个引用计数的值降至零,则表示没有进程使用相应的资源,可以释放掉进程占用的文件描述符、文件系统资源;

6)把 task_struct 的 exit_code 成员设置为进程的返回值;

7)调用 exit_notify() 向父进程发送信号,并把进程状态设置为 EXIT_ZOMBIE;

8)调用 schedule() 切换到新的进程,继续执行。由于 EXIT_ZOMBIE 状态的进程不会被再调度,所以这是进程所执行的最后一段代码, do_exit() 没有返回值。

  至此,与进程相关联的所有资源都被释放掉了,进程不可运行并处于 EXIT_ZOMBIE 退出状态。此时,进程本身所占用的内存还没有释放,如内核栈、thread_info 结构和 task_struct 结构等,它存在的意义是向父进程提供信息,当父进程收到信息后,或者通知内核那是无关的信息后,进程所持有的剩余的内存将被释放。父进程可以通过 wait4() 系统调用查询子进程是否终结,这其实使得进程拥有了等待特定进程执行完毕的能力。

  系统通过调用 release_task() 来释放进程描述符。

Linux内核学习笔记(3)-- 进程的创建和终结的更多相关文章

  1. Linux内核学习笔记-2.进程管理

    原创文章,转载请注明:Linux内核学习笔记-2.进程管理) By Lucio.Yang 部分内容来自:Linux Kernel Development(Third Edition),Robert L ...

  2. Linux内核学习笔记二——进程

    Linux内核学习笔记二——进程   一 进程与线程 进程就是处于执行期的程序,包含了独立地址空间,多个执行线程等资源. 线程是进程中活动的对象,每个线程都拥有独立的程序计数器.进程栈和一组进程寄存器 ...

  3. Linux内核学习笔记之seq_file接口创建可读写proc文件

    转自:http://blog.csdn.net/mumufan05/article/details/45803219 学习笔记与个人理解,如有错误,欢迎指正. 温馨提示:建议跟着注释中的编号顺序阅读代 ...

  4. Linux内核学习笔记-1.简介和入门

    原创文章,转载请注明:Linux内核学习笔记-1.简介和入门 By Lucio.Yang 部分内容来自:Linux Kernel Development(Third Edition),Robert L ...

  5. 20135316王剑桥Linux内核学习笔记

    王剑桥Linux内核学习笔记 <Linux内核分析>MOOC课程http://mooc.study.163.com/course/USTC-1000029000 计算机是如何工作的 个人理 ...

  6. Linux内核分析 笔记六 进程的描述和进程的创建 ——by王玥

    一.知识点总结 (一)进程的描述 1.操作系统内核里有三大功能: 进程管理 内存管理 文件系统 2.进程描述符:task_struct 2.进程描述符——struct task_struct 1. p ...

  7. (笔记)Linux内核学习(二)之进程

    一 进程与线程 进程就是处于执行期的程序,包含了独立地址空间,多个执行线程等资源. 线程是进程中活动的对象,每个线程都拥有独立的程序计数器.进程栈和一组进程寄存器. 内核调度的对象是线程而不是进程.对 ...

  8. Linux内核学习笔记(1)-- 进程管理概述

    一.进程与线程 进程是处于执行期的程序,但是并不仅仅局限于一段可执行程序代码.通常,进程还要包含其他资源,像打开的文件,挂起的信号,内核内部数据,处理器状态,一个或多个具有内存映射的内存地址空间及一个 ...

  9. linux内核学习之四:进程切换简述

    在讲述专业知识前,先讲讲我学习linux内核使用的入门书籍:<深入理解linux内核>第三版(英文原版叫<Understanding the Linux Kernel>),不过 ...

随机推荐

  1. navicat导出数据库字典

    select TABLE_SCHEMA,TABLE_NAME,COLUMN_NAME,COLUMN_TYPE,COLUMN_COMMENT from information_schema.column ...

  2. linux查看nginx、apache、php、php-fpm、mysql及配置项所在目录

    可以先总结下:大都是先用 which 获取目录:然后再获取配置项位置: which  mysql /usr/bin/mysql /usr/bin/mysql --help | grep -A1 'De ...

  3. webpack4.x最详细入门讲解

    前言 本文主要从webpack4.x入手,会对平时常用的Webpack配置一一讲解,各个功能点都有对应的详细例子,所以本文也比较长,但如果你能动手跟着本文中的例子完整写一次,相信你会觉得Webpack ...

  4. vlookup函数使用---execl公式

    目录 vlookup函数使用---execl公式 1.用途: 2.函数语法: 3.使用方式: 4.实际案例+步骤解析 5.常见错误 vlookup函数使用---execl公式 1.用途: 我们有一张工 ...

  5. Google Cloud Platform 续

    Google Cloud Platform 创建新实例 地区:australia-southeast1-a 机器类型:1个vCPU n1-standard-1 系统:Ubuntu 16.04 LTS ...

  6. 怎样在Win7系统中搭建Web服务器

    一.搭建web服务 1.打开控制面板,选择并进入“程序”,双击“打开或关闭Windows服务”,在弹出的窗口中选择“Internet信息服务”下面所有的选项,点击确定后,开始更新服务. 2.更新完成后 ...

  7. shell习题第8题:监控nginx的502状态

    [题目要求] 服务器上跑的是LNMP环境,近期总是有502现象.502为网站访问的状态码,200正常,502错误是nginx最为普遍的错误状态码. 由于502只是暂时的,并且只要一重启php-fpm服 ...

  8. jQuery属性操作之.val()函数

    目录 .val()实例方法的三种用法 .val()函数源码 调用形式:$('xxx').val(); 调用形式:$('xxx').val(value); 调用形式:$('xxx').val(funct ...

  9. Js错误: obj.parents is not a function

    代码:      (1)  <div class="ViewMore" id="viewmore${i}" onclick="CLICK(thi ...

  10. 通过XShell实现windows文件上传到Linux服务器上

    .XShell上传文件到Linux服务器上 在学习Linux过程中,我们常常需要将本地文件上传到Linux主机上,这里简单记录下使用Xsheel工具进行文件传输 1:首先连接上一台Linux主机 2: ...