新闻中心

EEPW首页>嵌入式系统>设计应用> 汇编入门学习笔记 (九)—— call和ret

汇编入门学习笔记 (九)—— call和ret

作者: 时间:2016-11-09 来源:网络 收藏
疯狂的暑假学习之汇编入门学习笔记 (九)——callret

本文引用地址://m.amcfsurvey.com/article/201611/317783.htm

参考: 《汇编语言》 王爽 第10章

call和ret都是转移指令。

1. ret和retf

ret指令:用栈中的数据,修改IP内容,从而实现近转移

相当于:

pop ip

retf指令:用栈中的数据,修改CS和IP,从而实现远转移

相当于:

pop ip

pop cs

例子:ret

  1. assumecs:code,ss:stack
  2. stacksegment
  3. db16dup(1)
  4. stackends
  5. codesegment
  6. movax,4c00H
  7. int21H
  8. start:movax,stack
  9. movss,ax
  10. movsp,16
  11. movax,0
  12. pushax
  13. ret
  14. codeends
  15. endstart

retf

  1. assumecs:code,ss:stack
  2. stacksegment
  3. db16dup(1)
  4. stackends
  5. codesegment
  6. movax,4c00H
  7. int21H
  8. start:movax,stack
  9. movss,ax
  10. movsp,16
  11. movax,0
  12. pushcs
  13. pushax
  14. retf
  15. codeends
  16. 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当做“函数”的返回值。

  1. assumecs:code,ds:data
  2. datasegment
  3. dw1,2,3,4,5,6,7,8
  4. dd0,0,0,0,0,0,0,0
  5. dataends
  6. codesegment
  7. start:movax,data
  8. movds,ax
  9. movsi,0
  10. movdi,16
  11. movcx,8
  12. s:movbx,ds:[si]
  13. callcube
  14. movds:[di],ax
  15. movds:[di+2],dx
  16. addsi,2
  17. adddi,4
  18. loops
  19. movax,4c00H
  20. int21H
  21. cube:movax,bx
  22. mulbx
  23. mulbx
  24. ret
  25. codeends
  26. endstart

寄存器数量有限,如果要传的参数,或者返回的参数过多。可以使用内存,或者栈。

例子:小写转大写。(用内存存放参数)

  1. assumecs:code,ds:data
  2. datasegment
  3. dbconversation
  4. dataends
  5. codesegment
  6. start:movax,data
  7. movds,ax
  8. movsi,0
  9. movcx,12
  10. callcaptial
  11. movax,4c00H
  12. int21H
  13. captial:andbyteptrds:[si],11011111b
  14. incsi
  15. loopcaptial
  16. codeends
  17. endstart

例子:计算 (a - b) ^3 假设a=3,b=1 (用栈来存放参数)

  1. assumecs:code
  2. codesegment
  3. start:movax,1
  4. pushax
  5. movax,3
  6. pushax
  7. calldifcube
  8. movax,4c00H
  9. int21H
  10. difcube:pushbp
  11. movbp,sp
  12. movax,[bp+4]
  13. subax,[bp+6]
  14. movbp,ax
  15. mulbp
  16. mulbp
  17. popbp
  18. ret4
  19. codeends
  20. endstart

上面代码中的 ret 4 表示:

pop ip

add sp,n

例子:小写转大写,用0结尾来判断。(用栈来处理寄存器冲突)

  1. assumecs:code,ds:data
  2. datasegment
  3. dbword,0
  4. dbcity,0
  5. dbgood,0
  6. dataends
  7. codesegment
  8. start:movax,data
  9. movds,ax
  10. movcx,3
  11. movbx,0
  12. s:pushcx
  13. movsi,bx
  14. callcapital
  15. addbx,5
  16. popcx
  17. loops
  18. movax,4c00H
  19. int21H
  20. capital:movcl,[si]
  21. movch,0
  22. jcxzok
  23. andbyteptr[si],11011111b
  24. incsi
  25. jmpshortcapital
  26. ok:ret
  27. codeends
  28. endstart

注意:要用栈保存cx

例子:实现show_str “函数” 在屏幕显示字符串。用dh指定函数 ,dl指定列号,cl指定颜色

  1. assumecs:code,ds:data,ss:stack
  2. datasegment
  3. dbWelcometomasm!,0
  4. dataends
  5. stacksegment
  6. dw8dup(0)
  7. stackends
  8. codesegment
  9. start:movax,data
  10. movds,ax
  11. movax,stack
  12. movss,ax
  13. movsp,16
  14. movdh,10;行
  15. movdl,17;列
  16. movcl,2;颜色
  17. movsi,0
  18. callshow_str
  19. movax,4c00h
  20. int21h
  21. show_str:pushax
  22. pushdi
  23. pushdx
  24. movax,10;确定行段es
  25. muldh
  26. addax,0b800h
  27. moves,ax
  28. movdh,0;确定列偏移di,注意,一个字符两个字节
  29. adddx,dx
  30. movdi,dx
  31. s:pushcx;保存cx
  32. movch,0
  33. movcl,ds:[si]
  34. jcxzok;如果为0跳转
  35. moves:[di],cl
  36. popcx
  37. moves:[di+1],cl
  38. incsi
  39. adddi,2
  40. jmpshorts
  41. ok:popcx;不要忘记pop,眼不让rec还原的ip就不对了
  42. popdx
  43. popdi
  44. popax
  45. ret
  46. codeends
  47. endstart



关键词:汇编入门callre

评论


技术专区

关闭