前言
想了一想,还是需要先导知识,一开始就读源码,实在是有点困难。
从内存管理开始吧,毕竟属于相对比较简单的一部分。
知识
内存分页管理机制
Linux的内存管理程序采用了分页管理方式。与内存最主要的初始化是在引导的时候完成的,在引导时完成了从实模式到保护模式的转换。内存分页管理是通过页面目录表和内存页表多组成的二级表进行。页目录表中的每一个表项(简称页目录项)(4B)用来寻址一个页表,而每个页表项(4B)用来指定一页物理内存页。因此,当指定了一个页目录项和一个页表项,可以唯一确定所对应的物理内存页。
页目录表占一页内存,因此最多可以寻址1024个页表。(这里不懂)
每个页表也同样占用一页内存,因此一个页表最多可以寻址1024个物理内存页面。(也不懂)
在80386中,一个页目录表所寻址的所有页表共可以寻址1024 x 1024 x 4096B = 4GB的内存空间。
所有进程都使用同一个页目录表,而每个进程都有自己的页表。
进程或内核其他部分在申请内存时使用的是线性地址。32位的线性地址被分成三个部分,分别用来指定一个页目录项,一个页表项和对应物理内存页上的偏移地址,因此可以间接地寻址到线性地址指定的物理内存位置。
ps:对应物理内存存在页框中。
写时复制机制
下节转载自写时复制
fork 函数,当一个应用程序调用fork 函数来创建另一个进程时,新进程所做的第一件事是调用exec 函数,用一个可执行程序来重新初始化它的地址空间。 在fork中,新进程不是拷贝整个地址空间,而是通过将页面标记为写时复制的方式,与父进程共享这些页面。如果子进程在这些页面中写入数据了,则生成一份进程私有的拷贝。如果没有写操作,则2个进程继续共享页面,不会执行拷贝动作。不管怎么样,内存管理器只拷贝一个进程试图要写入数据的那些页面,而不是整个地址空间。