在进行函数的调用(栈帧)的分析前,我们先了解下 main 函数的调用吧,通过以前的学习,我们知道 main 函数也是被调用的,先在 __tmainCRTStartup 函数中调用,而 __tmainCRTStartup 函数是在 mainCRTStartup 中被调用的。
恰如这样的调用,我们发现每一次函数调用都是一个过程,这个过程我们称之为:函数的调用过程。
函数调用过程要为函数开辟栈空间,用于本次函数的调用中临时变量的保存、现场保护。这块栈空间我们称之为函数栈帧。
栈帧的维护需要我们了解 ebp 和 esp 两个寄存器,在函数调用的过程中这两个寄存器存放了维护这个栈的栈底和栈顶指针。ebp 存放了指向函数栈帧栈底的地址;esp 存放了指向函数栈帧栈顶的地址。
知道了这些概念后,我们来重点通过反汇编代码了解下函数的调用过程。
首先给出源代码如下:
#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
int Add(int x,int y)
{
int sum = 0;
sum = x + y;
return sum;
}
int main()
{
int a = 10;
int b = 20;
int ret = Add(a, b);
system("pause");
return 0;
}
进入调试,转到反汇编:
1. 从main函数的地方开始,要展开main 函数的调用就得为 main 函数创建栈帧,那我们先来看 main 函数栈帧的创建。
重复拷贝值,初始化,验证如下:
2. Add 函数的调用,参数传递过程如下。
进入 call 指令后,继续调试,按 F11,跳转如下:
再按 F11 就进入 Add 函数的执行代码处。
按 F11 后跳转到 add 指令处。
以上就是关于函数调用及栈帧创建和销毁的代码文字分析,下面我画图解释一下函数的调用及栈帧(运行时堆栈)的创建。
此处图形并没有销毁的过程,栈帧大致就是这样的创建过程。
希望我总结的关于函数调用过程及栈帧方面的知识能够对你有所帮助。