以下面这段简单代码为例子
#include <stdio.h>
int Add(int x, int y)
{
int z = 0;
z = x + y;
return z;
}
int main()
{
int a = 10;
int b = 20;
int c = 0;
c = Add(a, b);
printf("c = %d\n", c);
return 0;
}
(注意我们对代码的执行,以及调试都在 VC6.0环境下进行)
我们对代码进行调试,查看一下函数调用堆栈
查看方式:view->Debug->call stack
调用堆栈可以清楚的看到函数被调用的详细过程。
我们发现原来main函数也是被调用的,而调用它的就是mainCRTStartup函数。
函数调用是一个过程,我们称这个过程为函数调用过程
每一次函数(无论什么函数)的调用都会开辟一块大的空间,用于本次函数的调用中临时变量的保存,现场保护。我们把这块空间就叫做函数的调用堆栈或者函数栈帧。
而栈帧的维护需要用到ebp和esp两个寄存器。在函数调用过程中这两个寄存器维护栈的栈底和栈顶指针。
比如调用main函数,我们为main函数分配栈空间,栈帧维护如下
图1:
ebp存放了指向函数栈帧栈底的地址
esp存放了指向函数栈帧栈顶的地址
我们要看函数调用过程,就要对应反汇编代码。
详细过程如图
图中相同颜色的文字描述和方框对应了相应相同颜色的过程。
如有不足还请大家多多指正。