如何设置x86分页?获得三重故障错误

时间:2011-08-24 22:16:07

标签: assembly x86 kernel bochs

我有一个玩具内核,我正在使用x86在bochs上运行。当我启用分页时,bochs会重置三故障错误。似乎每次和任何内存访问都会触发错误。所以,我假设我在设置分页时遇到错误,问题不在于我的中断处理程序。这是代码。

paging.c

#include "paging.h"
#include "lib.h"
#include "screen.h"
#include "descriptor_tables.h"

typedef struct page_dir_entry_s{
    bool present:1;
    bool writeable:1;
    bool user_access:1;
    bool write_through:1;
    bool cache_disabled:1;
    bool accessed:1;
    bool unused0:1;
    bool use_mb:1;//makes pages 4MB not 4KB
    bool unused1:4;
    u32 frame:20;
} page_dir_entry_t;

typedef struct page_table_entry_s{
    bool present:1;
    bool writeable:1;
    bool user_access:1;
    bool write_through:1;
    bool cache_disabled:1;
    bool accessed:1;
    bool dirty:1;
    bool unused0:1;
    bool global:1;
    bool unused1:3;
    u32 phys_page_addr:20;
} page_table_entry_t;

extern u32 end;//as declared in the linker script

static u32 next_addr=0;
static page_dir_entry_t* page_dir=NULL;
static page_table_entry_t* page_table=NULL;

extern void enable_paging(u32);

void InitPaging(){
    next_addr=end;
    while((next_addr%4096)!=0)
        ++next_addr;
    page_dir=(void*)next_addr;
    next_addr+=4*1024;
    memset(page_dir,0,4*1024);
    page_table=(void*)next_addr;
    next_addr+=4;
    u32 addr=0;
    u32 i=0;
    *(((u32*)page_table)+i)=0;//zero it out
    while(addr<next_addr){
        page_table[i].present=true;
        page_table[i].writeable=true;
        page_table[i].phys_page_addr=addr;
        ++i;
        *(((u32*)page_table)+i)=0;//zero it out
        addr+=(1024*4);//4KB
        next_addr+=4;
    }

    page_dir[0].writeable=true;
    page_dir[0].present=true;
    page_dir[0].frame=(u32)page_table;

    enable_paging((u32)page_dir);
}

paging_asm.s

[global enable_paging]
enable_paging:
    mov eax,[esp+4]
    mov cr3,eax
    mov eax,cr0
    or eax,0x80000000
    mov cr0,eax
    ret

1 个答案:

答案 0 :(得分:1)

“frame”和“phys_page_addr”字段是物理地址的第32到12位(在此分页模式下)。

分页对偏移量(0 - 4K)没有任何作用。

至少,您需要:

page_table[i].phys_page_addr=addr >> 12;

page_dir[0].frame=((u32)page_table) >> 12;

由于'addr'和'page_table'都与4096对齐,因此只删除了额外的零。