汇编中的外中断

1.写在前面

前面的博客,我介绍了CPU中的内中断以及int指令,还有就是端口的读写,今天我们来介绍外中断。

2.本篇博客的概述

在这里插入图片描述

3.外中端

CPU在计算机系统中,除了能够执行指令,进行运算以外,还应该能够对外部设备进行控制,接收它们的输入,向它们进行输出。也就是说,CPU除了有运算能力外,还要有I/O能力。

我们平时在键盘上输入一个键盘的时候,屏幕上立即显示,可以知道这些都是及时处理的。显然需要解决两个问题:外设的输入随时可能发生,CPU如何得知?CPU从何处得到外设的输入?

3.1接口芯片和端口

外设的输入不直接送入内存和CPU,而是送入相关的接口芯片的端口中;CPU向外设的输出也不是直接送入外设,而是先送入端口中,再由相关的芯片送入外设。CPU还可以向外设输出控制命令,而这些控制命令也是先送到相关芯片的端口中,然后再由相关的芯片根据命令对外设实施控制。

可见,CPU通断端口和外部设备进行联系。

3.2外中断信息

外设随时都可能发生需要CPU及时处理的事件,CPU如何及时得知并处理?CPU提供中断机制来满足这种需要。前面我们介绍了内中断,还有一种中断信息,来自于CPU外部,当CPU外部有需要处理的事情发生的时候,比如说,外设的输入到达,相关芯片将向CPU发生相应的中断信息。这种叫做外中断。

外中断源分成以下两类:

  1. 可屏蔽中断

    可屏蔽中断是CPU可以不响应的外中断。CPU是否响应可屏蔽中断,要看标志寄存器的IF位的设置。当CPU检测到可屏蔽中断信息时,如果IF=1,则CPU在执行完当前指令后响应中断,引发中断过程;如果IF=0,则不响应可屏蔽中断。

    内中断的过程如下:

    1. 取中断类型码n

    2. 标志寄存器入栈,IF=0,TF=0;

    3. CS、IP入栈

    4. (IP)=(n*4),(CS)=(n*4+2)

      由此转去执行中断处理程序。

      可屏蔽中断所引发的中断过程,除在第一步的实现上有所不同外,基本上和可屏蔽中断过程相同。可屏蔽中断信息来自于CPU外部,中断类型码通过数据总线送入CPU的,而内中断的中断类型码是CPU内部产生的。

      这儿我们也知道为什么将IF设置为0的原因。将IF设置为0的原因就是:在进入中断处理程序后,禁止其他的可屏蔽中断。

      当然,如果在中断处理程序中需要处理可屏蔽中断,可以用指令将IF设置为1。具体的指令如下:

      sti,设置IF=1;
      cli,设置IF=0
      
  2. 不可屏蔽中断

    不可屏蔽中断是CPU必须立即响应的外中断。当CPU检测到不可屏蔽中断信息时,则在执行完当前指令后,立即响应,引发中断过程。

    对于8086CPU,不可屏蔽中断的中断类型码固定为2,所以中断过程中,不需要取中断类型码,则不可屏蔽中断的中断过程为:

    1. 标志寄存器入栈,IF=0,TF=0

    2. CS、IP入栈

    3. (IP)=(8),(CS)=(0AH)

      几乎所有由外设引发的外中断,都是可屏蔽中断。当外设有需要处理的事情发生时,相关芯片向CPU发生可屏蔽中断信息。不可屏蔽中断是在系统中有必须处理的紧急情况发生时用来通知CPU的中断信息。

3.3PC机键盘的处理过程

3.31键盘输入

键盘上的每一个键相当于一个开关,键盘中有一个芯片对键盘上的每一个键的开关的状态进行扫描。

按下一个键时,开关接通,该芯片就产生一个扫描码,扫描码说明了按下的键在键盘上的位置。扫描码被送入主板上的相关接口芯片的寄存器中,该寄存器的端口地址为60h。

松开按下的键时,也产生一个扫描码,扫描码说明了松开键在键盘上的位置。松开按键时产生的扫描码也被送入60h端口中。

一般将按下一个键时产生的扫描码称为通码,松开一个键产生的扫描码称为断码。扫描码长度为一个字节,通码的第7位为0,断码的第7位为1,即:

断码=通码+80h
3.32引发9号中断

键盘的输入到达60h端口时,相关的芯片就会向CPU发出中断类型码为9的可屏蔽中断信息。CPU检测到该中断信息后,如果IF=1,则响应中断,引发中断过程,转去执行int9中断例程。

3.33执行int 9 中断例程

int 9中断例程,主要的工作如下:

  1. 读出60h端口中的扫描码
  2. 如果是字符键的扫描码,将该扫描码和它所对应的字符码送入内存中的BIOS键盘缓冲去;如果是控制键和切换键的扫描码,则将其转变为状态字节写入内存中存储状态字节单元。
  3. 对键盘系统进行相关的控制。

