CALL和RET指令
call和ret指令都是转移指令,它们经常被共同用来实现子程序的设计。
ret和retf
ret指令用栈中的数据实现修改IP的内容,从而完成近转移,执行ret指令时相当于执行:
pop IP
retf指令用栈中的数据实现修改CS和IP的内容,从而完成远转移,执行retf指令相当于执行:
pop IP
POP CS
在程序中就可以使用这两个指令来转移:
code segment
mov ax,4c00h
int 21h
start: ...
...
ret
code ends
end start
这样由于ret指令的执行,就会让CS:IP指向代码段的第一条指令,完成返回。
call指令
call指令执行后进行两步操作:
1、将当前的IP或CS和IP压入栈中
2、转移
转到标号处
call指令可以转到标号处,同时将当前的IP压入栈中:
call 标号
相当于执行:
push IP
jmp near ptr 标号
这个指令对应的机器码中有相对于当前IP的转移位移。
对应的段间转移指令:
call far ptr 标号
相当于执行:
push CS
push IP
jmp far ptr 标号
这个指令对应的机器码中有转移的目的地址。
使用内存地址和寄存器
call指令后也可以跟寄存器:
call 16位reg
相当于执行:
push IP
jmp 16位reg
call指令后也可以加内存地址,一般有两种形式:
1、段内转移:
call word ptr 内存单元地址
相当于执行:
push IP
jmp word ptr 内存单元地址
2、段间转移:
call dword ptr 内存单元地址
相当于执行:
push CS
push IP
jmp dword ptr 内存单元地址
实现子程序机制
结合ret和call指令我们可以总结一个实现子程序的框架:
assume cs:code
code segment
main: ...
...
call sub1 调用了子程序sub1
...
...
mov ax,4c00h
int 21h
sub1: ...
...
call sub2 调用了子程序sub2
...
ret sub1返回
sub2: ...
...
ret sub2返回
code ends
end main
乘法指令mul
使用格式:
mul reg
mul 内存单元
涉及内存单元时,可以中间加word ptr或byte ptr来指定处理的数据长度。
乘法指令中两个相乘的数要么都是8位,要么都是16位;如果都是8位,则一个默认放在AL寄存器中;如果都是16位,则一个默认放在AX寄存器中。
如果是8位乘法那么结果默认放在AX中,如果是16位乘法,那么高位默认放在DX中,低位放在AX中。
模块化程序设计
有了子程序我们就可以实现汇编语言编程的模块化设计。
参数和结果传递的问题
在使用子程序时有两个问题:参数存在哪里?结果存在哪里?
最常见的方法是使用寄存器,调用子程序者将参数送入参数寄存器,从结果寄存器中取到返回值;子程序从参数寄存器中取到参数,将返回值送入结果寄存器。
多个参数和多个结果的传递
如果我们要传递更多的参数怎么办?我们不可能都用寄存器来完成传递,因为寄存器的个数终究是有限的。
这个时候常用的做法是将批量数据放到内存中,然后将他们所在内存空间的首地址放在寄存器中,传递给需要的子程序。对于多个返回结果,我们也可以用同样的方法。
栈也可以用来完成这一类任务。
寄存器冲突问题
有时程序中会出现寄存器冲突问题,如在一个循环中调用某个子程序,子程序中也有循环,这些循环共用一个cx寄存器来计数,就会导致循环次数的混乱。此时正确的做法应该是在子程序的开始将子程序中所有用到的寄存器中的内容都保存起来,在子程序返回前都恢复。
案例需求:把数据段中的所有字符串都转换为大写:
assume cs:code
data segment
db 'word',0
db 'unix',0
db 'wind',0
db 'good',0
data ends
此时我们需要写一个子程序来处理单个字符串转换为大写:(ds:si指向字符串的首地址)
capital:mov cl,[si]
mov ch,0
jcxz ok 如果cx为0,就结束
and byte ptr [si],11011111b 否则就将该字母转换为大写,然后再将si自增
inc si
jmp short capital
ok: ret
但是这个子程序没有完成环境保存,所以需要再改动一下,加上数据恢复和取用:
capital:push cx
push si
change: mov cl,[si]
mov ch,0
jcxz ok 如果cx为0,就结束
and byte ptr [si],11011111b 否则就将该字母转换为大写,然后再将si自增
inc si
jmp short change
ok: pop si
pop cx
ret
然后再调用该子程序即可:
code segment
start: mov ax,data
mov ds,ax
mov bx,0 指定总的循环次数
mov cx,4
s: mov si,bx 处理字符串,然后将指针加5,处理下一个字符串
call capital
add bx,5
loop s
mov ax,4c00h
int 21h
capital:push cx
push si
change: mov cl,[si]
mov ch,0
jcxz ok 如果cx为0,就结束
and byte ptr [si],11011111b 否则就将该字母转换为大写,然后再将si自增
inc si
jmp short change
ok: pop si
pop cx
ret
code ends
end start