实验8汇编实验
- 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
- 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
- 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。
实验八编写子程序(4学时)
在本次实验中,我们将编写3个子程序,通过它们来认识几个常见的问题和掌握解决这些问题的方法。同前面的所有实验一样,这个实验室必须独立完成的,在后面的课程中,将要用到这个实验中编写的3个子程序。
1、显示字符串
显示字符串是下现实工作中经常要用到的功能,应该编写一个通用的字程序来实现这个功能。我们应该提供灵活的调用接口,使调用者可以决定显示的位置(行、列)、内容和颜色。
该子程序描述如下:
名称:show_str
功能:在指定位置用指定颜色显示一个用0结束的字符串。
参数:(dh)=行号(取值范围0~24),(dl)=列号(取值范围0~79),(cl)=颜色,ds:si 指向字符串的首地址。
返回:无
应用举例:在屏幕的8行3列,用绿色显示data段中的字符串。
assume cs: code
data segment
db ‘Welcome to masm!’, 0
data ends
code segment
start: mov dh, 8
mov dl, 3
mov cl, 2
mov ax, data
mov ds, ax
mov si, 0
call show_str
mov ax, 4c00h
int 21h
show_str: :
:
:
code ends
end start
提示:
(1)子程序的入口参数是屏幕上的行号和列号,注意在子程序内部要将它们转化为显存中的地址,首先要分析一下屏幕上的行列位置和显存地址的对应关系;
(2)注意保存子程序中用到的相关寄存器;
(3)这个子程序的内部处理和显存的结构密切相关,但是向外提供了与显存结构无关的接口。通过调用这个子程序,进行字符串的显示时可以不必了解显存的结果,为编程提供了方便。在实验中,注意体会这种设计思想。
data segment
db "Welcome to masm!"
data ends
stack segment
db "Welcome to masm!" stack ends
code segment
start:
mov dh, 8
mov dl, 3
mov cl, 2
mov ax, data
mov ds, ax
mov si, 0
mov ax,0B800h
mov es,ax
mov ax, stack
mov ss, ax
mov sp,10h
call show_str
mov ax, 4c00h
int 21h
show_str:
push dx
push cx
mov al,160
dec dh
mul dh
mov bx,ax
add dx,si
add dl,dl
add bl,dl
mov cl,[si]
mov ch,0
jcxz ok
mov es:[bx],cl
pop cx
mov es:[bx+1],cl
inc si
pop dx
jmp short show_str ok:
pop cx
pop dx
ret
code ends
end start
2、解决除法溢出的问题
前面讲过,div指令可以做除法。当进行8位除法的时候,用al存储商,ah存储余数;进行16位除法时,用ax存储商,dx存储余数。可是,现在有一个问题,如果结果的商大于al或ax所能存储的最大值,那么将如何?
比如,下面的程序段:
mov bh, 1
mov ax, 1000
div bh
进行的是8位除法,商为1000,而1000在al中放不下。
又比如,下面的程序段:
mov ax, 1000H
mov dx, 1
mov bx, 1
div bx
进行的是16位除法,商为11000H,而11000H在ax中放不下。
我们在用div指令做除法的时候,和可能发生上面的情况:结果的商过大,超出了寄存器所能存储的范围。当CPU执行div等除法指令时,如果发生这样的情况,将引发CPU的一个内部错误,这个错误被称为:除法溢出。我们可以通过特殊的程序来处理这个错误,但在这里我们不讨论这个错误的处理,这是后面课程中要涉及的内容。下面我们仅仅来看一下除法发生时的一些现象,如同8.1所示。
图8.1 除法溢出时发生的现象
图中展示了在Windows 2000中使用Debug执行相关程序段的结果,div指令引发了CPU的除法溢出,系统对其进行了相关的处理。
至此,我们已经清楚了问题所在;用div指令做除法的时候可能产生除法溢出。由于有这样的问题,在进行除法运算时要注意除数和被除数的值,比如1000000/10就不能用div指令来计算。那么怎么办呢?我们用下面的子程序divdw解决。
该子程序的描述如下:
名称:divdw
功能:进行不会产生溢出的除法运算,被除数为dword型,除数为word型,结果为dword型。
参数: (ax)=dword型数据的低16位;(dx)=dword型数据的高16位;(cx)=除数返回:(ax)=商的低16位;(dx)=商的高16位;(cx)=余数
应用举例:计算1000000/10(F424H/0AH)
mov ax, 4240H
mov dx, 000FH
mov cx, 0AH
call divdw
结果:(ax)=86A0H,(dx)=0001H, (cx)=0.
提示:
给出一个公式:
X: 被除数,范围:[0,FFFFFFFF]
N: 除数,范围: [0,FFFF]
H: X高16位,范围: [0,FFFF]