fork实际开销是 复制父进程页表以及给子进程创建唯一的task_struct;
fork ->clone->do_fork
1、dup_task_struuct创建内核栈、thread_info struct、task_struct 值与父进程完全相同;
2、check进程数目没有超过限制
3、清除一些描述符,以区别父子进程
4、状态设置为task_uninterruptable 保证不会运行
5、获取pid
6、根据参数表示 拷贝或者共享打开文件、信号hanlder、进程地址空间、命名空间等。
7、父子进程平分剩余时间片
8、返回一个指向子进程的指针
9、被唤醒投入运行;内核选择子进程首先执行,以避免copy开销。
内核线程 与普通进程区别:没有独立的地址空间。
进程的终结:
1.task_struct 标志设置为pf_exiting
2、放弃进程占用的mm_struct,如果没有进程使用它,就彻底释放
3、如果进程在排队IPC 就退出
4、设置退出code 供父进程查询
5、Exit_notify向父进程发送信号,处于task_zomble状态;现在占用内核栈、thread_info、task_struct,父进程检索到后,就释放其余内存,归还系统。
父进程的wat:
等待到会调用release_task:
1、减少进程引用计数。linux用用户高速缓存统计和记录用户进程占用的进程数据和文件数目,如果都为0就销毁缓存
2、从idhash上删除该进程,也从task_list上删除
3、释放内核栈和thread_info所占的页,并释放task_struct所占的slab高速缓存