Linux进程的创建函数fork()及其fork内核实现解析【转】
转自:http://www.cnblogs.com/zengyiwen/p/5755193.html
#include <unistd.h>pid_t fork(void);
/proc/sys/kernel/sched_child_runs_first
#include <stdio.h>#include <stdlib.h>#include <unistd.h>#include <string.h>#include <errno.h>#include <sys/types.h>#include <wait.h>int g_int = 1;//数据段的全局变量int main(){int local_int = 1;//栈上的局部变量int *malloc_int = malloc(sizeof(int));//通过malloc动态分配在堆上的变量*malloc_int = 1;pid_t pid = fork();if(pid == 0) /*子进程*/{local_int = 0;g_int = 0;*malloc_int = 0;fprintf(stderr,"[CHILD ] child change local global malloc value to 0\n");free(malloc_int);sleep(10);fprintf(stderr,"[CHILD ] child exit\n");exit(0);}else if(pid < 0){printf("fork failed (%s)",strerror(errno));return 1;}fprintf(stderr,"[PARENT] wait child exit\n");waitpid(pid,NULL,0);fprintf(stderr,"[PARENT] child have exit\n");printf("[PARENT] g_int = %d\n",g_int);printf("[PARENT] local_int = %d\n",local_int);printf("[PARENT] malloc_int = %d\n",local_int);free(malloc_int);return 0;}
[PARENT] wait child exit[CHILD ] child change local global malloc value to 0[CHILD ] child exit[PARENT] child have exit[PARENT] g_int = 1[PARENT] local_int = 1[PARENT] malloc_int = 1

