都建晓 發表於 2021-9-9 23:20:00

[汇编]《汇编语言》第13章 int指令

<h2 id="王爽汇编语言第四版-超级笔记">王爽《汇编语言》第四版 超级笔记</h2>
<p></p><div class="toc"><div class="toc-container-header">目录</div><ul><li>王爽《汇编语言》第四版 超级笔记<ul><li>第13章 int指令<ul><li>13.1 int指令、编写供应用程序调用的中断例程</li><li>13.2 对int、iret和栈的深入理解</li><li>13.3 BIOS和DOS所提供的中断例程及安装过程</li><li>13.4 BIOS和DOS中断例程应用</li></ul></li></ul></li></ul></div><p></p>
<h3 id="第13章-int指令">第13章 int指令</h3>
<p>中断信息可以来自CPU的内部和外部,当CPU的内部有需要处理的事情发生的时候,将产生需要马上处理的中断信息,引发中断过程。</p>
<p>在上一章中,我们讲解了中断过程和两种内中断的处理。</p>
<p>本章我们讲解另一种重要的内中断,由int指令引发的中断。</p>
<h4 id="131-int指令编写供应用程序调用的中断例程">13.1 int指令、编写供应用程序调用的中断例程</h4>
<p>int指令的格式为:int n为中断类型码,它的功能是引发中断过程。</p>
<p>CPU执行int n指令,相当于引发一个n号中断的中断过程,执行过程如下。</p>
<p>(1)取中断类型码n;<br>
(2)标志寄存器入栈,IF=0,TF=0;<br>
(3)CS、IP入栈;<br>
(4)(IP)=(nx4),(CS)=(nx4+2)。</p>
<p>从此处转去执行n号中断的中断处理程序。</p>
<p>可以在程序中使用int指令调用任何一个中断的中断处理程序。例如下面的程序:</p>
<pre><code>assume cs:code

code segment

start:mov ax,0b800h
      mov es,ax
      mov byte ptr es:,'!'
      int 0

code ends

end start
</code></pre>
<p>这个程序在Windows 2000中的DOS方式下执行时,将在屏幕中间显示一个“!”,然后显示"Divide overflow"后返回到系统中。“!”是我们编程显示的,而“Divide overflow"是哪里来的呢?</p>
<p>我们的程序中又没有做除法,不可能产生除法溢出。程序是没有做除法,但是在结尾使用了int 0指令。</p>
<p>CPU执行int 0指令时,将引发中断过程,执行0号中断处理程序,而系统设置的0号中断处理程序的功能是显示“Divide overflow",然后返回到系统。</p>
<p>可见int指令的最终功能和call指令相似,都是调用一段程序。</p>
<p>一般情况下,系统将一些具有一定功能的子程序,以中断处理程序的方式提供给应用程序调用。我们在编程的时候,可以用int指令调用这些子程序。</p>
<p>当然,也可以自己编写一些中断处理程序供别人使用。以后,我们可以将中断处理程序简称为中断例程。</p>
<hr>
<p>我们己经编写过中断0的中断例程了,现在我们讨论可以供应用程序调用的中断例程的编写方法。下面通过两个问题来讨论。</p>
<p>问题一:编写、安装中断7ch的中断例程。</p>
<p>功能:求一word型数据的平方。<br>
参数:(ax)=要计算的数据。<br>
返回值:dx、ax中存放结果的高16位和低16位。</p>
<p>应用举例:求2x3456^2。</p>
<pre><code>assume cs:code

code segment

start:mov ax,3456       ;(ax)=3456
      int 7ch         ;调用中断7ch的中断例程,计算ax中的数据的平方
      add ax,ax
      adc dx,dx         ;dx:ax存放结果,将结果乘以2
      mov ax,4c00h
      int 21h

code ends

end start
</code></pre>
<p>分析一下,我们要做以下3部分工作。</p>
<p>(1)编写实现求平方功能的程序;<br>
(2)安装程序,将其安装在0:200处;<br>
(3)设置中断向量表,将程序的入口地址保存在7ch表项中,使其成为中断7ch的中断例程。</p>
<p>安装程序如下。</p>
<pre><code>assume cs:code

code segment

start:mov ax,cs
      mov ds,ax
      mov si,offset sqr       ;设置ds:si指向源地址
      mov ax,0
      mov es,ax
      mov di,200h   ;设置es:di指向目的地址
      mov cx,offset sqrend-offset sqr   ;设置cx为传输长度
      cld   ;设置传输方向为正
      rep movsb

      mov ax,0
      mov es,ax
      mov word ptr es:,200h
      mov word ptr es:,0

      mov ax,4c00h
      int21h

