第十五章 外中断
CPU除了有运算能力,还有I/O能力
15.1 接口芯片和端口
1.在PC系统的接口卡和主板上,装有各种接口芯片,这些外设接口芯片的内部装有若干寄存器
CPU将这些寄存器当做【端口】访问
2.外设的输入不直接送入内存和CPU,而是送入相关的接口芯片的【端口】中
3.CPU向外设的输出也是要先送入【端口】中,再由相关芯片送入到外设
4.CPU可以向外设输出控制命令,这些控制命令也是先送到【端口】中,然后相关芯片根据命令进行相关工作
5.可见:CPU与外部设备的交流是通过【端口】进行的
CPU在执行完当前指令后,可以检测到发送过来的中断信息,引发中断过程,处理外设的输入
15.2 外中断信息
1.在PC系统中,外中断源一共有两类
1.可屏蔽中断
2.不可屏蔽中断
2.可屏蔽中断是CPU可以不响应的外中断。CPU是否响应可屏蔽中断
要看标志寄存器的IF位的设置
当CPU检测到可屏蔽中断信息时:
1.若IF=1,则CPU在执行完当前指令后相应中断,引发中断过程
2.若IF=0,则不响应可屏蔽中断
3.可屏蔽中断所引发的中断过程,除在第一步的实现上与内中断有所不同外,基本上和内中断的中断过程相同
4.因为可屏蔽中断信息来自于CPU外部,中断类型码是通过数据总线送入CPU的
而内中断的中断码是在CPU内部产生的
5.IF设置为0的原因:在进入中断处理程序后,禁止其他的可屏蔽中断
当然,如果中断处理程序中需要处理可屏蔽中断,可以用指令将IF设置为1
6.8086CPU提供的设置IF的指令如下:
sti ;用于设置IF=1
cli ;用于设置IF=0
7.不可屏蔽中断是CPU必须相应的外中断。
当CPU检测到不可屏蔽中断信息时,则在执行完当前指令后
立即响应,应发中断过程
8.8086CPU不可屏蔽中断的中断类型码固定为2,所以中断过程中,不需要取中断类型码
9.不可屏蔽中断的中断过程
1.标志寄存器入栈,IF=0,TF=0
2.CS,IP入栈
3.(IP)=(8),(CS)=(0AH) ;固定地址
10.几乎所有外中断,都是可屏蔽中断。当外设有需要处理的事件发生时
相关芯片向CPU发出可屏蔽中断信息。
不可屏蔽中断是系统中有必须处理的紧急情况发生时用来通知CPU的中断信息,本门课程中,主要讨论可屏蔽中断
15.3 PC机键盘的处理过程
下面看一个键盘输入的处理过程,并以此来体会PC机处理外设输入的基本方法
1.键盘输入
1.键盘上每一个键相当于一个开关,键盘中有一个芯片对键盘上的每一触键的开关状态进行扫描。
2.按下一个键时,开关接通,该芯片就产生一个扫描码,扫描码说明按下的键在键盘上的位置。扫描码被送入主板上的相关接口芯片的寄存器中,该寄存器的端口地址为60H
3.松开控下的键时,也产生一个扫描码,扫描码说明了松开的键在键盘上的位置,松开按键时,产生的扫描码也被送入60H端口中。
4. 一般按下一个键时,产生的扫描码称为通码,松开一个键产生的扫描码称为断码. 扫描码长度为一个字节,通码的第七位为0,断码的第七位为1
5.即:断码=通码+80H(10000000)比如,g键的通码为22h,断码为a2h。
表15.1是键盘上部分键的扫描码,只列出通码。断码=通码+80h。
2.引发9号中断
键盘的输入到达60H端口时,相关的芯片会向CPU发出中断类型码为9的可屏蔽中断信息。CPU检测到中断信息后,如果IF=1,则响应中断,同时将IF设置为0(不让其他可屏蔽中断进行干扰),引发中断过程,转去执行int9中断例程
3.执行int 9中断例程
BIOS提供了int9中断例程,用来进行基本的键盘输入处理,主要的工作如下:
1.读出60H端口中的扫描码
2.如果是字符键的扫描码,将该扫描码对应的字符码(即:ASCII码)送入内存中的BIOS键盘缓冲区,如果是控制键ctrl和切换键capslock的扫描码,则将其转变为状态字节,写入内存中存储状态字节的单元
3.对键盘系统进行相关的控制,比如说,向相关芯片发出应答信息。
BIOS键盘缓冲区是系统启动后,BIOS用于存放int9中断例程所接受的键盘输入的内存区。该内存区可以存储15个键盘输入,int9中断例程除了接收扫描码外,还要产生和扫描码对应的字符码,所以在BIOS键盘缓冲区中,一个键盘输入用一个字单元存放,高字节存放扫描码,低字节存放字符码。
0040:17单元存储键盘状态字节,该字节记录了控制键和切换键的状态。键盘状态字节各位记录的信息如下:
15.4 编写int9中断例程,并安装
梳理键盘输入的处理过程
1.键盘产生扫描码
2.扫描码送入60H端口
3.一旦侦测到60H端口有动静,引发9号中断
4.CPU执行int9中断例程处理输入
以上的过程,前三步都由硬件系统自动完成,能够修改的只有第四步,修改int9中断程序
【任务演示】在屏幕中依次显示“a”~“z”并可以让人看清。在显示过程中,按下Esc键后,该表显示的颜色
;程序1:实现连续显示“a”~“z”
;编程:在屏幕中间依次显示“a”~“z”,并可以让人看清。在显示的过程中,按下'Esc'键后,改变显示的颜色。
;部分功能代码:
stack segment
db 128 dup (0)
stack ends
code segment
assume cs:code
start:
mov ax,stack
mov ss,ax
mov sp,128
mov ax,0b800h
mov es,ax
mov ah,'a'
s: mov es:[160*12+40*2],ah
call delay ;为了让cpu延时显示字母,所以写一个子查询循环做运算
inc ah
cmp ah,'z'
jna s
mov ax,4c00h
int 21h
delay:
push ax
push dx
mov dx,1000h ;循环1000 0000h次
mov ax,0
s1:
sub ax,1 ; 0-1 = ffffh ,当ffffh减到0时,会借位,
sbb dx,0 ;带进制位减法 -cf 当上面借位了,这里才减到1
cmp ax,0 ;ax减到0时也就是10000h次
jne s1 ;ax不等于0,然后循环ffffh
cmp dx,0 ;继续执行上面的步骤直到dx减为0 所以是循环1000h次10000h
jne s1
pop dx
pop ax
ret
code ends
end start
;程序2:实现改变颜色
;编程:在屏幕中间依次显示“a”~“z”,并可以让人看清。在显示的过程中,按下'Esc'键后,改变显示的颜色。
stack segment
db 128 dup (0)
stack ends
data segment
dw 0,0
data ends
code segment
assume cs:code
start:
mov ax,stack
mov ss,ax
mov sp,128
mov ax,data
mov ds,ax
mov ax,0
mov es,ax
push es:[9*4]
pop ds:[0]
push es:[9*4+2]
pop ds:[2] ;将原来的int 9中断例程的入口地址保存在ds:0、ds:2单元中
mov word ptr es:[9*4],offset int9
mov es:[9*4+2],cs ;在中断向量表中设置新的int 9中断例程的入口地址
mov ax,0b800h
mov es,ax
mov ah,'a'
s:
mov es:[160*12+40*2],ah
call delay
inc ah
cmp ah,'z'
jna s
mov ax,0
mov es,ax
push ds:[0]
pop es:[9*4]
push ds;[2]
pop es;[9*4+2] ;将中断向量表中int 9中断例程的入口恢复为原来的地址
mov ax,4c00h
int 21h
delay:
push ax
push dx
mov dx,0010h
mov ax,0
s1:
sub ax,1
sbb dx,0
cmp ax,0
jne s1
cmp dx,0
jne s1
pop dx
pop ax
ret
;------以下为新的int 9中断例程--------------------
;int9中断例程是在进行键盘输入之后,由系统自动调用
int9:
push ax
push bx
push es
in al,60h ;从端口60h读出键盘的输入到al中
pushf ;标志寄存器入栈
pushf
pop bx
and bh,11111100b
push bx
popf ;IF=0,TF=0 他们在flag的第8、9位上
call dword ptr ds:[0] ; CS、IP入栈;(IP)= ((ds)*16+0),(CS)=((ds)*16+2) ;对int指令进行模拟,调用原来的int 9中断例程
cmp al,1 ;esc键盘扫描码01
jne int9ret ;如果不是esc就返回之前的显示字母任务
mov ax,0b800h ;如果是就变化颜色
mov es,ax
inc byte ptr es:[160*12+40*2+1] ;属性增加1,改变颜色
int9ret:
pop es
pop bx
pop ax
iret
code ends
end start