/*如果是写时拷贝, 那么无论是初始页表, 还是拷贝的页表, 都设置了写保护*后面无论父子进程, 修改页表对应位置的内存时, 都会触发page fault*/if (is_cow_mapping(vm_flags)) {ptep_set_wrprotect(src_mm, addr, src_pte);//设置为写保护pte = pte_wrprotect(pte);}

struct task_struct {...struct files_struct *files;...}
static int copy_files(unsigned long clone_flags,struct task_struct *tsk){struct files_struct *oldf, *newf;int error = 0;oldf = current->files;//获取父进程的文件结构体if (!oldf)goto out;/*创建线程和vfork, 都不用复制父进程的文件描述符, 增加引用计数即可*/if (clone_flags & CLONE_FILES) {atomic_inc(&oldf->count);goto out;}/*对于fork而言, 需要复制父进程的文件描述符*/newf = dup_fd(oldf, &error); //复制一份文件描述符if (!newf)goto out;tsk->files = newf;error = 0;out:return error;}
struct files_struct *dup_fd(struct files_struct *oldf,int *errorp){struct files_struct *newf;struct file **old_fds, **new_fds;int open_files, size, i;struct fdtable *old_fdt, *new_fdt;*errorp = -ENOMEM;newf = kmem_cache_alloc(files_cachep, GFP_KERNEL);if (!newf)goto out;
struct files_struct {atomic_t count;struct fdtable __rcu *fdt;struct fdtable fdtab;spinlock_t file_lock ____cacheline_aligned_in_smp;int next_fd;struct embedded_fd_set close_on_exec_init;struct embedded_fd_set open_fds_init;struct file __rcu * fd_array[NR_OPEN_DEFAULT];};struct fdtable //文件描述符表{unsigned int max_fds;struct file __rcu **fd; /* current fd array */fd_set *close_on_exec;fd_set *open_fds;struct rcu_head rcu;struct fdtable *next;};struct embedded_fd_set {unsigned long fds_bits[1];};
file类型指针的数组fd_array,也自带了两个大小为64的位图,其中open_fds_init位图用于记录文件的打开情况,close_on_exec_init位图用于记录文件描述符的FD_CLOSEXCE标志位是否置位。只要进程打开的文件个数小于64,file_struct结构体自带的指针数组和两个位图就足以满足需要。因此在分配了file_struct结构体后,内核会初始化file_struct自带的fdtable,代码如下所示:
atomic_set(&newf->count, 1);spin_lock_init(&newf->file_lock);newf->next_fd = 0;new_fdt = &newf->fdtab;new_fdt->max_fds = NR_OPEN_DEFAULT;new_fdt->close_on_exec = (fd_set *)&newf->close_on_exec_init;new_fdt->open_fds = (fd_set *)&newf->open_fds_init;new_fdt->fd = &newf->fd_array[0];new_fdt->next = NULL;

spin_lock(&oldf->file_lock);old_fdt = files_fdtable(oldf);open_files = count_open_files(old_fdt);/*如果父进程打开文件的个数超过NR_OPEN_DEFAULT*/while (unlikely(open_files > new_fdt->max_fds)) {spin_unlock(&oldf->file_lock); /* 如果不是自带的fdtable而是曾经分配的fdtable, 则需要先释放*/if (new_fdt != &newf->fdtab)__free_fdtable(new_fdt);/*创建新的fdtable*/new_fdt = alloc_fdtable(open_files - 1);if (!new_fdt) {*errorp = -ENOMEM;goto out_release;}/*如果超出了系统限制, 则返回EMFILE*/if (unlikely(new_fdt->max_fds < open_files)) {__free_fdtable(new_fdt);*errorp = -EMFILE;goto out_release;}spin_lock(&oldf->file_lock);old_fdt = files_fdtable(oldf);open_files = count_open_files(old_fdt);}
old_fds = old_fdt->fd;/*父进程的struct file 指针数组*/- new_fds = new_fdt->fd; /*子进程的struct file 指针数组*/
- /* 拷贝打开文件位图 */
- memcpy(new_fdt->open_fds->fds_bits,old_fdt->open_fds->fds_bits, open_files/8);
- /* 拷贝 close_on_exec位图 */
- memcpy(new_fdt->close_on_exec->fds_bits,old_fdt->close_on_exec->fds_bits, open_files/8);
- for (i = open_files; i != 0; i--) {
- struct file *f = *old_fds++;
- if (f) {
- get_file(f); /* f对应的文件的引用计数加1 */
- } else {
- FD_CLR(open_files - i, new_fdt->open_fds);
- }
- /* 子进程的struct file类型指针, *指向和父进程相同的struct file 结构体*/
- rcu_assign_pointer(*new_fds++, f);
- }
- spin_unlock(&oldf->file_lock);/* compute the remainder to be cleared */
- size = (new_fdt->max_fds - open_files) * sizeof(struct file *);
- /*将尚未分配到的struct file结构的指针清零*/
- memset(new_fds, 0, size);/*将尚未分配到的位图区域清零*/
- if (new_fdt->max_fds > open_files) {
- int left = (new_fdt->max_fds-open_files)/8;
- int start = open_files / (8 * sizeof(unsigned long));
memset(&new_fdt->open_fds->fds_bits[start], 0, left);memset(&new_fdt->close_on_exec->fds_bits[start], 0, left);}rcu_assign_pointer(newf->fdt, new_fdt);return newf;out_release:kmem_cache_free(files_cachep, newf);out:return NULL;}

#include<stdio.h>#include <stdlib.h>#include <unistd.h>int glob = 88 ;int main(void) {int var;var = 88;pid_t pid;if ((pid = vfork()) < 0) {printf("vfork error");exit(-1);} else if (pid == 0) { /* 子进程 */var++;glob++;return 0;}printf("pid=%d, glob=%d, var=%d\n",getpid(), glob, var);return 0;}
Linux进程的创建函数fork()及其fork内核实现解析【转】的更多相关文章
- Linux进程的创建函数fork()及其fork内核实现解析
进程的创建之fork() Linux系统下,进程可以调用fork函数来创建新的进程.调用进程为父进程,被创建的进程为子进程. fork函数的接口定义如下: #include <unistd.h& ...
- linux 进程的创建
1. 进程号: 每个进程在被初始化的时候,系统都会为其分配一个唯一标识的进程id,称为进程号: 进程号的类型为pid_t,通过getpid()和getppid()可以获取当前进程号和当前进程的父进程的 ...
- linux进程学习-创建新进程
init进程将系统启动后,init将成为此后所有进程的祖先,此后的进程都是直接或间接从init进程“复制”而来.完成该“复制”功能的函数有fork()和clone()等. 一个进程(父进程)调用for ...
- linux进程解析--进程的创建
通常我们在代码中调用fork()来创建一个进程或者调用pthread_create()来创建一个线程,创建一个进程需要为其分配内存资源,文件资源,时间片资源等,在这里来描述一下linux进程的创建过程 ...
- Linux进程-进程的创建
今天学习了Linux的进程创建的基本原理,是基于0.11版本核心的.下面对其作一下简单的总结. 一.Linux进程在内存中的相关资源 很容易理解,Linux进程的创建过程就是内存中进程相关资源产生 ...
- Linux 进程,线程,线程池
在linux内核,线程与进程的区别很小,或者说内核并没有真正所谓单独的线程的概念,进程的创建函数是fork,而线程的创建是通过clone实现的. 而clone与fork都是调用do_fork(),差异 ...
- 撸代码--linux进程通信(基于共享内存)
1.实现亲缘关系进程的通信,父写子读 思路分析:1)首先我们须要创建一个共享内存. 2)父子进程的创建要用到fork函数.fork函数创建后,两个进程分别独立的执行. 3)父进程完毕写的内容.同一时候 ...
- Operating System-Process(1)什么是进程&&进程的创建(Creation)&&进程的终止(Termination)&&进程的状态(State)
本文阐述操作系统的核心概念之一:进程(Process),主要内容: 什么是进程 进程的创建(Creation) 进程的终止(Termination) 进程的状态(State) 一.什么是进程 1.1 ...
- 通过fork函数创建进程的跟踪,分析linux内核进程的创建
作者:吴乐 山东师范大学 <Linux内核分析>MOOC课程http://mooc.study.163.com/course/USTC-1000029000 一.实验过程 1.打开gdb, ...
随机推荐
- noip模拟题《序》sort
[问题背景] zhx 给他的妹子们排序.[问题描述] zhx有N个妹子,他对第i个妹子的好感度为ai, 且所有ai两两不相等.现在N个妹子随意站成一 排,他要将她们根据好感度从小到 ...
- C++函数中的那些坑
平时写程序时,我们可能或多或少对一些用法感到朦胧,下面我对一些易困惑大家,或者易用错的地方作点介绍. 一.函数的一些注意点 1.函数返回类型不能是数组类型或函数类型,但可以是指向数组或函数的指针. 2 ...
- Contest 1
A:注意到模数是要求lcm的数的倍数,直接先取模就可以了.考场脑抽,对其质因数分解判了一下每个因子有没有,当然也行. #include<iostream> #include<cstd ...
- 【刷题】LOJ 6041 「雅礼集训 2017 Day7」事情的相似度
题目描述 人的一生不仅要靠自我奋斗,还要考虑到历史的行程. 历史的行程可以抽象成一个 01 串,作为一个年纪比较大的人,你希望从历史的行程中获得一些姿势. 你发现在历史的不同时刻,不断的有相同的事情发 ...
- Android 通过浏览器打开应用
在很多应用的web站,其实都有这样一个功能,就是直接在网页中打开应用,接下来的就来探讨一下这个功能的实现,有些地方也我还没弄明白,还请懂的大神指点. 首先,得说一点不好消息,在微信中,这样的方式是行不 ...
- Linux下C高手成长过程----经典书籍推荐
http://www.cnblogs.com/shanzhizi/archive/2012/07/10/2585357.html
- [CodeVs3196]黄金宝藏(DP/极大极小搜索)
题目大意:给出n(≤500)个数,两个人轮流取数,每次可以从数列左边或者右边取一个数,直到所有的数被取完,两个人都以最优策略取数,求最后两人所得分数. 显然这种类型的博弈题,第一眼就是极大极小搜索+记 ...
- Codeforces Round #298 (Div. 2)A B C D
A. Exam time limit per test 1 second memory limit per test 256 megabytes input standard input output ...
- Chiaki Sequence Revisited HDU - 6304 lowbit找规律法
Problem Description Chiaki is interested in an infinite sequence a1,a2,a3,..., which is defined as f ...
- [洛谷P1709] [USACO5.5]隐藏口令Hidden Password
洛谷题目链接:[USACO5.5]隐藏口令Hidden Password 题目描述 有时候程序员有很奇怪的方法来隐藏他们的口令.Binny会选择一个字符串S(由N个小写字母组成,5<=N< ...