潘恒 原创做品转载请注明出处《Linux内核分析》MOOC课程http://mooc.study.163.com/course/USTC-1000029000node
task_struct结构:linux
struct task_struct {
volatile long state;进程状态
void *stack; 堆栈
pid_t pid; 进程标识符
unsigned int rt_priority;实时优先级
unsigned int policy;调度策略
struct files_struct *files;系统打开文件
...
}
内核处理函数sys_clone:
系统调用经过do_fork实现进程的建立:
return do_fork(clone_flags, newsp, 0, parent_tidptr, child_tidptr);
linux系统是经过复制父进程的信息来建立子进程,在do_fork函数中,真正实现复制的是copy_process函数:
p = copy_process(clone_flags, stack_start, stack_size,child_tidptr, NULL, trace);
p = dup_task_struct(current);建立内核栈
retval = security_task_create(clone_flags);
retval = sched_fork(clone_flags, p);和调度相关的设置,cpu将调度这个task
retval = copy_thread(clone_flags, stack_start, stack_size, p); 复制父进程堆栈的内容到子进程的堆栈中去.这其中,copy_thread函数中的语句p->thread.ip = (unsigned long) ret_from_fork决定了新进程的第一条指令地址.
建立栈函数dup_task_struct:
tsk = alloc_task_struct_node(node);开辟内存空间
ti = alloc_thread_info_node(tsk, node);ti指向thread_info的首地址,同时也是系统为新进程分配的两个连续页面的首地址。
err = arch_dup_task_struct(tsk, orig);复制父进程的task_struct信息到新的task_struct里
tsk->stack = ti;task对应栈
setup_thread_stack(tsk, orig);初始化thread info结构
set_task_stack_end_magic(tsk);栈结束的地址设置数据为栈结束标示
新进程从哪里开始执行:
在以前的分析中,谈到copy_process中的copy_thread()函数,正是这个函数决定了子进程从系统调用中返回后的执行.ret_from_fork决定了新进程的第一条指令地址。p->thread.ip = (unsigned long) ret_from_fork;将子进程的ip设置为ret_from_fork的首地址,子进程从ret_from_fork开始执行
实验:分析Linux内核建立一个新进程的过程c#
1.Linux经过复制父进程来建立一个新进程,经过调用do_fork来实现。函数
2.Linux为每一个新建立的进程动态地分配一个task_struct结构。线程
3.为了把内核中的全部进程组织起来,Linux提供了几种组织方式,其中哈希表和双向循环链表方式是针对系统中的全部进程(包括内核线程),而运行队列和等待队列是把处于同一状态的进程组织起来。code
4.fork()函数被调用一次,但返回两次。blog