汇编入门学习笔记 (九)—— call和ret
参考: 《汇编语言》 王爽 第10章
call和ret都是转移指令。
1. ret和retf
ret指令:用栈中的数据,修改IP内容,从而实现近转移
相当于:
pop ip
retf指令:用栈中的数据,修改CS和IP,从而实现远转移
相当于:
pop ip
pop cs
例子:ret
- assumecs:code,ss:stack
- stacksegment
- db16dup(1)
- stackends
- codesegment
- movax,4c00H
- int21H
- start:movax,stack
- movss,ax
- movsp,16
- movax,0
- pushax
- ret
- codeends
- endstart
retf
- assumecs:code,ss:stack
- stacksegment
- db16dup(1)
- stackends
- codesegment
- movax,4c00H
- int21H
- start:movax,stack
- movss,ax
- movsp,16
- movax,0
- pushcs
- pushax
- retf
- codeends
- endstart
2. call指令
call指令,执行操作:
1.将当前IP或CS和IP压入栈中
2.跳转
(1)依据位移进行转移的call指令
格式: call 标号
将下一条的指令的ip压入栈中,在转到标号处
相当于:
push ip
jmp near ptr 标号
(2)转移的目的地址在指令中的call指令
格式:
call far ptr 标号
将下一条的指令的CS和IP压入栈中,在转到标号处
相当于:
push cs
push ip
jmp far ptr
(3)转移地址地址在寄存器中的call指令
格式:call 16位reg
相当于:
push ip
jmp 16位reg
(4)转移地址在内存中的call指令
1. call word ptr 内存单元
相当于:
push ip
jmp word ptr 内存单元
2. call dword ptr 内存单元
相当于:
push cs
push ip
jmp dword ptr 内存单元
3. mul 指令
mul 是乘法指令
表示两个数相乘,它必须是都是8位或者都是16位
8位相乘 结果默认存放在ax中
16位相乘 结果高位存放在dx中,低位存放在ax中
例子见下面。
3. call和ret配合使用
call于ret结合使用,就相当于函数。
例子:求dw中数值的3次方。把bx当做“函数”参数,ax当做“函数”的返回值。
- assumecs:code,ds:data
- datasegment
- dw1,2,3,4,5,6,7,8
- dd0,0,0,0,0,0,0,0
- dataends
- codesegment
- start:movax,data
- movds,ax
- movsi,0
- movdi,16
- movcx,8
- s:movbx,ds:[si]
- callcube
- movds:[di],ax
- movds:[di+2],dx
- addsi,2
- adddi,4
- loops
- movax,4c00H
- int21H
- cube:movax,bx
- mulbx
- mulbx
- ret
- codeends
- endstart
寄存器数量有限,如果要传的参数,或者返回的参数过多。可以使用内存,或者栈。
例子:小写转大写。(用内存存放参数)
- assumecs:code,ds:data
- datasegment
- dbconversation
- dataends
- codesegment
- start:movax,data
- movds,ax
- movsi,0
- movcx,12
- callcaptial
- movax,4c00H
- int21H
- captial:andbyteptrds:[si],11011111b
- incsi
- loopcaptial
- codeends
- endstart
例子:计算 (a - b) ^3 假设a=3,b=1 (用栈来存放参数)
- assumecs:code
- codesegment
- start:movax,1
- pushax
- movax,3
- pushax
- calldifcube
- movax,4c00H
- int21H
- difcube:pushbp
- movbp,sp
- movax,[bp+4]
- subax,[bp+6]
- movbp,ax
- mulbp
- mulbp
- popbp
- ret4
- codeends
- endstart
上面代码中的 ret 4 表示:
pop ip
add sp,n
例子:小写转大写,用0结尾来判断。(用栈来处理寄存器冲突)
- assumecs:code,ds:data
- datasegment
- dbword,0
- dbcity,0
- dbgood,0
- dataends
- codesegment
- start:movax,data
- movds,ax
- movcx,3
- movbx,0
- s:pushcx
- movsi,bx
- callcapital
- addbx,5
- popcx
- loops
- movax,4c00H
- int21H
- capital:movcl,[si]
- movch,0
- jcxzok
- andbyteptr[si],11011111b
- incsi
- jmpshortcapital
- ok:ret
- codeends
- endstart
注意:要用栈保存cx
例子:实现show_str “函数” 在屏幕显示字符串。用dh指定函数 ,dl指定列号,cl指定颜色
- assumecs:code,ds:data,ss:stack
- datasegment
- dbWelcometomasm!,0
- dataends
- stacksegment
- dw8dup(0)
- stackends
- codesegment
- start:movax,data
- movds,ax
- movax,stack
- movss,ax
- movsp,16
- movdh,10;行
- movdl,17;列
- movcl,2;颜色
- movsi,0
- callshow_str
- movax,4c00h
- int21h
- show_str:pushax
- pushdi
- pushdx
- movax,10;确定行段es
- muldh
- addax,0b800h
- moves,ax
- movdh,0;确定列偏移di,注意,一个字符两个字节
- adddx,dx
- movdi,dx
- s:pushcx;保存cx
- movch,0
- movcl,ds:[si]
- jcxzok;如果为0跳转
- moves:[di],cl
- popcx
- moves:[di+1],cl
- incsi
- adddi,2
- jmpshorts
- ok:popcx;不要忘记pop,眼不让rec还原的ip就不对了
- popdx
- popdi
- popax
- ret
- codeends
- endstart
汇编入门callre 相关文章:
- Windows CE 进程、线程和内存管理(11-09)
- RedHatLinux新手入门教程(5)(11-12)
- uClinux介绍(11-09)
- openwebmailV1.60安装教学(11-12)
- Linux嵌入式系统开发平台选型探讨(11-09)
- Windows CE 进程、线程和内存管理(二)(11-09)