深度解析Linux内核中fork工作原理和实现

2024-06-07 10:52

本文主要是介绍深度解析Linux内核中fork工作原理和实现,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

      Linux内核中的fork()系统调用是用来创建新进程的核心机制。它的主要工作是为新创建的子进程复制当前进程(父进程)的数据结构和内存空间,从而产生一个几乎完全相同的副本。fork()的实现涉及到操作系统内核中许多重要部分的交互和协作,过程比较复杂。

fork()的基本原理

      当一个进程调用fork()时,内核会为新创建的子进程分配必要的资源,如进程控制块(task_struct)、内存空间等。然后,内核会将当前进程的数据(代码段、数据段、堆、栈等)复制到新进程中,使得父子进程之间存在相同的内存映像。在复制结束后,内核会为子进程创建新的独立执行环境,并决定父子进程谁先获得CPU执行权。

需要注意的是,为了提高效率,Linux采用了写时复制(Copy-on-Write,COW)技术。真正的复制工作并不是在fork()时进行,而是延迟到父子进程中有一方第一次试图修改某个内存页时才执行。这个技术避免了大量不必要的数据复制,从而提高了fork()的性能。

进程的关键数据结构

      理解fork()的实现,需要先了解Linux进程相关的重要数据结构,下图描述的是进程相关的主要数据结构。

图 1  Linux进程相关数据结构图

  • - 进程描述符task_struct:每个进程都在内核中有一个对应的task_struct结构体,它记录了该进程的所有相关信息,如进程id、状态、内存映射区域、文件描述符表等。
  • - 进程内存管理描述符mm_struct: 管理每个进程的虚拟内存和物理内存。
  • - 虚拟内存描述符vm_area_struct:描述一个进程的虚拟内存区域,包括起始和结束地址、访问权限、映射的物理页框号等信息。

fork()的实现过程

Linux内核中fork()系统调用的实现过程具体如下。

  • 进入内核态

当一个进程在用户态调用fork()时,会通过软件中断(如int 0x80或sysenter指令)进入内核态,进入内核的sys_fork()系统调用处理函数。

  • 获取新进程ID(PID)和创建进程描述符task_struct

内核会先获取一个可用的PID,作为新创建的子进程的进程ID。接下来,内核会调用copy_process()函数,为子进程分配和初始化一个新的进程描述符task_struct。copy_process()主要完成以下工作:

- 为task_struct分配内核内存

- 从父进程的task_struct复制大部分内容,包括文件系统相关数据(如打开的文件描述符表)、信号处理函数表、命名空间、进程状态等

- 设置新进程的状态为TASK_UNINTERRUPTIBLE(不可中断睡眠状态)

- 为新进程分配一个独立的内核栈

- 初始化计时器、信号等数据结构

  • 复制虚拟内存映射区域vm_area_struct

在copy_process()中,会调用dup_mmap()函数来复制父进程的内存映射区域。dup_mmap()会遍历父进程的所有vm_area_struct,并为子进程创建相应的内存映射区域,但这时只是简单地让父子进程共享同一组页表项,实际的物理内存页还未复制。

  • 写时复制(Copy-on-Write)设置

在复制完vm_area_struct后,dup_mmap()会调用pud_mkwrite等函数,将父子进程共享的所有页表项都标记为只读(设置页表项的权限位为非可写)。这是为了启用写时复制机制,当父子进程中有一方试图写入共享的内存页时,CPU会触发页保护异常,从而引发内核的异常处理程序执行写时复制操作。

  • 信号处理程序设置

在copy_process()中,会复制父进程的信号处理程序表,确保子进程也能正确响应不同的信号。另外,还会为子进程设置SIGCHLD信号的默认处理程序,以便父进程能够捕获子进程的结束信号。

  • 内核线程设置

如果新创建的进程是一个内核线程,copy_process()会进行一些额外的设置,如禁止内核线程加载执行用户空间代码、禁止访问用户态内存等。

  • 调度策略设置

copy_process()会复制父进程的调度策略、优先级等相关信息,并为子进程分配新的运行时统计数据结构,用于CPU调度。

  • 进程链表挂载

新创建的子进程会被加入相应的进程链表中,如任务队列、反馈优先级链表等,以便内核进行进程调度和管理。

  • 写时复制异常处理

在完成上述所有设置后,进入copy_process()的最后阶段。此时,内核会设置写时复制异常处理程序,以响应子进程对共享内存区域的写操作而发生的页保护异常。

写时复制异常处理程序do_cow_fault()的主要工作是:

- 为发生写操作的内存页分配新的内核页框(物理内存页)

- 将原有的内存页内容复制到新的页框中

- 修改相应的页表项,使其指向新分配的物理内存页框,并设置为可写

- 在原有的物理内存页上设置写保护,避免不必要的复制