BIOS键盘缓冲区是系统启动后,BIOS用于存放int 9中断例程所接收的键盘输入的内存区。该内存区可以存储15个键盘输入,因为int 9中断例程除了接收扫描码外,还要产生和扫描码对应的字符码,所以在BIOS键盘缓冲区中,一个键盘输入用一个字节单元存放,高位字节存放扫描码,低位字节存放字符码。

0040:17单元存储键盘状态字节,改字节记录了控制键和切换键的状态。键盘状态字节各位记录的信息如下:

0:右shift状态,置1表示按下右shift键
1:左shift状态,置1表示按下左shift键
2:Ctrl状态,置1表示按下Ctrl键
3:Alt状态,置1表示按下Alt键
4:ScrollLock状态,置1表示Scroll指示灯亮
5:NumLock状态,置1表示小键盘输入的是数字
6:CapsLock状态,置1表示输入大写字母
7:Insert状态,置1表示处于删除态

3.4编写int 9中断例程

从上面的内容中,可以看出键盘的输入的过程中:键盘产生扫描码,扫描码送入60h端口,引发9号中断,CPU执行int9中断例程处理键盘输入。

编程:在屏幕中间依次显示"a"~“z”,并可以让人看清。在显示的过程中,按下Esc键后,改变显示的颜色。

我们先来看一下如何依次显示"a"~“z”

assume cs:code
	code segment
	start: mov ax,0b800h
				 mov es,ax
				 mov ah,'a'
			s: mov es:[160*12+40*2],ah
				 inc ah
				 cmp ah,'z'
				 jna s
				 mov ax,4c00h
				 int 21h
		code ends
		end start

在上面的程序的执行过程中,我们无法看清屏幕上的显示。因为一个字母刚显示到屏幕上,CPU执行几条指令后,就又变成了另一个字母,字母之间切换得太快,无法看清。

应该在每显示一个字母后,延时一段时间,让人看清后,再显示下一个字母。我们可以通过CPU执行一段时间的空循环。具体的代码如下:

  mov dx,10h
  mov ax,0
s:sub ax,1
  sbb dx,0
  cmp ax,0
  jne s
  cmp dx,0
  jne s

上面的程序,循环100000h次。我们可以将循环延时的程序段写为一个子程序。

程序如下:

assume cs:code
	stack segment
		db 128 dup(0)
	stack ends
	code segment
	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
				 inc ah
				 cmp ah,'z'
				 jna s
				 
				 mov ax,4c00h
				 int 21h
	
	delay: push ax
				 push dx
				 mov dx,10h ;循环100000h次
				 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
	code ends
end start

显示"a"~“z”,并可以让人看清,这个功能就已经实现了。接下来就是按下Esc键后,改变显示的颜色呢?

键盘输入到达60h端口后,就会引发9号中断,CPU则转去执行int 9中断例程。我们可以编写int 9中断例程,功能如下:

  1. 从60h端口读出键盘的输入
  2. 调用BIOS的int 9的中断例程,处理其他硬件细节
  3. 判断是否为Esc的扫描码,如果是,改变显示的颜色后返回,如果不是则直接返回。

