CPU执行call指令时,进行两步操作:
- 将当前的IP或CS和IP压入栈中;
- 转移。
call指令不能实现短转移,除此之外,call指令实现转移的方法和jmp指令的原理相同。
1)依据位移进行转移的call指令
call 标号(将当前的IP压栈后,转到标号处执行指令)
CPU执行此种格式的call指令时,进行如下的操作:
- (sp)=(sp)-2
((ss)*16+(sp))=(IP)
- (IP)=(IP)+16位位移
16位位移=标号处的地址-call指令后的第一条字节的地址
16位位移的范围为-32768~32767,用补码表示
16位位移由编译程序在编译时算出。
2)转移目的地址在指令中的call指令
"call far ptr 标号"实现的是段间转移。
CPU执行此种格式的call指令时,进行如下的操作。
- (sp)=(sp)-2
((ss)*16+(sp))=(cs)
(sp)=(sp)-2
((ss)*16+(sp))=(IP)
- (CS)=标号所在段的段地址
(IP)=标号在段中的偏移地址
3)转移地址在寄存器中的call指令
指令格式: call 16位寄存器
功能:
(sp)=(sp)-2
((ss)*16+(sp))=(IP)
(IP)=(16位寄存器)
4)转移地址在内存中的call指令
a)call word ptr 内存单元地址
功能:
(sp)=(sp)-2
((ss)*16+(sp))=(IP)
(IP)=(内存单元地址)
b)call dword ptr 内存单元地址
功能:
- (sp)=(sp)-2
((ss)*16+(sp))=(cs)
(sp)=(sp)-2
((ss)*16+(sp))=(IP)
- (IP)=(内存单元地址)