任何一个通用的的CPU,比如8086,都具备一种能力,可以在执行完当前正在执行的指令之后,检测到从cpu外部发来的或内部产生的一种特殊信息,并且可以立即对所接受到的信息进行处理。这种特殊的信息,我们可以称为:中断信息。中断的意思是指,CPU不再接着向下执行,而是转去处理这个特殊信息。
内中段的产生
对于8086CPU,当CPU内部有下面的情况发生的时候,将产生相应的的中断信息。
(1)除法错误,比如,执行div指令产生的除法溢出;
(2)单步执行;
(3)执行into指令;
(4)执行int指令。
8086CPU用称为中断类型码的数据来标识中断信息的来源。中断类型码为一个字节性数据,可以表示256种中断信息的来源。以后,我们将产生中断信息的事件,即中断信息的来源,简称中断源,上述的中断源在8086CPU中的中断类型嘛如下:
(1)除法错误:0
(2)单步执行:1
(3)执行into指令:4
(4)执行int指令,该指令的格式位int n,指令中的n为字节型立即数,是提供给CPU的中断类型码。
我们编写的中断信息处理程序称之为中断处理程序。每个中断信息对应不同的中断处理程序,比如中断源的类型码为4,则对应的中断处理程序也为4。
中断向量表:就是中断处理程序入口地址的列表,中断向量表中存放着256个中断源所对应的中断处理程序的入口。
所以CPU只要知道了中断类型码,就可以将中断类型码作为中断向量表的表项号,定位相应的表项,从而得到中断处理程序的入口地址。
ps:对于8086CPU机,中断向量表的初始地址为0,因为对应的地址应包含段地址和段内偏移量(一共四字节来存储),所以每一个中断源对应4个字节大小,所以中断源和中断若为n则中断向量表中地址为[4n,4n+3],而中断向量表中的地址指向了中断处理函数的首地址。
中断过程
(1)从中断信息中取得中断类型码;
(取得中断类型码N)
(2)标志寄存器的值入栈(因为在中断过程中要改变标志寄存器的值,所以先将其保存在栈中);
(pushf)
(3)设置标志寄存器的第8位TF和第九位IF的值为0;
(TF=0,IF=0)
(4)CS的内容入栈;
(push CS)
(5)IP内容入栈;
(push IP)
(6)从内存地址在中断类型*4和中断类型码*4+2的两个字单元中读取中断处理程序的入口地址设置IP和CS。
((IP)=(N*4),(CS)=(N*4+2) )
iret指令
执行iret指令的功能用汇编语法描述为:
POP IP
POP CS
POPF
iret通常和硬件自动完成的中断过程配合使用。iret指令执行后,CPU回到执行前的执行点继续执行程序。(都是在进行出栈操作,为了方便程序退出)
安装do0用来修改0号中断处理函数
安装程序:
assume cs:code
code segment
start:
mov ax,cs
mov ds,ax
mov si,offset do0 ;设置ds:si指向源地址(do0子程序的地址,即中断处理程序的地址)
mov ax,0
mov es,ax
mov di,200h
mov cx,offset do0end - offset do0
cld
rep movsb ;将do0子程序整体复制到已经确定地址的内存空间中,以便所有的程序使用
mov ax,4c00h
int 21h
do0: ;子程序,中断处理函数
jmp short do0start
db "divide error!"
do0start:
mov ax,cs
mov ds,ax
mov si,202h
mov ax,0b800h
mov es,ax
mov di,160*12+36*2
mov cx,13
s:
mov al,[si]
mov es:[di],al
mov byte ptr es:[di+1],01000010b
inc si
add di,2
loop s
mov ax,4c00h
int 21h
do0end:
nop
code ends
end start
assume cs:code
code segment
start:
mov ax,0
mov es,ax
mov word ptr es:[0*4],200h ;将该地址的子程序设置为0号中断处理函数
mov word ptr es:[0*4+2],0
int 0
mov ax,4c00h
int 21h
code ends
end start
单步中断
CPU在执行一条指令之后,如果检测到标志寄存器的TF位为1,则产生单步中断,引发中断过程。单步中断的中断类型码为1,则它所引发的中断过程如下:
(1)取得中断类型码1;
(2)标志寄存器入栈,TF、IF设置为0
(3)CS,IP入栈
(4)(IP)=(1*4),(CS)=(1*4+2)
eg:在debug中用单步中断实现了t命令的实现,用t命令执行完一条命令后立即中断并显示各个寄存器的信息。
因为每次运行指令后都会检查TF是否为0去判断是否进入单步中断。而每次TF设置为0,就是为了防止重复的触发单步中断。
相应中断的特殊情况
有些情况下,CPU在完成当前指令后,即便是发生中断,也不会响应。
ss:sp联合指向栈顶,而对他们的设置应该连续完成(在两条指令中中断不会被响应)。
mov ss,ax
mov sp,idata
两条指令中间不响应中断,而t命令依靠单步中断实现显示各寄存器信息,所以不相应单步中断则在t命令调试时,遇到上述指令则在mov ss,ax后不显示寄存器信息。