linux 中断相关的几个问题

一 linux实现的几个门

intel 提供了三种类型的中断描述符:任务门,中断门,陷阱门。linux稍有不同,根据intel的定义,实现了一下几类门。

1 中断门

用户态的进程不能访问的一个intel中断门,DPL = 0。

2 系统门

用户态的进程可以访问的一个intel陷阱门,DPL = 3, 通过系统门可以激活linux下的三个异常:4,5,128(即0x80)。

3 系统中断门

用户态的进程可以访问的一个intel中断门,DPL = 3,中断异常就属于系统中断门,int 0x03。

4 陷阱门

用户态的进程不能够访问的intel陷阱门,DPL = 0.

5 任务门

不能被用户态进程访问的intel任务门, DPL = 0。

 

二 中断处理函数数组 interrupt[i]

在entry_32.S中的下面的汇编代码,声明和初始化了interrupt数组,有几个细节值得分析。

 /*interrupt数组的声明和初始化,里面存储中断处理函数的地址*/

/*interrupt数组就像c语言的数组一样,属于数据段的东西,因此在数据段声明。但是对数组的初始化就是代码段的内容了,这里需要注意*/
.data
ENTRY(interrupt)
.text

ENTRY(irq_entries_start)
 RING0_INT_FRAME
vector=0
.rept NR_IRQS
 ALIGN
 .if vector
 CFI_ADJUST_CFA_OFFSET -4
 .endif
1: pushl $~(vector)
 CFI_ADJUST_CFA_OFFSET 4
 jmp common_interrupt
 .previous      /*previous 知名下面内容上接上一个段,即data段*/
 .long 1b
 .text
vector=vector+1
.endr
END(irq_entries_start)

.previous
END(interrupt) /*这句是数据段的内容*/
.previous 

 

可以总结为:

上面一段汇编首先在数据段声明了一个interrupt数组,如下面代码:

.data
ENTRY(interrupt)

 .long 1b

END(interrupt)

数组中每个元素的初始值是标号1的地址。因此访问数组中的元素时,都会跳到标号1处,执行相应的指令。

 

还有一个细节问题:在i8259_32.c中调用函数set_intr_gate(vector, interrupt[i]);

这里引用了在entry_32.S中数据段中定义的全局的interrupt数组,但是,数据段中定义的这个intterrupt数组只能用在连接的时候,这个符号是已经定义过得,至于它的类型和大小等c编译器无法确定,因此,在hw_irq_32.h中重新声明了extern void (*interrupt[NR_IRQS])(void),这样c编译器才会确定这个数组的类型和大小。

 

你可能感兴趣的:(linux,汇编,vector,语言,任务,编译器)