MIT6.828_LAB4_PartB_Copy-on-Write Fork

Part B: Copy-on-Write Fork

正如先前所提到的,Unix提供了fork()系统调用作为创建进程原语,fork()会将父进程的地址空间复制来创建一个子进程。
xv6 Unix通过将父进程所有页的数据复制到新分配给子进程的页上实现了fork(),实际上dumbfork()中也采用了相同的方法,复制父进程的地址空间到子进程是fork()操作中开销最大的部分。
但是,在子进程中fork()之后通常紧接着是对exec()的调用,新程序会替换掉子进程的内存。 这就是shell程序通常所做的事情。 在这种情况下,复制父进程地址空间所花费的时间被大大浪费了,因为子进程在调用exec()之前只会使用很少的内存。
因此,Unix之后的版本利用虚拟内存硬件来允许父进程和子进程共享映射到其各自地址空间的内存,直到其中一个进程实际对其进行修改为止。这种技术称为copy-on-write。为此,内核将在fork()上将地址空间映射从父进程复制到子进程,而不是将实际页面的内容复制过去,同时将当前共享的页面标记为只读。当两个进程之一尝试写入这些共享页面之一时,该进程将导致页面错误。此时,Unix内核意识到该页面实际上是“virtual”副本或“copy-on-write”副本,因此它为导致错误的进程创建了新的,私有的,可写的副本。这样,各个页面的内容实际上不会被复制,直到它们被实际写入为止。这种优化使fork()之后调用exec()的子进程不必要的开销变小了许多:子进程在调用exec()之前可能只需要复制一页(其堆栈的当前页)。
在本实验的下一部分中,我们将实现一个具有copy-on-write功能的类unix fork(),作为一个用户空间库例程,在用户空间实现fork()和copy-on-write有助于保持内核的精简性和正确性,它还允许单个用户程序为fork()定义自己的语义。 想要一个稍有不同的fork()的程序(例如,类似dumbfork()的完全复制版本,或者父子进程间共享内存的版本)可以轻松地进行修改。

User-level page fault handling

用户级别的copy-on-write fork()需要知道受写保护的页面上的页面错误,因此这是首先要实现的。 copy-on-write只是用户级页面错误处理的多种可能用途之一。
设置地址空间是很常见的,以便页面错误指示何时需要执行某些操作。 例如,大多数Unix内核最初只在新进程的堆栈区域中映射单个页面,然后在该进程的堆栈消耗增加时“按需”分配和映射其他堆栈页面,并在尚未映射的堆栈地址上引起页面错误。 典型的Unix内核必须跟踪在进程空间的每个区域中发生页面错误时应采取的措施。 例如,堆栈区域中的故障通常将分配并映射新的物理内存页面。 程序的BSS区域中的故障通常会分配一个新页面,并用零填充并映射它。 在具有按需分页的可执行文件的系统中,文本区域中的错误将从磁盘读取二进制文件的相应页面,然后将其映射。
这是内核要跟踪的很多信息。 我们无需采取传统的Unix方法,而可以决定如何处理用户空间中每个页面错误(这些错误对破坏性的影响较小)。 这种设计的附加好处是允许程序在定义其内存区域时具有极大的灵活性。 我们稍后将使用用户级页面错误处理来映射和访问基于磁盘的文件系统上的文件。
为了处理自己的页面错误,用户进程需要在JOS内核中注册页面错误处理程序入口点。 用户进程通过新的sys_env_set_pgfault_upcall系统调用注册其页面错误入口点。 我们在Env结构中添加了一个新成员env_pgfault_upcall,以记录此信息。
练习8.实现sys_env_set_pgfault_upcall系统调用。 查找目标进程的进程ID时,请确保启用权限检查,因为这是“危险的”系统调用。
sys_env_set_pgfault_upcall:
在这里插入图片描述

别忘了在syscall函数中加上这一行:

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 在xv6中,copy-on-write fork是一种优化技术,它可以在子进程创建时避免不必要的内存复制。具体来说,当父进程调用fork()创建子进程时,子进程会共享父进程的内存页表,而不是复制一份父进程的内存。只有当子进程尝试修改共享的内存时,才会发生实际的复制操作。这种技术可以减少内存使用和复制时间,提高系统性能。 ### 回答2: xv6是一个操作系统教学项目,这是一个现代化风格的UNIX第六版。copy-on-write fork是xv6中实现的一种机制,它与fork系统调用有关。这种机制可减少在进行进程复制时所涉及的空间和时间开销,从而增加操作系统的效率。 在fork系统调用中,操作系统会复制原始进程,创建一个独立的进程。传统方法是,操作系统会将原有进程的内存空间全部复制一份给新进程,并在新进程中对地址进行修正。这样做会消耗大量的空间和时间,尤其是当进程较大时,复制整个内存空间会非常耗时。 copy-on-write fork的实现与传统方法不同。当原始进程需要创建新进程时,操作系统会将进程的内存空间标记为只读状态,并保留原内存页的映射关系。这样,当进程尝试写入内存时,操作系统将会产生一个缺页异常。在此时,操作系统会创建一个新页,将原内存页的内容复制到新页中,并在新页上进行写入操作。这样可以减少空间和时间开销,因为新页仅在需要写入时被复制,而不是在进程创建时。 copy-on-write fork有许多优点。首先,这种机制使系统更高效。使用copy-on-write fork可以显著降低进程复制的时间和空间开销。其次,这种机制还可以提高系统的可扩展性。当进程需要更多内存时,操作系统会重新映射新的内存页,而不是将整个进程复制一次。因此,系统可以更轻松地扩展。 总之,copy-on-write fork是xv6中非常有用的一个机制。它可以减少进程复制所需的时间和空间开销,从而提高操作系统的效率和可扩展性。 ### 回答3: 在操作系统课程xv6中,实现了一种名为“copy-on-write fork”的操作,这种操作可以让父进程和子进程在初始时共享相同的物理内存。当父进程或子进程试图修改内存时,内存页会被复制并分配新的物理内存,以避免父进程和子进程之间的竞争条件。 这种“copy-on-write”技术可以减少系统中的内存浪费,并且在分配内存时减少了复制操作,从而提高了系统的性能。在实现中,当父进程调用fork()创建一个新的子进程时,子进程将直接引用父进程的地址空间。父进程和子进程都共享相同的物理内存,但是它们各自有自己的页目录和页表来管理地址空间和虚拟内存。 当父进程或子进程尝试读取数据时,它们可以访问共享的物理内存。然而,当父进程或子进程试图修改数据时,操作系统会将所涉及的内存页复制到另一个物理内存地址,并使涉及的进程引用新的物理内存地址。这样,父进程和子进程将各自拥有自己的数据副本,一个进程修改数据不会影响另一个进程。 这种技术在许多操作系统中都有广泛应用,因为它可以提供更高效的内存管理和更好的性能。实现“copy-on-writefork操作在操作系统课程中具有教育意义,因为它可以让学生更深入了解xv6的内部机制和操作系统的基本理论。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值