linux-0.11分析:init文件 main.c的第二个初始化函数trap_init() 第五篇随笔

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还没有设置中断信息,所以这个时候键盘是没有办法输入的;这用到的中断后面再说

posted @ 2022-08-06 10:35  水三丫  阅读(250)  评论(0编辑  收藏  举报