函数调用栈比较有意思
作者:liigo
原文链接:http://blog.csdn.net/liigo/archive/2006/12/23/1456938.aspx
转载请注明出处:http://blog.csdn.net/liigo
昨天和海洋一块研究了下函数调用栈,顺便写两句。不足或错误之处请包涵!
理解调用栈最重要的两点是:栈的结构,EBP寄存器的作用。
首先要认识到这样两个事实:
1、一个函数调用动作可分解为:零到多个PUSH指令(用于参数入栈),一个CALL指令。CALL指令内部其实还暗含了一个将返回地址(即CALL指令下一条指令的地址)压栈的动作。
2、几乎所有本地编译器都会在每个函数体之前插入类似如下指令:PUSH EBP; MOV EBP ESP;
即,在程序执行到一个函数的真正函数体时,已经有以下数据顺序入栈:参数,返回地址,EBP。
由此得到类似如下的栈结构(参数入栈顺序跟调用方式有关,这里以C语言默认的CDECL为例):
+| (栈底方向,高位地址) |
| .................... |
| .................... |
| 参数3 |
| 参数2 |
| 参数1 |
| 返回地址 |
-| 上一层[EBP] | <-------- [EBP]
“PUSH EBP”“MOV EBP ESP”这两条指令实在大有深意:首先将EBP入栈,然后将栈顶指针ESP赋值给EBP。“MOV EBP ESP”这条指令表面上看是用ESP把EBP原来的值覆盖了,其实不然——因为给EBP赋值之前,原EBP值已经被压栈(位于栈顶),而新的EBP又恰恰指向栈顶。
此时EBP寄存器就已经处于一个非常重要的地位,该寄存器中存储着栈中的一个地址(原EBP入栈后的栈顶),从该地址为基准,向上(栈底方向)能获取返回地址、参数值,向下(栈顶方向)能获取函数局部变量值,而该地址处又存储着上一层函数调用时的EBP值!
一般而言,ss:[ebp+4]处为返回地址,ss:[ebp+8]处为第一个参数值(最后一个入栈的参数值,此处假设其占用4字节内存),ss:[ebp-4]处为第一个局部变量,ss:[ebp]处为上一层EBP值。
由于EBP中的地址处总是“上一层函数调用时的EBP值”,而在每一层函数调用中,都能通过当时的EBP值“向上(栈底方向)能获取返回地址、参数值,向下(栈顶方向)能获取函数局部变量值”。
如此形成递归,直至到达栈底。这就是函数调用栈。
编译器对EBP的使用实在太精妙了。
从当前EBP出发,逐层向上找到所有的EBP是非常容易的:
unsigned int _ebp;
__asm _ebp, ebp;
while (not stack bottom)
{
//...
_ebp = *(unsigned int*)_ebp;
}
如果要写一个简单的调试器的话,注意需在被调试进程(而非当前进程——调试器进程)中读取内存数据。
分享到:
相关推荐
这是一篇有关编写函数调用原理的文章,通过大量的示意图从比较基础的概念开始分解函数调用时交替的完整变化过程,同时将通过特定示例来分析在X-64平台上函数调用在汇编级的表示,从而深刻理解函数调用原理。...
递归算法是通过函数调用自身来解决问题的方法,适用于解决具有自相似结构的问题,如斐波那契数列和汉诺塔问题。在C#中,理解递归的基本原理并正确设置递归基(结束条件)至关重要,以防止无限循环。 试探算法是一种...
类是引用类型,它的实例存储在堆中,可以有默认的无参构造函数,并且可以继承和实现接口。结构是值类型,它的实例存储在栈中或结构体内,不支持继承,但可以实现接口。类可以有任意数量的实例,而结构通常用于存储...
6. **递归**:通过实现斐波那契数列或汉诺塔游戏来演示递归的概念,这是函数调用自身的一种方法。 7. **文件操作**:创建、读取、修改和删除文件,这是任何软件开发中的基础技能。 8. **错误处理**:学习如何使用...
通过好玩的源代码,可以看到函数如何被调用和参数传递。 4. **类与对象**:面向对象编程的核心在于类和对象。类是数据和操作数据的方法的集合,而对象是类的实例。初学者可以通过源代码理解如何定义类、创建对象...
由于oSIP是比较底层的SIP协议栈实现,新手较难上手,而官方的示例大都是一些伪代码,需要有实际的例子程序参考学习,而最好的例子就是同样官方发布的oSIP的扩展开发库exosip2,使用exoSIP可以很...
描述中的“在aogo上看到的,呵呵,很有意思的哦,欢迎下载!!!”提示这是一个在线平台上的资源分享,表明这个项目可能是由个人或社区成员创建并分享的。它也表达了发布者对这个项目的兴趣和对读者的欢迎,暗示可能...
在解决汉诺塔问题时,虽然不需要特别复杂的数据结构,但基本的理解如栈、队列等概念仍然是必要的,因为它们帮助我们理解递归过程中的函数调用堆栈。另一方面,Java是实现这个算法的工具,它是一种广泛使用的面向对象...
在示例中,用户被要求输入人数`x`和报数间隔`y`,然后调用`Yosef`函数并打印结果。例如,当x=100,y=17时,最后留下的那个人的原始编号是52。 约瑟夫环问题可以通过递归、栈、队列等多种数据结构和算法来解决,每种...