_set_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))
这里共输入5个数据(这里没有输出部),%1与gate_addr结合,%2与
(gate_addr+4)结合,它们都是内存单元;先将%%dx的低16位移入%%ax的低16位
(注意%%dx与%%edx的区别)。这样,在%eax中就形成了所需要的中断门的第一个
长整数,其高16位为_KERNEL_CS,而低16位为addr的低16位。接着把第一个输入
立即数(0x8000+(dpl<<13)+(type<<8)装入%%edx的低16位。这样,%%edx中高16
位为addr的高16位,而低16位的P位为1(因为是0x8000),DPL位段为DPL(因为
dpl<<3),而D位加上类型位段则为type(因为type<<8),其余各位皆为0。这就形
成了中断门中的第2个长整数。然后,将%eax 写入*gate_addr,而%%edx写入
*(gate_addr+4)。呵呵,效率高,但难得读懂。但对于像设置IDT表项一类并不
是平凡发生的操作,这样做是否值得牺牲可读性。。。(我想也没有多少人愿意
读这样很底层的东东,更谈不上维护了) |