10.《汇编语言》-王爽第三版学习笔记 CALL 和 RET指令
1. RET && RETF 指令
-
CPU 执行 ret 指令时,进行下面两部操作:
(1)(IP) = ((SS)*16+(SP))
(2)(SP)=(SP)+2 -
CPU 执行 retf 指令时,进行下面两部操作:
(1)(IP) = ((SS)16+(SP))
(2)(SP)=(SP)+2
(3)(CS) = ((SS)16+(SP))
(4)(SP)=(SP)+2k -
可以看出,如果用汇编语法来解释 ret,retf 指令,则:
CPU 执行 ret 指令时,相当于:
pop IP
CPU 执行 retf 指令时,相当于:
pop IP
pop CS
2. CALL 指令
-
CPU 执行 call 指令时,进行下面两部操作:
(1) 将当前的 IP 或 CS 和 IP 压入 栈中;
(2) 转移; -
call指令不能实现短转移,call指令实现转移的方法和jmp指令的原理相同。
2. 依据位移进行转移的 CALL 指令
- call 标号(将当前的 IP 压栈后,转移到标号处执行指令。)
CPU 执行此种格式的 call 指令后,会进行如下的操作:
(1)(SP)=(SP)-2
((SS)*16+(SP))=(IP)
(2)(IP) = (IP) + 16位位移
相当于:
push IP
jmp near ptr 标号
3. 转移的目的地址在指令中的 CALL 指令
call far ptr 标号 实现的是段间转移
CPU 执行此种格式的 call 指令后,会进行如下的操作:
(1)(SP)=(SP)-2
((SS)16+(SP))=(CS)
(SP)=(SP)-2
((SS)16+(SP))=(IP)
(2)(CS) = 标号所在段的段地址
(IP)= 标号所在段中的偏移地址
相当于:
push CS
push IP
jmp far ptr 标号
4. 位移地址寄存器中的 CALL 指令
call 16位reg
CPU 执行此种格式的 call 指令后,会进行如下的操作:
(1)(SP)=(SP)-2
((SS)*16+(SP))=(IP)
(IP)= (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 内存单元地址
5. CALL RET 配合使用
assume cs:code
code segment
main: :
:
call sub1
:
:
mov ax,4c00h
int 21h
sub1 :
:
call sub2
:
ret
sub2 :
:
ret
code ends
end main
5. mul 指令
- 格式如下:
mul reg
mul 内存单元 - 注意点:
(1)两个相乘的数:
两个相乘的数,要么都是8位,要么都是16位。
如果是8位,一个默认存放在 AL 中,另一个存放在 8 位reg 或者 内存字节单位中;
如果是16位,一个默认存放在 AX 中,另一个存放在 16 位reg 或者 内存字单位中;
(2)结果:
如果是8位乘法,结果默认存放在 AX 中;
如果是16位乘法,结果高位默认在DX中存放, 地位在AX中存放。
6. 参数和结果传递
assume cs:codesg,ds:datasg
datasg segment
dw 1,2,3,4,5,6,7,8
dd 0,0,0,0,0,0,0,0
datasg ends
codesg segment
start: mov ax,datasg
mov ds,ax
mov si,0
mov di,10h
mov cx,8
s: mov bx,[si]
call cube
mov [di],ax
mov [di].2,dx
add si,2
add di,4
loop s
mov ax,4c00h
int 21h
cube: mov ax,bx
mul bx
mul bx
ret
codesg ends
end start
6. 批量数据的传递
assume cs:codesg,ds:datasg
datasg segment
db 'conversation'
datasg ends
codesg segment
start: mov ax,datasg
mov ds,ax
mov si,0
mov cx,12
call capital
mov ax,4c00h
int 21h
capital: and byte ptr [si],11011111b
inc si
loop capital
ret
codesg ends
end start
6. 用栈传递参数
assume cs:codesg,ss:stacksg
stacksg segment
dw 8 dup (0)
stacksg ends
codesg segment
start: mov ax,stacksg
mov ss,ax
mov sp,10h
mov ax,1
push ax
mov ax,3
push ax
call difcube
mov ax,4c00h
int 21h
difcube: push bp
mov bp,sp
mov ax,[bp+4]
sub ax,[bp+6]
mov bp,ax
mul bp
mul bp
pop bp
ret 4
codesg ends
end start
7. 寄存器冲突问题
assume cs:codesg,ds:datasg
datasg segment
db 'word',0
db 'unix',0
db 'wind',0
db 'good',0
datasg ends
codesg segment
start: mov ax,datasg
mov ds,ax
mov bx,0
mov cx,4
s: mov si,bx
call capital
add bx,5
loop s
mov ax,4c00h
int 21h
capital: push cx
push si
change: mov cl,[si]
mov ch,0
jcxz ok
and byte ptr [si],11011111b
inc si
jmp short change
ok: pop si
pop cx
ret
codesg ends
end start
8. 显式字符串
在第8行的第三列,用绿色显式 data 段中的字符串:
assume cs:codesg,ds:datasg
datasg segment
db 'welcome to masm!',0
datasg ends
codesg segment
start: mov dh,8
mov dl,3
mov cl,2
mov ax,datasg
mov ds,ax
mov si,0
call show_str
mov ax,4c00h
int 21h
show_str: push cx
push si
push di
push bx
mov ax,0B800H
mov es,ax
mov di,0
mov al,160
mul dh
mov bx,ax
mov al,2
mul dl
add ax,bx
mov bx,ax
mov ah,cl
show: mov cl,ds:[si]
mov ch,0
jcxz ok
mov al,ds:[si]
mov es:[bx+di],al
mov es:[bx+di+1],ah
inc si
add di,2
jmp short show
ok: pop bx
pop di
pop si
pop cx
ret
codesg ends
end start
8. 进行不会产生溢出的出发运算
assume cs:codesg
codesg segment
start: mov ax,4240H
mov dx,000FH
mov cx,0AH
call divdw
mov ax,4c00h
int 21h
divdw: push bx
mov bx,ax
mov ax,dx
mov dx,0000H
div cx
push ax
mov ax,bx
div cx
mov cx,dx
pop dx
pop bx
ret
codesg ends
end start