linux-0.11分析:init文件 main.c的第二个初始化函数trap_init() 第五篇随笔
阅读原文时间:2023年07月09日阅读:1

2、第二个初始化函数,trap_init()


参考 [github这个博主的 厉害][ https://github.com/sunym1993/flash-linux0.11-talk ]

kernel文件夹 -> traps.c

void trap_init(void)
{
    int i;
    set_trap_gate(0,&divide_error);
    set_trap_gate(1,&debug);
    set_trap_gate(2,&nmi);
    set_system_gate(3,&int3);   /* int3-5 can be called from all */
    set_system_gate(4,&overflow);
    set_system_gate(5,&bounds);
    set_trap_gate(6,&invalid_op);
    set_trap_gate(7,&device_not_available);
    set_trap_gate(8,&double_fault);
    set_trap_gate(9,&coprocessor_segment_overrun);
    set_trap_gate(10,&invalid_TSS);
    set_trap_gate(11,&segment_not_present);
    set_trap_gate(12,&stack_segment);
    set_trap_gate(13,&general_protection);
    set_trap_gate(14,&page_fault);
    set_trap_gate(15,&reserved);
    set_trap_gate(16,&coprocessor_error);
    for (i=17;i<48;i++)
        set_trap_gate(i,&reserved);
    set_trap_gate(45,&irq13);
    outb_p(inb_p(0x21)&0xfb,0x21);
    outb(inb_p(0xA1)&0xdf,0xA1);
    set_trap_gate(39,&parallel_interrupt);
}

大部分都是两类set_xxx_gate函数 set_trap_gate、set_system_gate、

先看看这两个函数的宏定义的位置:

include文件 -> asm文件 -> system.h
#define _set_gate(gate_addr,type,dpl,addr) \
__asm__ ("movw %%dx,%%ax\n\t" \
    "movw %0,%%dx\n\t" \
    "movl %%eax,%1\n\t" \
    "movl %%edx,%2" \
    : \
    : "i" ((short) (0x8000+(dpl<<13)+(type<<8))), \
    "o" (*((char *) (gate_addr))), \
    "o" (*(4+(char *) (gate_addr))), \
    "d" ((char *) (addr)),"a" (0x00080000))

#define set_intr_gate(n,addr) \
    _set_gate(&idt[n],14,0,addr)

#define set_trap_gate(n,addr) \
    _set_gate(&idt[n],15,0,addr)

#define set_system_gate(n,addr) \
    _set_gate(&idt[n],15,3,addr)

可以看出:set_trap_gate、set_system_gate宏定义引用了同一个宏定义:_set_gate

可以看出他们两个只是参数的不同:第三参数:set_trap_gate它0;set_system_gate它3;

而这个函数_set_gate就在上面,可惜看不懂;就参考了别人文章,了解大致干了啥;但是大概可以看出是一段汇编代码;

大佬分析是就是把汇编代码嵌入到c语言中;确实很厉害

这个trap_init是在初始哈一些中断信息

set_trap_gate(0,&divide_error);
......
set_system_gate(3,&int3);    /* int3-5 can be called from all */
set_system_gate(4,&overflow);
set_system_gate(5,&bounds);
......
set_trap_gate(16,&coprocessor_error);

可以看出它先把 0-16号中断,分别对应一个函数地址,其中3~5是系统中断,那么trap和system中断有什么差别了,差别就是级别不一样,system是0级别高于trap是3;

for (i=17;i<48;i++)
        set_trap_gate(i,&reserved);

接着这个循环可以看出它把,17~47初始化了同一个中断reserved,这个中断是没有写的,留到后面更改

set_trap_gate(45,&irq13);
......
set_trap_gate(39,&parallel_interrupt);

然后就是把中断45为irq13;中断39位parallel_interrupt

看到这个长度48是不是想到了前面的中断描述符表idt,先看看这个表的位置吧

这段代码的目的就是把这个中断描述符插入到idt表中,看看插入中断标识符后的样子吧

这个时候有一种中断是键盘中断,是int 0x21 前面可以看出21还没有设置中断信息,所以这个时候键盘是没有办法输入的;这用到的中断后面再说

手机扫一扫

移动阅读更方便

阿里云服务器
腾讯云服务器
七牛云服务器

你可能感兴趣的文章