sqr:    mul ax
      iret

sqrend:nop

code ends

end start
</code></pre>
<p>注意,在中断例程sqr的最后,要使用iret指令。用汇编语法描述,iret指令的功能为:</p>
<blockquote>
<p>pop IP<br>
pop CS<br>
popf</p>
</blockquote>
<p>CPU执行int 7ch指令进入中断例程之前,标志寄存器、当前的CS和IP被压入栈中,在执行完中断例程后,应该用iret指令恢复int 7ch执行前的标志寄存器和CS、IP的值,从而接着执行应用程序。</p>
<p>int指令和iret指令的配合使用与call指令和ret指令的配合使用具有相似的思路。</p>
<p>问题二:编写、安装中断7ch的中断例程。</p>
<p>功能:将一个全是字母,以0结尾的字符串,转化为大写。<br>
参数:ds:si指向字符串的首地址。</p>
<p>应用举例:将data段中的字符串转化为大写。</p>
<pre><code>assume cs:code

data segment
    db 'conversation',0
data ends

code segment

start:mov ax,data
      mov ds,ax
      mov si,0
      int 7ch

      mov ax,4c00h
      int 21h

code ends

end start
</code></pre>
<p>安装程序如下。</p>
<pre><code>assume cs:code

code segment

start:mov ax,cs
      mov ds,ax
      mov si,offset capital
      mov ax,0
      mov es,ax
      mov di,200h
      mov ex,offset capitalend-offset capital
      cld
      rep movsb

      mov ax,0
      mov es,ax
      mov word ptr es:,200h
      mov word ptr es: ,0
      mov ax,4c00h
      int 21h

capital:push cx
      push si

change: mov cl,
      mov ch,0
      jcxz ok
      and byte ptr ,11011111b
      inc si
      jmp short change

    ok: pop si
      pop cx
      iret

capitalend:nop

code ends

end start
</code></pre>
<p>在中断例程capital中用到了寄存器si和cx,编写中断例程和编写子程序的时候具有同样的问题,就是要避免寄存器的冲突。应该注意例程中用到的寄存器的值的保存和恢复。</p>
<h4 id="132-对intiret和栈的深入理解">13.2 对int、iret和栈的深入理解</h4>
<p>问题:用7ch中断例程完成loop指令的功能。</p>
<p>loop s的执行需要两个信息,循环次数和到s的位移,所以7ch中断例程要完成loop指令的功能,也需要这两个信息作为参数。我们用cx存放循环次数,用bx存放位移。</p>
<p>应用举例:在屏幕中间显示80个“!”。</p>
<pre><code>assume cs:code

code segment

start:mov ax,0b800h
      mov es,ax
      mov di,160*12

      mov bx,offset s-offset se       ;设置从标号se到标号s的转移位移
      mov cx,80

    s:mov byte ptr es:,'!'
      add di,2
      int 7ch   ;如果(cx)不等于0,转移到标号s处

   se:nop

      mov ax,4c00h
      int 21h

code ends

end start
</code></pre>
<p>在上面的程序中,用int 7ch调用7ch中断例程进行转移,用bx传递转移的位移。</p>
<p>分析:为了模拟loop指令,7ch中断例程应具备下面的功能。</p>
<p>(1)dec cx;<br>
(2)如果(cx)不等于0,转到标号s处执行,否则向下执行。</p>
<p>下面我们分析7ch中断例程如何实现到目的地址的转移。</p>
<p>(1)转到标号s显然应设(CS)=标号s的段地址,(IP)=标号s的偏移地址。</p>
<p>(2)那么,中断例程如何得到标号s的段地址和偏移地址呢?</p>
<p>int 7ch引发中断过程后,进入7ch中断例程,在中断过程中,当前的标志寄存器、CS和IP都要压栈,此时压入的CS和IP中的内容,分别是调用程序的段地址(可以认为是标号S的段地址)和int 7ch后一条指令的偏移地址(即标号se的偏移地址)。</p>
<p>在中断例程中,可以从栈里取得标号s的段地址和标号se的偏移地址,而用标号se的偏移地址加上bx中存放的转移位移就可以得到标号s的偏移地址。</p>
<p>(3)现在知道,可以从栈中直接和间接地得到标号s的段地址和偏移地址,那么如何用它们设置CS:IP呢?</p>
<p>可以利用iret指令,我们将栈中的se的偏移地址加上bx中的转移位移,则栈中的se的偏移地址就变为了s的偏移地址。我们再使用iret指令,用栈中的内容设置CS、IP,从而实现转移到标号s处。</p>
<p>7ch中断例程如下。</p>
<pre><code>    lp: push bp
      mov bp,sp
      dec cx
      jcxz Ipret
      add ,bx

