《linux内核原理与分析》第六周作业
这个作业属于哪个课程 | 2020-2021-1 Linux内核原理与分析 |
这个作业要求在哪里 | 2020-2021-1Linux内核原理与分析第六周作业 |
这个作业的目标 | 给MenuOS添加上周实现的系统调用、使用gdb分析sys_time执行过程、了解系统调用执行过程 |
作业正文 | 本博客链接 |
1.分析进程创建的过程
1.1初始化MenuOS
1.2启动gdb,并在各个接口添加断点
1.3开始调试
2.查看详细的fork.c进行分析
查看fork.c的详细代码,可以发现,fork、vfork和clone都可以创建一个新进程,而且都是通过do_fork函数来创建进程的。
查看源码中的do_fork函数,函数有5个参数:
clone_flags:子进程创建相关标志,通过此标志可以对父进程的资源进行有选择的复制
stack_start:子进程用户态堆栈的地址
regs:指向pt_regs结构体的指针
stack_size:用户态站的大小,通常是不必要的,被设置为0
parent_tidptr和child_tidptr:父进程子进程用户台下的pid地址
do_fork主要完成了调用copy_process复制父进程信息、获得pid、调用wake_ip_new_task将子进程加入调度器队列等待获得分配CPU资源运行、通过clone_flags标志做一些辅助工作,其中copy_process是创建一个进程内容的主要代码。接下来分析copy_process函数是如何复制父进程的。
copy_process函数主要完成了调用dup_task_struct复制当前进程(父进程)描述符task_struct、信息检查、初始化、把进程状态设置为TASK_RUNNING(此时子进程置为就绪态)、采用写时复制技术之一复制所有其他进程资源、调用copy_thread初始化子进程内核栈、设置子进程pid等。其中最关键的就是dup_task_struct复制当前进程描述符task_struct和copy_thread初始化子进程内核栈。
duo_task_struct函数中位子进程分配好了内核栈,copy_thread真正完成内核栈关键信息的初始化。
这些工作都完成后,如果创建的是内核线程则从ret_from_kernel_thread执行返回,如果是用户态进程,则从ret_from_frok返回。