汇编实验15:安装新的int 9中断例程

时间:2023-03-08 21:49:07
汇编实验15:安装新的int 9中断例程

汇编实验15:安装新的int 9中断例程

任务

安装一个新的int 9中断例程,功能:在DOS下,按下“A”键后,除非不在松开,一旦松开后,就显示满屏幕的“A”,其他键照常处理。

预备知识概要

这次实验其实不难,王爽的教材中已经给出了许多实例代码,依葫芦画瓢都能圆满完成任务。

这次我们学习的是外中断,以外设的输入为例,CPU通过中断机制来处理外设的输入。

外中断源分为两大类:

  • 可屏蔽中断
  • 不可屏蔽中断

顾名思义,对于前者CPU可以选择不去响应中断,对于后者,CPU无论如何都要执行完当前指令后立即响应。

不可屏蔽中断的中断过程很容易理解,在8086CPU中,它的中断类型码是固定的,就是2,接下来的过程和内中断一模一样。

  • 标志寄存器入栈,并设置IF=0,TF=0
  • 将CS,IP寄存器压栈
  • 使(IP) = (8) ,(CS) = (0AH)

下面我们重点聊一聊可屏蔽中断

IF标志位

CPU通过IF标志为的状态来决定是否响应可屏蔽中断

  • 如果IF=1,CPU执行完当前指令后,响应中断,并引发中断过程
  • 如果IF=0,CPU不响应可屏蔽中断

我们可以通过以下两条指令设置标志位IF

  • STI 设置标志位IF = 1
  • CLI 设置标志位IF = 0

可屏蔽中断的中断过程

可屏蔽中断的中断过程和内中断的过程几乎一模一样,只不过取得中断类型码的方式不同,可屏蔽中断的中断类型码是通过数据总线进入CPU的。
为了表述完整,我还是总结一下这个中断过程(我怎么突然变勤快了吧,因为可以复制粘贴嘛!):

  • 获取中断类型码N(其实是为了获得中断例程的内存地址)
  • 标志寄存器入栈,并设置IF=0,TF=0
  • 将CS,IP寄存器压栈
  • 使(IP) = (4N) ,(CS) = (4N+2)

键盘输入过程

我们重点学习键盘输入的过程,总结起来就以下四步:

  • 键盘产生扫描码
  • 扫描码送入60h端口
  • 引发9号中断
  • CPU执行9号中断例程

键盘的每一个键相当于一个开关,按下一个键,开关接通,键盘上的芯片产生一个扫描码(大小为一个字节),我们称它为“通码”,来说明按键的位置,然后送入主板上相关的接口芯片的寄存器中,就是60H端口。
同样的,松开按键时,也会产生一个扫描码,然后被送入60H端口,这个扫描码被称为“断码”。

同一个按键有两个扫描码,就是断码和通码,它们的区别就是最高位1和0的差别。
可以总结为“断码 = 通码 + 80H”。

最后说一下BIOS的键盘缓冲区,它能存储15个键盘输入。每个键盘输入占用两个字节大小的空间。高字节放扫描码,低字节放字符码(就是ASCII码)。

代码实现

虽然是自己编写int 9中断处理程序,但是我们没必要完整的处理一个键盘的输入,中间涉及一些硬件细节,会偏离内容主线(总之就是现在的你太菜了,怕你伤自尊,就不告诉你了)。没关系,对于硬件细节的处理,交给BIOS的int 9的中断例程就好了嘛!

为此,王爽的书里就讲了一些技巧,比如怎么用汇编指令模拟int中断过程,怎么在已经修改了中断向量表的情况下,调用原始的BIOS的int 9中断例程。详细内容请见王爽的《汇编语言(第三版)》(我真的不想在打字了)

下面贴出我的代码:

assume      cs:code,ss:stack

stack       segment
db 256 dup(0)
stack ends code segment main: mov ax,stack
mov ss,ax
mov sp,256 call install
s: jmp s mov ax,4c00h
int 21h
;********************************************************
install: ;寄存器保护
push ax
push cx
push si
push di
push ds
push es
;将中断处理程序do9安装到内存单元0:204h中
push cs
pop ds
mov si,offset do9 mov ax,0
mov es,ax
mov di,204h
mov cx,offset do9end - offset do9
cld
rep movsb
;保存BIOS的9号中断例程地址
push es:[9*4]
pop es:[200h] ;偏移地址存放到内存单元0:200h中
push es:[9*4+2]
pop es:[202h] ;段地址存放到内存单元0:202h中 ;修改原来的中断向量表,此过程中不响应可屏蔽中断
cli
mov word ptr es:[9*4],204h
mov word ptr es:[9*4+2],0
sti
;恢复寄存器
pop es
pop ds
pop di
pop si
pop cx
pop ax
ret
;----------------------------------------------------
do9: push ax
push bx
push cx
push es in al,60h pushf
call dword ptr cs:[200h] cmp al,9eh ;A键的断码为9eh
jne do9ret ;显示满屏幕的字符A
mov ax,0b800h
mov es,ax
mov bx,0
mov cx,2000
do9_s: mov byte ptr es:[bx],'A'
add bx,2
loop do9_s do9ret: pop es
pop cx
pop bx
pop ax
ret
;----------------------------------------------------
do9end: nop
code ends end main

对,你没看错,这回的代码是有注释的!(我其实还是挺勤快的)。下面是执行的结果:

汇编实验15:安装新的int 9中断例程

汇编实验15:安装新的int 9中断例程

总结

这几次的实验其实都没有什么挑战性,都是依葫芦画瓢,只要熟悉教材前面的示范代码,都可以轻松的完成实验。关键就是对CPU中断机制和中断过程的理解。毕竟,在以后的学习中,我们不可能真的要自己用16位的汇编语言自己写一个中断处理程序(不然我选择狗带)。关键是对加深对计算机的理解,对汇编语言有个初步的认识与体验,为今后的学习作一定的铺垫(我其实挺想知道高级语言怎么编译成相应的机器指令的,可惜由于专业原因,这辈子都学不到了……可以自学……如果我不是个懒人的话……)。