!!!非常重要
以前我们知道在调用函数的时候会有寄存器来执行一些操作,常见的有ebp 、esp、eax、ebx、ecx等 而在函数调用过程中也会调用两个寄存器,一个是ebp,一个是esp。ebp指向的是栈底,esp指向的是栈顶。
我们那下面这个例子来分析函数调用过程的具体细节:
#include <stdio.h>
int Add(int x, int y)
{
int ret = 0;
ret = x + y;
return ret;
}
int main()
{
int n = 10;
int m = 20;
printf("%d\n", Add(n, m));
getchar();
return 0;
}
通过调试查看函数栈帧我们发现:
其实main函数在 _scrt_common_main 函数中调用的,而_scrt_common_main函数又是在mainCRTSartup被调用。
并且知道每一次函数调用都是一个过程------>函数调用过程
这个过程要为函数开辟栈空间,用于本次函数的调用中临时变量的保存、现场保护,也叫—>函数栈帧
而栈帧的维护也必须用到开始我们提到的ebp和esp两个寄存器。
下图是两个寄存器的初始值
研究函数调用过程就必须的懂一点汇编代码:
1、从main函数的地方开始,要展开main函数的调用就得为main函数创建栈帧,下图是main函数栈帧的创建:
2.接下来是Add函数的调用,参数传递过程
执行call指令的时候按F11就会进入到Add函数:
在按F11就进入到了Add函数的执行代码处:
下面是函数返回及整个函数栈帧的创建调用过程:(这是一张超大图,点开慢慢研究)
注:函数栈帧在不同的编译器上实现的内容存在差异,但是思想都是一样的。
博主能力有限,只能描述到此了,有些地方是用语言没有描述,若有疑问请留言。