|
发表于 2012-2-15 02:12:01
|
显示全部楼层
你的子程序show_str错的太多,还是用我的这个吧!
;名称:show_str
;功能:在指定的位置,用指定的颜色,显示一个用0结束的字符串。
;参数:dh=行号(取值范围0—24),dl=列号(取值范围0-79),cl=颜色,
;ds:si指向字符串的首地址。返还:无。
;应用举例:在屏幕的8行3列,用绿色显示data段中的字符串。
assume cs:codesg,ds:datasg,ss:stacksg
datasg segment
db 'welcome to masm!',0 ;定义一个字符串
datasg ends
stacksg segment
dw 8 dup(0) ;定义一个空栈
stacksg ends
codesg segment
start: mov dh,8h
mov dl,3h
mov cl,2h
mov ax,datasg
mov ds,ax
mov ax,stacksg
mov ss,ax
mov sp,16
mov si,0
call show_str ;调用子函数
mov ax,4c00h
int 21h
show_str: ;显示字符串的子程序(定义开始)。
push cx
push si
mov al,0a0h ;每行有160个字节,转换成16进制为A0。
dec dh ;行号在显存中下标从0开始,所以自减1.
mul dh ;相当于从第(n-1)*0A0H个Byte单元开始......
mov bx,ax ;定位好的位置偏移地址存放在bx里(行)
mov al,2 ;每个字符占两个字节
mul dl ;定位列,结果ax存放的是定位好的列的位置
sub ax,2 ;列号在显存中下标从0开始,又因为偶字节存放字符,所以减2
add bx,ax ;此时bx中存放的是行与列号的偏移地址
mov ax,0B800h ;显存开始的地址
mov es,ax ;es中存放的是显存的第0页(共0--7页)的起始的段地址
mov di,0 ;di指向显存的偏移地址,确定指向下一个要处理的字符的位置
mov al,cl ;cl是存放颜色的参数,这时候al存放颜色了,因为cl下边要用来临时存放要处理的字符
mov ch,0 ;下边cx存放的是每次准备处理的字符
s: mov cl,ds:[si] ;ds:[si]指向“Welcome to masm!”,0
jcxz ok ;当cl的值为0时候,cx == 0, 则发生跳转,到OK处结束处理!
mov es:[bx+di],cl ;偶地址存放字符
mov es:[bx+di+1],al ;奇地址存放字符的颜色属性
inc si
add di,2 ;指向了下个字符
jmp short s ;无条件跳转,jcxz是离开的关键跳!
ok:pop si
pop cx
ret ;显示字符串的子程序[定义结束]
codesg ends
end start
|
|