使用call和ret实现汇编中的子程序

在汇编中,我们可以使用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

关于字母大小写的转换的方法,参考本站的另外一篇文章:汇编中的大小写字母转换的方法

arthinking wechat
欢迎关注itzhai公众号