张狂的暑假学习之 汇编入门学习笔记 (九)—— 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