Ipret:pop bp
      iret
</code></pre>
<p>因为要访问栈,使用了bp,在程序开始处将bp入栈保存,结束时出栈恢复。当要修改栈中se的偏移地址的时候,栈中的情况为:栈顶处是bp原来的数值,下面是se的偏移地址,再下面是s的段地址,再下面是标志寄存器的值。</p>
<p>而此时,bp中为栈顶的偏移地址,所以((ss)x16+(bp)+2)处为se的偏移地址,将它加上bx中的转移位移就变为s的偏移地址。最后用iret岀栈返回,CS:IP即从标号s处开始执行指令。</p>
<p>如果(cx)=0,则不需要修改栈中se的偏移地址,直接返回即可。CPU从标号se处向下执行指令。</p>
<h4 id="133-bios和dos所提供的中断例程及安装过程">13.3 BIOS和DOS所提供的中断例程及安装过程</h4>
<p>在系统板的ROM中存放着一套程序,称为BIOS(基本输入输岀系统),BIOS中主要包含以下几部分内容。</p>
<p>(1)硬件系统的检测和初始化程序;<br>
(2)外部中断和内部中断的中断例程;<br>
(3)用于对硬件设备进行I/O操作的中断例程;<br>
(4)其他和硬件系统相关的中断例程。</p>
<p>操作系统DOS也提供了中断例程,从操作系统的角度来看,DOS的中断例程就是操作系统向程序员提供的编程资源。</p>
<p>BIOS和DOS在所提供的中断例程中包含了许多子程序,这些子程序实现了程序员在编程的时候经常需要用到的功能。</p>
<p>程序员在编程的时候,可以用int指令直接调用BIOS和DOS提供的中断例程,来完成某些工作。</p>
<p>和硬件设备相关的DOS中断例程中,一般都调用了BIOS的中断例程。</p>
<hr>
<p>前面课程中,我们都是自己编写中断例程,将它们放到安装程序中,然后运行安装程序,将它们安装到指定的内存区中。此后,别的应用程序才可以调用。</p>
<p>而BIOS和DOS提供的中断例程是如何安装到内存中的呢?</p>
<p>我们下面讲解它们的安装过程。</p>
<p>(1)开机后,CPU一加电,初始化(CS)=0FFFFH,(IP)=0,自动从FFFF:0单元开始执行程序。FFFF:0处有一条转跳指令,CPU执行该指令后,转去执行BIOS中的硬件系统检测和初始化程序。</p>
<p>(2)初始化程序将建立BIOS所支持的中断向量,即将BIOS提供的中断例程的入口地址登记在中断向量表中。注意,对于BIOS所提供的中断例程,只需将入口地址登记在中断向量表中即可,因为它们是固化到ROM中的程序,一直在内存中存在。</p>
<p>(3)硬件系统检测和初始化完成后,调用int 19h进行操作系统的引导。从此将计算机交由操作系统控制。</p>
<p>(4)DOS启动后,除完成其他工作外,还将它所提供的中断例程装入内存,并建立相应的中断向量。</p>
<h4 id="134-bios和dos中断例程应用">13.4 BIOS和DOS中断例程应用</h4>
<p>我们举几个例子,来看一下BIOS中断例程的应用。</p>
<p>int 10h中断例程是BIOS提供的中断例程,其中包含了多个和屏幕输出相关的子程序。</p>
<p>一般来说,一个供程序员调用的中断例程中往往包括多个子程序,中断例程内部用传递进来的参数来决定执行哪一个子程序。BIOS和DOS提供的中断例程,都用ah来传递内部子程序的编号。</p>
<p>下面看一下int 10h中断例程的设置光标位置功能。</p>
<blockquote>
<p>mov ah,2 ;置光标<br>
inov bh,0 ;第0页<br>
mov dh,5 ;dh中放行号<br>
mov dl,12 ;dl中放列号<br>
int 10h</p>
</blockquote>
<p>(ah)=2表示调用第10h号中断例程的2号子程序,功能为设置光标位置,可以提供光标所在的行号(80x25字符模式下:0~24)、列号(80x25字符模式下:0-79),和页号作为参数。</p>
<p>(bh)=0,(dh)=5,(dl)=12,设置光标到第0页,第5行,第12列。</p>
<p>bh中页号的含义:内存地址空间中,B8000H~BFFFFH共32kB的空间,为80x25彩色字符模式的显示缓冲区。一屏的内容在显示缓冲区中共占4000个字节。</p>
<p>显示缓冲区分为8页,每页4KB(约等于4000B),显示器可以显示任意一页的内容。一般情况下,显示第0页的内容。</p>
<p>通常情况下,B8000H~B8F9FH中的4000个字节的内容将出现在显示器上。</p>
<p>再看一下int 10h中断例程的在光标位置显示字符功能。</p>
<blockquote>
<p>mov ah,9;在光标位置显示字符<br>
mov al,'a';字符<br>
mov bl,7;颜色属性<br>
mov bh,0;第0页<br>
mov cx,3;字符重复个数<br>
int 10h</p>
</blockquote>
<p>(ah)=9表示调用第10h号中断例程的9号子程序,功能为在光标位置显示字符,可以提供要显示的字符、颜色属性、页号、字符重复个数作为参数。</p>
<p>bl中的颜色属性的格式如下。</p>
<p><img src="https://img2020.cnblogs.com/blog/1162354/202109/1162354-20210909231858810-2024427928.png" alt="image" loading="lazy"></p>
<p>可以看出,和显存中的属性字节的格式相同。</p>
<p>编程:在屏幕的5行12列显示3个红底高亮闪烁绿色的“a”。</p>
<pre><code>assume cs:code

