(1)分析上面 int 9 中断例程,看看是否可以精简?
在 int 9 中断例程中,模拟 int 指令调用原 int 9 中断例程的程序段是可以精简的,因为在进入中断例程后,IF 和 TF 都已经置 0,没有必要再设置。对于程序段:
pushf
pushf
pop ax
and ah,11111100b
push ax
popf
call dword ptr ds:[0]
因为 int 指令的其中一个步骤是将 IF 和 TF 置零,所以中间将 IF 和 TF 置零的部分为重复操作。可将上述代码精简为
pushf
call dword ptr ds:[0]
(2)分析上述程序的主程序部分,看看有什么潜在问题?
在主程序中,如果在设置 int 9 中断例程的段地址和偏移地址之间发生了键盘中断,则 CPU 将转去一个错误的地址执行,将发生错误。找出这样的程序段,并改写它们。
相关代码部分为设置int9中断新地址时
mov word ptr es:[9*4],offset int9
;在中断向量表中设置新的int 9中断例程的偏移地址
------------ 发生键盘中断,导致段地址设置错误 ------------
mov es:[9*4+2],cs
;在中断向量表中设置新的int 9中断例程的段地址
和恢复原来int9的地址时
push ds:[0]
pop es:[9*4]
------------ 发生键盘中断,导致段地址设置错误 ------------
push ds:[2]
pop es:[9*4+2] ;恢复原int 9中断例程的地址
来自键盘的中断是一种可屏蔽中断,CPU 根据标志寄存器 IF 的值来决定是否响应该中断,如果为 1 则响应,为 0 则不响应。
在汇编语言中,cli 指令表示不可发生中断,sti 表示可发生中断。在上述设置地址前加上 cli,设置地址后加上 sti 即可避免因中断造成的地址设置错误。
如下图代码中所示
cli ;关可屏蔽中断
mov word ptr es:[9*4],offset int9
;在中断向量表中设置新的int 9中断例程的偏移地址
------------ 发生键盘中断,导致段地址设置错误 ------------
mov es:[9*4+2],cs
;在中断向量表中设置新的int 9中断例程的段地址
sti; ;开可屏蔽中断
cli ;关可屏蔽中断
push ds:[0]
pop es:[9*4]
------------ 发生键盘中断,导致段地址设置错误 ------------
push ds:[2]
pop es:[9*4+2] ;恢复原int 9中断例程的地址
sti ;开可屏蔽中断