4.2:内存管理-重看fork

1:建立段表

nr为进程号,所以每个进程都会分割64M的虚拟内存使用,且不会重叠
set_base很明显可以看出是在设置段表

copy_process->copy_mem
        new_data_base = new_code_base = nr * 0x4000000;
        p->start_code = new_code_base;
        set_base(p->ldt[1],new_code_base);
        set_base(p->ldt[2],new_data_base);

2:建立页表

将父进程的虚拟内存和内存页框的映射关系复制给子进程。

old_data_base = get_base(current->ldt[2]);
copy_page_tables(old_data_base,new_data_base,data_limit)

通过虚拟地址偏移22位得到页目录号,每个页目录项长位4字节,所以*4就是页目录项的位置。再加上页目录表的起始地址就是页表的位置,页目录表起始地址为0。
找到页表后,通过for循环将所有的映射关系都copy过来,就完成了子进程的页表建立。

int copy_page_tables(unsigned long from,unsigned long to,long size)
{ 
....
        from_dir = (unsigned long *) ((from>>20) & 0xffc); /* _pg_dir = 0 */
        to_dir = (unsigned long *) ((to>>20) & 0xffc);
        size = ((unsigned) (size+0x3fffff)) >> 22;
        for( ; size-->0 ; from_dir++,to_dir++) {

## 
...
	}
}

3:写时复制

现在父子进程使用相同的物理内存,那么就存在一个问题,父进程在执行p=7后,再执行fork,当子进程如果执行p=8,由于使用相同的物理地址,那么会不会使得父进程的*p也等于8呢?
当然是不会,详细的操作就是在fork时将该内存设置为只可读不可写,当要写内存时就会出发读写异常中断,中断的结果就是分配一个新的内存页给子进程。这就是写时复制。

void un_wp_page(unsigned long * table_entry)
{
        unsigned long old_page,new_page;

        old_page = 0xfffff000 & *table_entry;
        if (old_page >= LOW_MEM && mem_map[MAP_NR(old_page)]==1) {
                *table_entry |= 2;
                invalidate();
                return;
        }
        if (!(new_page=get_free_page()))	//申请一个新的页面
                oom();
        if (old_page >= LOW_MEM)
                mem_map[MAP_NR(old_page)]--;
        *table_entry = new_page | 7;	//改变页面属性
        invalidate();
        copy_page(old_page,new_page);	//复制内容
}
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值