创建进程需要使用系统调用,一共有三个,分别是:sys_fork,sys_vfork,sys_clone ?所有的这三个系统调用都要使用do_fork函数,它是创建进程的主体,我们将在下面重点介绍 ?原型分别如下: /* Fork a new task - this creates a new program thread. * This is called indirectly via a small wrapper */ asmlinkage int sys_fork(struct pt_regs *regs) { return do_fork(SIGCHLD, regs->ARM_sp, regs, 0); } /* Clone a task - this clones the calling program thread. * This is called indirectly via a small wrapper */ asmlinkage int sys_clone(unsigned long clone_flags, unsigned long newsp, struct pt_regs *regs) { if (!newsp) newsp = regs->ARM_sp; return do_fork(clone_flags, newsp, regs, 0); } asmlinkage int sys_vfork(struct pt_regs *regs) { return do_fork(CLONE_VFORK | CLONE_VM | SIGCHLD, regs->ARM_sp, regs, 0); }?这三个系统调用的区别是: sys_fork是完整的从父进程派生出一个子进程;sys_clone可以通过参数clone_flags决定需要复制给子进程的资源;而sys_vfork则只是产生了一个新的task_struct,它还是和父进程共享其余的所有的资源,所以它不是真正的进程,只能算是线程,他将组塞父进程的运行直到自己运行结束为止。 ?有一个信号叫做SIGCHLD,当子进程停止或结束,父进程将收到一个SIGCHLD信号,默认动作是忽略。 ?也就是说fork才是真正的clone。?背景知识:进程和进程的状态 #define TASK_RUNNING 0 #define TASK_INTERRUPTIBLE 1 #define TASK_UNINTERRUPTIBLE 2 #define TASK_ZOMBIE 4 #define TASK_STOPPED 8?1) TASK_RUNNING:含义是假定任务已经处于运行队列中。至于不是已经处于运行状态的原因是由于将一个任务标识为TASK_RUNNING和将该任务移动到运行队列不是一个原子操作。 ?2) TASK_INTERRUPTIBLE:含义是任务处于休眠状态但可以通过一个信号或者休眠中止时钟唤醒。 ?3) TASK_UNINTERRUPTIBLE:含义类似于TASK_INTERRUPTIBLE,但任务不能被唤醒。 ?4) TASK_ZOMBIE:含义是任务已经被中止但它的状态还没被父进程获取。 ?5) TASK_STOPPED:含义是由于任务控制信号或者ptrace系统调用,任务已经被停止。 ?int do_fork(unsigned long clone_flags, unsigned long stack_start, ? struct pt_regs *regs, unsigned long stack_size) ?{ ? int retval; ? struct task_struct *p; ? struct completion vfork; ?? retval = -EPERM; if (clone_flags & CLONE_PID) { ? if (current->pid) ? goto fork_out; ? } ?? retval = -ENOMEM; ?上面这段代码的意思是: 如果clone_flags的CLONE_PID置位,则除非是0号进程 否则将返回错误信息EPERM. clone_flags的定义将在后面介绍,0号进程就是操作 系统启动时创建的第一个进程.上面的代码中if(current->pid)要成立,除非是0号 进程.其中pid是进程标志符 ?p = alloc_task_struct(); /* franc: we will alloc one page space as task_struct */ ? if (!p) ? goto fork_out; ?? *p = *current; /* franc: just copy all the info of its parent to this new process */ ?? retval = -EAGAIN; ?上面代码的意思是: 申请一个新的内存控制块,其中alloc_task_struct是在/arch /arm/kernel/process.c中定义: struct task_struct *alloc_task_struct(void)其中: ll_alloc_task_struct() 在/include/asm-arm/proc-armv/processor.h中定义: #define ll_alloc_task_struct() ((struct task_struct *) __get_free_pages(GFP_KERNEL,1)) ?_get_free_pages是要申请两个页面,页面是内存管理的有关内容,这儿大家只要一个页面占4K个字节就可以了.两个页面就是8K个字节,而大家查一下进程控制块的大小就可以知道它只有1600B左右,也就是说它只有1.5K左右,那么还剩下将近6.5K的空间去有什么用呢?其实剩余的空间是作为内核栈空间.如果申请内存空间失败则直接栈道fork_out处执行,fork_out在后面有定义 ?然后就将父进程的所有信息直接拷贝给它。 ?if (atomic_read(&p->user->processes) >= p->rlim[RLIMIT_NPROC].rlim_cur ? && !capable(CAP_SYS_ADMIN) && !capable(CAP_SYS_RESOURCE)) /* franc: check if we have spawned too many processes */ ? goto bad_fork_free; ?? atomic_inc(&p->user->__count); /* make the process number tracable */ ? atomic_inc(&p->user->processes); ?上面代码的意思: 如果父进程已经用了用户进程,则修改p->user->_count和 p->user->processes,将其数目加1,当然除非用户进程的 RLIMIT_NPROC 已经超过额定数目。
|