1、汇编语言课件,王爽 著清华大学出版社,制作工具:Microsoft PowerPoint2003,本课件由汇编网()制作提供,第13章 int指令,13.1 int指令 13.2 编写供应用程序调用的中断例程 13.3 对int、iret和栈的深入理解 13.4 BIOS和DOS所提供的中断例程 13.5 BIOS和DOS中断例程的安装过程 13.6 BIOS中断例程应用 13.7 DOS中断例程应用,引言,中断信息可以来自 CPU 的内部和外部,当CPU 的内部有需要处理的事情发生的时候,将产生需要马上处理的中断信息,引发中断过程。,引言,在第12章中,我们讲解了中断过程和两种内中断的处理。
2、 这一章中,我们讲解另一种重要的内中断,由int指令引发的中断。,13.1 int指令,int格式: int n,n为中断类型码。它的功能是引发中断过程。 CPU 执行int n指令,相当于引发一个 n号中断的中断过程,执行过程如下: (1)取中断类型码n; (2)标志寄存器入栈,IF = 0,TF = 0; (3)CS、IP入栈; (4)(IP) = (n*4),(CS) = (n*4+2)。 从此处转去执行n号中断的中断处理程序。,13.1 int指令,可以在程序中使用int指令调用任何一个中断的中断处理程序。 比如,下面的程序:,assume cs:codecode segmentsta
3、rt: mov ax,0b800hmov es,axmov byte ptr es:12*160+40*2,!int 0code endsend start,13.1 int指令,这个程序在 Windows 2000中的 DOS方式下执行时,将在屏幕中间显示一个“!”,然后显示“Divide overflow”后返回到系统中。 “!”是我们编程显示的,而,“Divide overflow”是哪里来的呢?我们的程序中又没有做除法,不可能产生除法溢出。,13.1 int指令,程序是没有做除法,但是在结尾使用了int 0指令。 CPU执行int 0指令时,将引发中断过程,执行 0号中断处理程序,而系
4、统设置的 0号中断处理程序的功能是显示“Divide overflow”,然后返回到系统。,13.1 int指令,可见,int 指令的最终功能和call指令相似,都是调用一段程序。 一般情况下,系统将一些具有一定功能的子程序,以中断处理程序的方式提供给应用程序调用。,13.1 int指令,我们在编程的时候,可以用int指令调用这些子程序。 当然,也可以自己编写一些中断处理程序供别人使用。 以后,我们可以将中断处理程序简称为中断例程。,13.2 编写供应用程序调用的中断例程,前面,我们已经编写过中断0 的中断例程了,现在我们讨论可以供应用程序调用的中断例程的编写方法。 我们通过两个问题来讨论:
5、1、示例一 2、示例二,13.2 编写供应用程序调用的中断例程,示例一 编写、安装中断7ch的中断例程: 功能:求一word型数据的平方。 参数: (ax)=要计算的数据。 返回值:dx、ax中存放结果的高16位和低16位。 应用举例:求2*34562 程序及分析,13.2 编写供应用程序调用的中断例程,assume cs:codecode segmentstart: mov ax,3456;(ax)=3456int 7ch;调用中断7ch的中断例程,计算ax中的数据的平方add ax,ax;dx:ax存放结果,讲结果乘以2mov ax,4c00hint 21hcode endsend sta
6、rt 分析一下,13.2 编写供应用程序调用的中断例程,我们要做三部分工作: (1)编程实现求平方功能的程序; (2)安装程序,我们将其安装在0:200处; (3)设置中断向量表,将程序的入口地址保存在7ch表项中,使其成为中断7ch的中断例程。 安装程序代码,13.2 编写供应用程序调用的中断例程,注意,在中断例程 sqr的最后,要使用iret指令。 用汇编语法描述,iret指令的功能为: pop IPpop CSpopf,13.2 编写供应用程序调用的中断例程,CPU执行int 7ch指令进入中断例程之前,标志寄存器、当前的CS和IP被压入栈中,在执行完中断例程后,应该用iret指令恢复i
7、nt 7ch 执行前的标志寄存器和CS、IP的值,从而接着执行应用程序。 int指令和 iret指令的配合使用与call指令和ret指令的配合使用具有相似的思路。,13.2 编写供应用程序调用的中断例程,示例二 编写、安装中断7ch的中断例程: 功能:将一个全是字母,以 0结尾的字符串,转化为大写。 参数:ds:si指向字符串的首地址。 应用举例:将data段中的字符转化为大写。,13.2 编写供应用程序调用的中断例程,应用举例:将data段中的字符转化为大写。assume cs:codedata segmentdb conversation,0data endscode segmentsta
8、rt: mov ax,datamov ds,axmov si,0int 7chmov ax,4c00hint 21hcode endsend start,13.2 编写供应用程序调用的中断例程,安装程序源码 在中断例程capital中用到了寄存器 si和cx,编写中断例程和编写子程序的时候具有同样的问题,就是要避免寄存器的冲突。 应该注意例程中用到的寄存器的值的保存和恢复。,13.3 对int、iret和栈的深入理解,问题:用7ch中断例程完成 loop指令的功能。 loop s 的执行需要两个信息,循环次数和到s的位移,所以,7ch中断例程要完成loop指令的功能,也需要这两个信息作为参数。
9、 我们用cx存放循环次数,用bx存放位移。,13.3 对int、iret和栈的深入理解,应用举例:在屏幕中间显示80个!。 assume cs:codecode segmentstart: mov ax,0b800hmov es,axmov di,160*12mov bx,offset s - offset se;设置从标号se到标号s的转移位移mov cx,80s: mov byte ptr es:di,!add di,2int 7ch ;如果(cx)0,转移到标号s处se: nopmov ax,4c00hint 21hcode endsend start,13.3 对int、iret和栈的
10、深入理解,在上面的程序中,用int 7ch调用7ch中断例程进行转移,用 bx 传递转移的位移。 分析:为了模拟loop指令,7ch中断例程应具备下面的功能: (1)dec cx (2)如果(cx)0,转到标号s 处执行,否则向下执行。,13.3 对int、iret和栈的深入理解,下面我们分析7ch中断例程如何实现到目的地址的转移: (1)转到标号s显然应设(CS)=标号s的段地址,(IP)=标号s的偏移地址; (2)那么,中断例程如何得到标号s的段地址和偏移地址呢? 分析 (3)现在知道,可以从栈中直接和间接地得到标号s的段地址和偏移地址,那么如何用它们设置CS:IP呢? 分析,13.3 对
11、int、iret和栈的深入理解,int 7ch引发中断过程后,进入 7ch中断例程,在中断过程中,当前的标志寄存器、CS和IP都要压栈,此时压入的CS和IP中的内容,分别是调用程序的段地址(可以认为是标号 s 的段地址)和int 7ch后一条指令的偏移地址(即标号se的偏移地址)。,13.3 对int、iret和栈的深入理解,可见,在中断例程中,可以从栈里取得标号s 的段地址和标号 se的偏移地址,而用标号se的偏移地址加上bx中存放的转移位移就可以得到标号s的偏移地址。,13.3 对int、iret和栈的深入理解,可以利用iret指令,我们将栈中的se的偏移地址加上 bx 中的转移位移,则栈
12、中的se的偏移地址就变为了s的偏移地址。 我们再使用iret指令,用栈中的内容设置CS、IP,从而实现转移到标号s处。,13.3 对int、iret和栈的深入理解,7ch中断例程如下:lp: push bpmov bp,spdec cxjcxz lpretadd bp+2,bxlpret: pop bpiret,13.3 对int、iret和栈的深入理解,因为要访问栈,使用了 bp,在程序开始处将bp 入栈保存,结束时出栈恢复。 当要修改栈中se的偏移地址的时候,栈中的情况为; 栈顶处是bp 原来的数值,下面是se的偏移地址,再下面是s的段地址,再下面是标志寄存器的值。,13.3 对int、i
13、ret和栈的深入理解,而此时,bp中为栈顶的偏移地址,所以(ss)*16+(bp)+2)处为se 的偏移地址,将它加上bx 中的转移位移就变为s的偏移地址。最后用iret出栈返回,CS:IP即从标号s处开始执行指令。 如果(cx)=0,则不需要修改栈中 se的偏移地址,直接返回即可。 CPU从标号se处向下执行指令。,特别提示,检测点13.1(page246)没有通过此检测点,请不要向下进行!,13.5 BIOS和DOS中断例程的安装过程,而BIOS和DOS提供的中断例程是如何安装到内存中的呢? 我们下面讲解它们的安装过程。,13.5 BIOS和DOS中断例程的安装过程,(1) 开机后,CPU
14、 一加电,初始化(CS)=0FFFFH,(IP)=0,自动从FFFF:0单元开始执行程序。FFFF:0处有一条转跳指令,CPU执行该指令后,转去执行BIOS中的硬件系统检测和初始化程序。,13.5 BIOS和DOS中断例程的安装过程,(2)初始化程序将建立BIOS 所支持的中断向量,即将BIOS提供的中断例程的入口地址登记在中断向量表中。,13.5 BIOS和DOS中断例程的安装过程,(3) 硬件系统检测和初始化完成后,调用int 19h进行操作系统的引导。从此将计算机交由操作系统控制。,13.5 BIOS和DOS中断例程的安装过程,(4)DOS 启动后,除完成其它工作外,还将它所提供的中断例
15、程装入内存,并建立相应的中断向量。,特别提示,检测点13.2(page248 独立思考后可共同讨论)没有通过此检测点,请不要向下进行!,13.6 BIOS中断例程应用,下面我们举几个例子,来看一下BIOS中断例程的应用。int 10h中断例程是BIOS提供的中断例程,其中包含了多个和屏幕输出相关的子程序。,13.6 BIOS中断例程应用,一般来说,一个供程序员调用的中断例程中往往包括多个子程序,中断例程内部用传递进来的参数来决定执行哪个子程序。 BIOS 和DOS 提供的中断例程,都用ah来传递内部子程序的编号。,13.6 BIOS中断例程应用,我们看一下int 10h中断例程的设置光标位置功
16、能。mov ah,2mov bh,0mov dh,5mov dl,12int 10h,13.6 BIOS中断例程应用,(ah)=2表示调用第 10h号中断例程的 2号子程序,功能为设置光标位置,可以提供光标所在的行号(80*25字符模式下:024)、列号(80*25字符模式下:079),和页号作为参数。 (bh)=0,(dh)=5,(dl)=12,设置光标到第0页,第5行,第12列。,13.6 BIOS中断例程应用,bh中页号的含义:内存地址空间中,B8000hBFFFFh共 32K的空间,为80*25 彩色字符模式的显示缓冲区。 一屏的内容在显示缓冲区中共占4000个字节。,13.6 BIO
17、S中断例程应用,显示缓冲区分为8页,每页4K(4000),显示器可以显示任意一页的内容。一般情况下,显示第 0 页的内容。 也就是说,通常情况下,B8000B8F9F中的4000个字节的内容将出现在显示器上。,13.6 BIOS中断例程应用,再看一下int 10h中断例程的在光标位置显示字符功能。mov ah,9 ;置光标mov al,a ;字符mov bl,7 ;颜色属性mov bh,0 ;第0页mov cx,3 ;字符重复个数int 10h,13.6 BIOS中断例程应用,(ah)=9 表示调用第10h号中断例程的9号子程序; 功能为在光标位置显示字符,可以提供要显示的字符、颜色属性、页号
18、、字符重复个数作为参数。,13.6 BIOS中断例程应用,(bh)中的颜色属性格式如下:可以看出,和显存中的属性字节的格式相同。,13.6 BIOS中断例程应用,编程:在屏幕的5行12列显示3个红底高亮闪烁绿色的a。 程序源码 注意:闪烁的效果必须在全屏DOS 方式下才能看到。,13.7 DOS中断例程应用,int 21h 中断例程是DOS提供的中断例程,其中包含了DOS提供给程序员在编程时调用的子程序。 我们从前一直使用的是 int 21中断例程的4ch号功能,即程序返回功能,如下:mov ah,4ch ;程序返回mov al,0 ;返回值int 21h,13.7 DOS中断例程应用,(ah
19、)=4ch表示调用第21h号中断例程的 4ch号子程序,功能为程序返回,可以提供返回值作为参数。 我们前面使用这个功能的时候经常写作: mov ax,4c00hint 21h,13.7 DOS中断例程应用,我们看一下int 21h中断例程的在光标位置显示字符串的功能:ds:dx指向字符串 ;要显示的字符串需用“$”作为结束符mov ah ,9 ;功能号9,表示在光标位置显示字符串int 21h,13.7 DOS中断例程应用,(ah)=9表示调用第21h号中断例程的 9号子程序,功能为在光标位置显示字符串,可以提供要显示字符串的地址作为参数。,13.7 DOS中断例程应用,编程:在屏幕的5列12行显示字符串“Welcome to masm!”。源代码,13.7 DOS中断例程应用,上述程序在屏幕的5列12行显示字符串“Welcome to masm!”,直到遇见“$”(“$” 本身并不显示,只起到边界的作用)。 如果字符串比较长,遇到行尾,程序会自动转到下一行开头处继续显示;如果到了最后一行,还能自动上卷一行。,13.7 DOS中断例程应用,DOS为程序员提供了许多可以调用的子程序,都包含在 int 21h 中断例程中。 我们这里只对原理进行了讲解,对于DOS提供的所有可调用子程序的情况,读者可以参考相关的书籍。,小结,