下面对这些功能的实现一一进行分析。

  1. 从端口60h读出键盘的输入

    in al,60h
    
  2. 调用BIOS的int 9中断的例程

    我们写的中断处理程序要称为新的int 9中断例程,主程序必须要将中断向量表中的int 9中断例程的入口地址改成为我们写的中断处理程序的入口地址。则在新的中断处理程序中调用原来的int 9 中断例程时,中断向量表中的int 9中断例程的入口地址却不是原来的int9中断例程的地址,所以不能使用int指令调用。

    要能在我们写的新中断例程中调用原来的的中断例程,就必须在将中断向量例程的入口地址改成新地址之前,将原来的入口地址保存起来。这样,在需要调用的时候,我们才能找到原来的中断例程的入口。

    对于现在的问题,假设将原来int 9中断例程的偏移地址和段地址保存在ds:[0]和ds:[2]单元中。那么我们在需要调用原来的int 9中断例程时候,就可以在ds:[0]、ds:[2]单元中找到它的入口地址。

    有了入口地址后,我们如何进行调用呢?

    当然不能使用int 9来调用,我们可以用别的指令来对int指令进行一些模拟,从而实现对中断例程的调用。

    int指令在执行的时候,CPU进行下面的工作。

    1. 取中断类型码n
    2. 标志寄存器入栈
    3. IF=0、TF=0
    4. CS、IP入栈
    5. (IP)=(n*4),(CS)=(n*4+2)

    取中断类型码是为了定位中断例程的入口地址,但是我们已经知道中断例程的入口地址。所以,我们用别的指令模拟int 指令时候,不需要第一步。在假设要调用的中断例程的入口地址在ds:0和ds:2单元中断前提下,我们将int过程用下面几步模拟

    1. 标志寄存器入栈
    2. IF=0、TF=0
    3. CS、IP入栈
    4. (IP)=((ds)*16+0),(CS)=((ds)*16+2)

    可以注意到第3、4步和call dword ptr ds:[0]的功能一样,call dword ptr ds:[0]的功能也是:

    1. CS、IP入栈
    2. (IP)=((ds)*16+0),(CS)=((ds)*16+2)

    所以int的过程的模拟过程变成:

    1. 标志寄存器入栈
    2. IF=0、TF=0
    3. call dword ptr ds:[0]

    对于第一步,可用pushf实现;

    对于第二步,可用下面的指令实现

    pushf
    pop ax
    and ah 11111100b ;IF和TF为标志寄存器的第9位和第8位
    push ax
    popf
    

    则模拟int指令的调用功能,调用入口地址在ds:0、ds:2中的中断例程的程序为:

    pushf ;标志寄存器入栈
    
    pushf
    pop ax
    and ah 11111100b ;IF和TF为标志寄存器的第9位和第8位
    push ax
    popf ;IF=0 TF=0
    
    call dword ptr ds:[0];CS、IP入栈 (IP)=((ds)\*16+0),(CS)=((ds)\*16+2)
    
  3. 如果是Esc的扫描码,改变显示的颜色后返回

    显示的位置在屏幕的中间,即第12行40列,现存中的偏移地址为:160*12+40*2。所以字符的ASCII码要送入段地址b800h,偏移地址160*12+40*2处。而段地址b800h,偏移地址160*12+40*2+1处是字符的属性,只要改变此处的数据就可以改变在段地址b800h,偏移地址160*12+40*2处显示的字符的颜色了。

    最后一个问题,要在程序返回前,将中断向量表中的int 9中断例程的入口地址恢复为原来的地址。否则程序返回后,别的程序将无法使用键盘。

    经过分析,完整的程序如下:

    assume cs:code
      stack segment
        db 128 dup (0)
      stack ends
    
      data segment
        dw 0,0
      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] ;将原来的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,10h
             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
    
        int9:push ax
             push bx
             push es
    
             in al,60h
    
             pushf
             pushf
             pop bx
             and bh,11111100b
             push bx
             popf
             call dword ptr ds:[0];对int指令进行模拟,调用原来的int 9 中断例程
    
             cmp al,1
             jne int9ret
    
             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
    

3.5安装新的int9中断例程

任务:安装一个新的int 9中断例程。

功能:在DOS下,按F1键后改变当前屏幕的显示颜色,其他的键照常处理

分析:

  1. 改变屏幕的显示颜色

    改变从B8000H开始的4000个字节中的所有奇地址单元中的内容,当前屏幕的显示颜色即发生改变。程序如下:

       mov ax,0b800h
       mov es,ax
       mov bx,1
       mov cx,2000
    s: inc byte ptr es:[bx]
       add bx,3
       loop s
    
  2. 其他键照常处理

    可以调用原int 9中断处理程序,来处理其他的键盘输入

  3. 原int9中断例程入口地址的保存

    因为在编写的新int 9中断例程中要调用原int9的中断例程,所以,要保存原int9中断例程的入口地址。不能保存在安装程序中,因为安装程序返回后地址将丢失。我们将地址保存在0:200单元处。

  4. 新int 9中断例程的安装

    assume cs:code
      stack segment
        db 128 dup (0)
      stack ends
      
      code segment
      start: mov ax,stack
             mov ss,ax
             mov sp,128
             
             push cs
             pop ds
             
             mov ax,0
             mov es,ax
             
             mov si,offset int9 ;设置ds:si指向源地址
             mov di,204h ;设置es:di指向目的地址
             mov cx,offset int9end - offset int9;设置cx为传输长度
             cld ;这只传输方向为正
             rep movsb
             
             push es:[9*4]
             pop es:[200h]
             push es:[9*4+2]
             pop es:[202h]
             
             cli
             mov word ptr es:[9*4],204h
             mov word ptr es:[9*4+2],0
             sti
             
             mov ax,4c00h
             int 21h
         
        int9:push ax
             push bx
             push cx
             push es
             
             in al,60h
             
             pushf
             call dword ptr cs:[200h];为此中断离职执行时(CS)=0
             
             cmp al3bh ;F1的扫描码为3bh
             
             mov ax,0b800h
             mov es,ax
             mov bx,1
             mov cx,2000
           s:inc byte ptr es:[bx]
             add bx,2
             loop s
             
     int9ret:pop es
             pop cx
             pop bx
             pop ax
             iret
             
     int9end:nop
    
    code ends 
    end start
    

4.写在最后

这章我们主要讲解了CPU对外设输入的通常处理方法。外设的输入送入端口,向CPU发出外中断信息,CPU检测到可屏蔽中断信息,如果IF=1,CPU在执行完当前指令后响应中断,执行响应的中断例程。可在中断例程中实现对外设输入的处理。

  • 2
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值