汇编网首页登录博客注册
masmaster的学习博客
博客首页博客互动【做检测题】论坛求助

我的博客

个人首页 |  我的文章 |  我的相册 |  我的好友 |  最新访客 |  文章收藏 |  论坛提问 |  友情链接 |  给我留言  
图片载入中
文章收藏

[2010-07-18 15:14] 子程序指令:call/ret

子程序通常是执行一个任务的代码组。 是程序结构中非常重要的组成部分。 子程序存储在存储器中,可以重复使用(调用)的一段程序,子程序的缺点就是从调用到返回时需要花费少量时间。这一点,没有使用宏有优势。call指令调用子程序,而ret指令从子程序返回。
每当程序调用子程序时,利用栈存储子程序返回的地址。call指令将它后面的指令的地址push入栈。ret指令从栈中取回这个地址, 这样就返回到call后面的指令,使程序继续执行。
在汇编程序中队子程序的存放有特殊的规定。 子程序 要以伪指令proc开始并且以伪指令endp结束。每个伪指令必须和子程序名字一起出现。proc后面是子程序的类型:far或者near,near是段内的调用,far是段间的调用。near或far后还可以有伪指令uses,uses雨具允许子程序中任何数目的寄存器自动入栈或自动弹出。格式如下:
------------------------------------------------------
my    proc    near/far    uses    ax bx cx dx
...
...
ret
my    endp
------------------------------------------------------
使用near和far返回的操作码不同。 段内(near)返回的操作码是c3h,而段间(far)返回的操作码是cbh。段内返回从栈中移出16位数字放入IP,而段间返回从栈中取出的是32位地址,分别放入IP和CS从而实现返回。

call指令:
call指令不同于转移指令。因为call在栈中保存ret使用的返回地址。

近call:
近call指令3字长,第一字节为操作码,16位cpu的第二第三字节包含±32K字节的偏移量或距离。80386以上处理器按保护模式操作时用32位的偏移量。允许±2G字节的距离。
执行近call时先将下一条指令的偏移地址压入栈中。存储完这个地址后,将第二第三字节的偏移量与ip相加,从而跳转到新的ip地址去执行。

远call:
远call指令类似远转移, 因为他能够调用存放在存贮器中任意地址的程序。远call有5个字节长,操作码后面跟的是IP和CS的内容。第二三字节包含IP内容,第四五字节包含CS内容。
远call指令在转移到由字节2~5指示的地址之前,先把IP和CS的内容压入堆栈。这就允许远call指令调用位于存储器任何位置的程序。然后从这个子程序返回。

带有寄存器操作数的call:
类似转移, call指令也可以用寄存器作它的操作数。例如:call ax ;这条指令先将当前的IP的内容入栈。然后跳至当前代码段中位于AX寄存器中的偏移地址处执行程序。这种call指令总是使用除了段寄存器之外的任何16位寄存器中的16位偏移地址。

用间接存储器寻址的call指令:
当程序中要求选择使用不同的子程序时,用间接存储器寻址的call指令非常恰当。比如: call table[bx] ;

ret指令:
返回指令(ret)从栈中取出16位数字(near)放入ip或者取出32位数字(far)把他们放入IP和CS中。proc语句定义了是近的还是远的调用。这样就自动选择了合适的返回指令。
当改变ip或cs时,下一条指令地址指向存储器新的位置。
这个新的地址是紧跟在最近call指令过程的指令之后的地址。
另外一种形式的返回指令,是在SP的内容加上一个数以后再从栈中取回返回地址。
评论次数(0)  |  浏览次数(664)  |  类型(汇编语言笔记) |  收藏此文  | 
 
 请输入验证码  (提示:点击验证码输入框,以获取验证码