汇编学习记录15 外中断

背景问题:
外部中断集中要处理的问题就是外设(特别是I/O)输入中断,我们有三个要解决的问题:
(1)外设的输入随时可能发生,CPU如何得知?
(2)CPU从何处得到外设的输入?
(3)CPU怎样处理这些输入?

1.外中断信息

我们已经知道,PC的主板上装有许许多多的接口芯片,这些外设接口芯片上有许许多多的寄存器,CPU把这些寄存器当做端口来访问。

  • 外中断:CPU提供一种机制——外中断机制,来处理这些外设的请求。相关的外设芯片向CPU发送中断信息,CPU执行完当前指令,便可以检测到这些信息,并触发相应的中断过程

PC系统中,外中断一般被分为两类:
(1)可屏蔽中断->CPU可以不响应的中断,是否响应由标志寄存器IF位决定。IF=1,则CPU执行完当前指令相应中断;IF=0,CPU屏蔽中断。
补充说明:此类中断的中断类型码一般由外设直接给CPU,中断处理程序入口地址也存放在中断向量表中;一般所有由外设引发的中断都是可屏蔽中断。

(2)不可屏蔽中断->CPU必须响应的中断。中断类型码固定为2。不可屏蔽中断一般来说是由于系统中有必须处理的紧急请况发生时才会引发的中断

2.外中断举例——PC机键盘处理过程

(BIOS提供了9号中断例程来处理基本的键盘输入处理)
(1)键盘上的通码和断码。
键盘上每个按键都对应一个扫描码,存放在内存的一张表中。通码是按键按下时键盘端口给出的扫描码,而断码是按键释放时给出的扫描码(按键按下和释放均会产生中断信号)
(2)键盘输入触发中断过程
键盘也被CPU当作一个端口,键盘输出数据的端口号为60H,传出来的是键盘的扫描码。当9号中断触发时(9号中断对应键盘输入),CPU进入中断处理程序拿出60号端口的扫描码解析处理,再进行相应的显示。
(3)编程实战:编写外中断9号中断例程

9号中断例程编程——Esc键改变屏幕显示字符颜色:

实验说明:在屏幕上一个个输出字母a~z,在输出的过程中实现点击Esc键就改变字符的颜色(注意这个程序不是安装到中断向量表上,而是直接将9中断入口地址改到程序段上

assume cs:code,ss:stack,ds:data
stack segment
    db 128 dup (?)
stack ends

data segment
    dw 0,0 ; 存储9号中断原本的入口
data ends

code segment
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] ; 两次入栈两次出栈,将9号中断处理程序原地址存入数据段

    cli ; 将IF设置为0,在以下程序段不在理会可屏蔽外中断,以防在中断地址还未修改完成时发生中断
    mov word ptr es:[9*4],offset int9
    mov es:[9*4+2],cs
    sti ; 将IF设置为1,可以接受外中断

    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] ; 恢复九号中断处理程序的入口地址

    mov ax,4c00H
    int 21H

delay:  
    push dx
    push ax
    mov dx,10H
    mov ax,0
S1: sub ax,1
    sbb dx,0
    cmp ax,0
    jne S1
    cmp dx,0
    jne S1  
    pop ax
    pop dx
    ret ;这是一个延时程序,以防字符跳动太快难以看清楚

int9:
    push ax
    push bx
    push es

    in al,60H ; 从键盘输出端口,读出扫描码

    pushf
    call dword ptr ds:[0] ; 调用原来9号中断

    cmp al,1 ; 1是Esc的扫描码
    jne int9ret

    mov ax,0b800H
    mov es,ax
    inc byte ptr es:[160*12+40*2+1] ; 将显示字符的位置属性改变

int9ret:
    pop es
    pop bx
    pop ax
    iret ; 中断返回

code ends
end start 

实验结果示例:


12345.GIF

课后实验:

实验说明:
安装新的9号中断例程,功能:按下A后,除非不松开;如果松开就显示满屏幕的‘A’字符。
参考代码:

assume cs:code,ss:stack
stack segment
    db 128 dup (?)
stack ends

code segment
start:
    mov ax,stack
    mov ss,ax
    mov sp,128

    mov ax,0
    mov es,ax ; ds中存中断向量表的段地址
    push es:[9*4]
    pop es:[200H]
    push es:[9*4+2] 
    pop es:[202H] ; 首先保存原9号中断处理程序入口

    mov di,204H ; 将在0:204H处安装新的9号中断程序

    mov ax,code
    mov ds,ax
    mov si,offset int9 ; ds:si为将要安装的程序源地址
    mov cx,offset int9ret-offset int9
    cld
    rep movsb ; 串传送指令,将程序传到目标位置

    cli
    mov word ptr es:[9*4],204H ; 修改9号中断程序入口
    mov es:[9*4+2],es
    sti

    mov bx,100H ; 这是一个延迟程序等待按键
    mov ax,0
S:  sub ax,1
    sbb bx,0
    cmp ax,0
    jne S
    cmp bx,0
    jne S
    
    cli
    push es:[200H] ; 将9号中断程序入口恢复
    pop es:[9*4]
    push es:[202H]
    pop es:[9*4+2]
    sti
    
    mov ax,4c00H
    int 21H

int9:   
    push ax
    push es
    push dx
    push cx
    in al,60H
    cmp al,9EH
    je s1
    pushf
    call dword ptr es:[200H] ; 调用原9号中断程序
done:
    pop cx
    pop dx
    pop es
    pop ax
    iret

s1: mov ax,0b800H
    mov es,ax
    mov dx,0
    mov cx,2000 ; 4000个字节,2000个字符
s2: mov byte ptr es:[dx],'A'
    add dx,2
    loop s2
    jmp done

int9ret:    
    nop

code ends
end start

实验结果展示:


1.GIF
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。