任何一个通用的 CPU,都具备一种能力,可以在执行完当前正在执行的指令之后,检测到从 CPU 外部发送过来的或内部产生的一种特殊信息,并且可以立即对所接收到的信息进行处理。这种特殊的信息,称为中断信息。CPU 接收到中断信息时,就会转去处理这个中断信息。
中断信息有两种,内中断和外中断。
内中断
当 CPU 内部有如下情况发生时,就会产生中断信息:
- 除法错误,如,执行除法时产生溢出错误
- 单步执行
- 执行 into 指令
- 执行 int 指令
CPU 在收到中断信息后,需要对中断信息进行处理。可是,CPU 是怎么识别中断信息和对中断信息进行相应的处理呢?
CPU 在设计时,对中断信息和处理程序入口地址之间建立了一种联系,中断信息中包含有识别中断源的类型码,中断类型码的作用就是用来定位中断处理程序的。这种关系被放在了一个中断向量表中,CPU根据中断类型码在中断向量表中查找到对应的处理程序入口地址。
对于8086 PC机来说,中断向量表指定放在内存地址 0 处。从内存 0000:0000 到 0000:03FF 的1024 个单元中存放着中断向量表。
程序的入口地址由段地址和偏移地址定位,占用两个字,所以对于一个中断码对应的中断向量表中的数据来说,低地址存放偏移地址,高地址存放段地址。
通过debug可以查看dos中的中断向量表部分如下:
根据中断码,可以知道对应的程序入口在向量表中的位置:
0号中断码:
对应的入口地址在向量表中的偏移地址为0
取两个字单元,低地址为偏移地址,高地址为段地址,即 CS:IP = 0F000H:1060H
所以,获取 CS 和 IP 的值时,IP在向量表中的偏移地址为 0 ,CS在向量表中的偏移地址为 2
1号中断码:
对应的入口地址在向量表中的偏移地址为4
取两个字单元,低地址为偏移地址,高地址为段地址,即 CS:IP = 0192H:08EDH
所以,获取 CS 和 IP 的值时,IP在向量表中的偏移地址为 4 ,CS在向量表中的偏移地址为 6
2号中断码:
对应的入口地址在向量表中的偏移地址为8
取两个字单元,低地址为偏移地址,高地址为段地址,即 CS:IP = 0070H:0008H
所以,获取 CS 和 IP 的值时,IP在向量表中的偏移地址为 8 ,CS在向量表中的偏移地址为 10
3号中断码:
对应的入口地址在向量表中的偏移地址为12,取两个字单元,低地址为偏移地址,高地址为段地址,
取两个字单元,低地址为偏移地址,高地址为段地址,即 CS:IP = 0192H:08E6H
所以,获取 CS 和 IP 的值时,IP在向量表中的偏移地址为 12 ,CS在向量表中的偏移地址为 14
n号中断码:
对应的入口地址在向量表中的偏移地址为 4 * n
取两个字单元,低地址为偏移地址,高地址为段地址
所以,获取 CS 和 IP 的值时,IP在向量表中的偏移地址为 4 * n ,CS在向量表中的偏移地址为 4 * n + 2
CPU 根据中断类型码查找到中断处理逻辑程序并执行后,还需要返回到中断的地方继续执行接下来的程序,所以CPU需要在执行中断程序前保存当前 CS 和 IP 的值,中断过程如下:
- 从中断信息中获取中断类型码 N
- 标志寄存器入栈
- 设置标志位IF =0 和 TF = 0
- CS 内容入栈
- IP 内容入栈
- 从向量表中获取入口地址并设置 IP 和 CS
- 执行中断处理程序
在中断过程中,设计到两个标志位的设置:
TF:
单步中断标志位,当 TF=1 时,允许 CPU 进行单步中断处理,否则不进行对单步中断的处理
产生单步中断的类型码为1,debug中的 T 等指令就是通过单步中断来实现的
该标志位和单步类型中断为我们提供了调试程序的功能
之所以设置TF=0,是为了防止在执行中断的时候,一直响应单步中断指令,导致当前中断无法正常处理完成
对于单步中断,有些指令不会立即响应中断,如
mov sp, ax
mov s