在这次实验中,我们将要编写3个子程序,通过它们来认识几个常见的问题和掌握解 决这些问题的方法。同前面的所有实验一样,这个实验是必须独立完成的,在后面的课程 中,将要用到这个实验中编写的3个子程序。
1. 显示字符串
问题
显示字符串是现实工作中经常要用到的功能,应该编写一个通用的子程序来实现这个功 能。我们应该提供灵活的调用接口,使调用者可以决定显示的位置(行、列)、内容和颜色。
子程序描述
名称:show str
功能:在指定的位置,用指定的颜色,显示一个用0结束的字符串。
参数:(dh)=行号(取值范围0~24),(dl)=列号(取值范围0~79),
(cl)=颜色,ds:si指向字符串的首地址
返回:无
应用举例:在屏幕的8行3列,用绿色显示 data段中的字符串。
提示
(1)子程序的入口参数是屏幕上的行号和列号,注意在子程序内部要将它们转化为显存中的地址,首先要分析一下屏幕上的行列位置和显存地址的对应关系;
(2)注意保存子程序中用到的相关寄存器;
(3)这个子程序的内部处理和显存的结构密切相关,但是向外提供了与显存结构无关的接口。通过调用这个子程序,进行字符串的显示时可以不必了解显存的结构,为编程提 供了方便。在实验中,注意体会这种设计思想。
;t10_sy1.asm
assume cs:code
data segment
db 'Welcome to masm!',0
data ends
code segment
start: mov dh, 8 ;第8行
mov dl, 3 ;第3列
mov cl, 2 ;颜色值 绿色
mov ax, data
mov ds, ax
mov si, 0
call show_str
mov ax, 4c00h
int 21h
show_str: push ax
push bx
push es
push si
mov ax, 0b800h
mov es, ax
mov ax, 160
mul dh ;8位乘法, 默认值放在al, 乘数放在dh中,结果:放在ax中 ax=al*dh=160*8=1280=0500h
mov bx, ax ;bx=160*dh
mov ax, 2
mul dl ;8位乘法, 默认值放在al, 乘数放在dl中,结果:ax=dl*2=3*2=6
add bx, ax ;mov bx, (160*dh + dl*2)设置es:bx指向显存首地址
mov al, cl ;把颜色cl赋值al
mov cl, 0
show0: mov ch, [si]
jcxz show1 ;(ds:si)=0时,转到show1执行
mov es:[bx], ch
mov es:[bx].1, al
inc si ;ds:si指向下一个字符地址
add bx, 2 ;es:bx指向下一个显存地址
jmp show0
show1: pop si
pop es
pop bx
pop ax
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 的一个内部错误,这个错误被称为: 除法溢出。 我们可以通过特殊的程序来处理 这个错误,但在这里我们不讨论这个错误的处理,这是后面的课程中要涉及的内容。下面 我们仅仅来看一下除法溢出发生时的一些现象,如下图所示
这上面可以看出,执行div指令后,执行位置就错误了。