在汇编中,我们可以使用call和ret实现子程序的机制。ret指令使用栈中的数据,修改IP的内容,从而实现近转移。retf指令使用栈中的数据,修CS和IP的内容,从而实现远转移。call指令执行时需要进行两步的操作:
- ① 将当前的IP或CS和IP压入栈中;
- ②转移。 子程序的框架如下:
汇编代码
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22
| assume cs:code code segment main: : : call sub1 ;调用子程序sub1 : : mov ax,4c00h int 21h
sub1: : ;子程序开始 : call sub2 ;调用子程序sub2 : : ret ;子程序返回
sub2: : ;子程序sub2开始 : ret ;子程序返回 code ends ;结束代码段 end main
|
以下是一个例子:将一个由字母和以0结尾的字符串,转化为大写。
汇编代码
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41
| ;说明:将一个由字母和以0结尾的字符串转换为大写 ;参数:ds:si指向字符串的首地址 ;结果:无返回值 assume cs:code data segment db 'Java',0 db 'good',0 db 'unix',0 db 'yeah',0 data ends
code segment start:mov ax,data mov ds,ax mov bx,0
mov cx,4 ;循环4次 s:mov si,bx ;设置每次循环时的字母所在的偏移地址 call capital ;调用capital子程序转换成大写 add bx,5 ;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 ;and操作转换字母为大写 inc si ;si加1 jmp short change ;跳到change继续处理
ok:pop si pop cx ;si,cx出栈恢复原来的值 ret ;返回子程序继续处理 code ends
end start
|
关于字母大小写的转换的方法,参考本站的另外一篇文章:汇编中的大小写字母转换的方法