code segment

      mov ah,2      ;置光标
      mov bh,0      ;第0页
      mov dh,5      ;dh中放行号
      mov dl,12       ;dl中放列号
      int 10h

      mov ah,9      ;在光标位置显示字符
      mov al,'a'      ;字符
      mov bl,11001010b;颜色属性
      mov bh,0      ;第0页
      mov cx,3      ;字符重复个数
      int 10h

      mov ax,4c00h
      int 21h

code ends

end
</code></pre>
<p>注意,闪烁的效果必须在全屏DOS方式下才能看到。</p>
<hr>
<p>int 21h中断例程是DOS提供的中断例程,其中包含了DOS提供给程序员在编程时调用的子程序。</p>
<p>我们前面一直使用的是int 21h中断例程的4ch号功能,即程序返回功能,如下:</p>
<blockquote>
<p>mov ah,4ch&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;;程序返回<br>
mov al,0&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; ;返回值<br>
int 21h</p>
</blockquote>
<p>(ah)=4ch表示调用第21h号中断例程的4ch号子程序,功能为程序返回,可以提供返回值作为参数。</p>
<p>我们前面使用这个功能的时候经常写做:</p>
<blockquote>
<p>mov ax,4c00h<br>
int 21h</p>
</blockquote>
<p>我们看一下int21h中断例程在光标位置显示字符串的功能:</p>
<blockquote>
<p>ds:dx指向字符串&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;;要显示的字符串需用"$"作为结束符<br>
mov ah,9&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;   ;功能号9,表示在光标位置显示字符串<br>
int 21h</p>
</blockquote>
<p>(ah)=9表示调用第21h号中断例程的9号子程序,功能为在光标位置显示字符串,可以提供要显示字符串的地址作为参数。</p>
<p>编程:在屏幕的5行12列显示字符串"Welcome to masm!”。</p>
<pre><code>assume cs:code

data segment
    db 'Welcome to masm','$'
data ends

code segment

start:mov ah,2      ;置光标
      mov bh,0      ;第0页
      mov dh,5      ;dh中放行号
      mov dl,12       ;dl中放列号
      int 10h

      mov ax,data
      mov ds,ax
      mov dx,0      ;ds:dx指向字符串的首地址data:0
      mov ah,9
      int 21h

      mov ax,4c00h
      int 21h

code ends

end start
</code></pre>
<p>上述程序在屏幕的5行12列显示字符串"Welcome to masm!”,直到遇见"$" $本身并不显示,只起到边界的作用。</p>
<p>如果字符串比较长,遇到行尾,程序会自动转到下一行开头处继续显示;如果到了最后一行,还能自动上卷一行。</p>
<p>DOS为程序员提供了许多可以调用的子程序,都包含在int 21h中断例程中。我们这里只对原理进行了讲解,对于DOS提供的所有可调用子程序的情况,读者可以参考相关的书籍。</p><br><br>
来源:https://www.cnblogs.com/jpSpaceX/p/15221261.html
頁: [1]
查看完整版本: [汇编]《汇编语言》第13章 int指令