通过写时复制机制,父子进程最终会拥有各自独立的物理内存副本,从而可以进行自身的数据写入而不会相互影响。

  • 执行切换和系统调用返回

      最后,内核会决定父进程和子进程的执行顺序。一般情况下,内核会先让子进程执行,因为子进程的执行状态被设置为TASK_UNINTERRUPTIBLE。在子进程执行时,会执行一些额外的初始化工作,如清理上下文、设置执行计数器等。

fork()系统调用在父子进程中的返回值不同:

- 在子进程中,fork()返回0

- 在父进程中,fork()返回新创建子进程的PID

通过这种方式,父子进程可以区分不同的执行路径。

结论

      Linux内核中fork()系统调用的实现过程十分复杂,需要内核中多个子系统的通力合作,包括进程管理、内存管理、CPU调度等。写时复制(COW)技术的应用是fork()实现中的一大亮点,极大地提高了系统性能。在现代操作系统中,fork()作为创建新进程的核心机制,对于理解内核原理有着非常重要的意义。

这篇关于深度解析Linux内核中fork工作原理和实现的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1038978

相关文章

Springboot @Autowired和@Resource的区别解析

《Springboot@Autowired和@Resource的区别解析》@Resource是JDK提供的注解,只是Spring在实现上提供了这个注解的功能支持,本文给大家介绍Springboot@... 目录【一】定义【1】@Autowired【2】@Resource【二】区别【1】包含的属性不同【2】@

Java枚举类实现Key-Value映射的多种实现方式

《Java枚举类实现Key-Value映射的多种实现方式》在Java开发中,枚举(Enum)是一种特殊的类,本文将详细介绍Java枚举类实现key-value映射的多种方式,有需要的小伙伴可以根据需要... 目录前言一、基础实现方式1.1 为枚举添加属性和构造方法二、http://www.cppcns.co

使用Python实现快速搭建本地HTTP服务器

《使用Python实现快速搭建本地HTTP服务器》:本文主要介绍如何使用Python快速搭建本地HTTP服务器,轻松实现一键HTTP文件共享,同时结合二维码技术,让访问更简单,感兴趣的小伙伴可以了... 目录1. 概述2. 快速搭建 HTTP 文件共享服务2.1 核心思路2.2 代码实现2.3 代码解读3.

Linux系统中卸载与安装JDK的详细教程

《Linux系统中卸载与安装JDK的详细教程》本文详细介绍了如何在Linux系统中通过Xshell和Xftp工具连接与传输文件,然后进行JDK的安装与卸载,安装步骤包括连接Linux、传输JDK安装包... 目录1、卸载1.1 linux删除自带的JDK1.2 Linux上卸载自己安装的JDK2、安装2.1

MySQL双主搭建+keepalived高可用的实现

《MySQL双主搭建+keepalived高可用的实现》本文主要介绍了MySQL双主搭建+keepalived高可用的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,... 目录一、测试环境准备二、主从搭建1.创建复制用户2.创建复制关系3.开启复制,确认复制是否成功4.同

Java实现文件图片的预览和下载功能

《Java实现文件图片的预览和下载功能》这篇文章主要为大家详细介绍了如何使用Java实现文件图片的预览和下载功能,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... Java实现文件(图片)的预览和下载 @ApiOperation("访问文件") @GetMapping("

SpringCloud动态配置注解@RefreshScope与@Component的深度解析

《SpringCloud动态配置注解@RefreshScope与@Component的深度解析》在现代微服务架构中,动态配置管理是一个关键需求,本文将为大家介绍SpringCloud中相关的注解@Re... 目录引言1. @RefreshScope 的作用与原理1.1 什么是 @RefreshScope1.

Java并发编程必备之Synchronized关键字深入解析

《Java并发编程必备之Synchronized关键字深入解析》本文我们深入探索了Java中的Synchronized关键字,包括其互斥性和可重入性的特性,文章详细介绍了Synchronized的三种... 目录一、前言二、Synchronized关键字2.1 Synchronized的特性1. 互斥2.

使用Sentinel自定义返回和实现区分来源方式

《使用Sentinel自定义返回和实现区分来源方式》:本文主要介绍使用Sentinel自定义返回和实现区分来源方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录Sentinel自定义返回和实现区分来源1. 自定义错误返回2. 实现区分来源总结Sentinel自定

Linux卸载自带jdk并安装新jdk版本的图文教程

《Linux卸载自带jdk并安装新jdk版本的图文教程》在Linux系统中,有时需要卸载预装的OpenJDK并安装特定版本的JDK,例如JDK1.8,所以本文给大家详细介绍了Linux卸载自带jdk并... 目录Ⅰ、卸载自带jdkⅡ、安装新版jdkⅠ、卸载自带jdk1、输入命令查看旧jdkrpm -qa