深度讲解函数调用过程
函数调用时,操作系统会为函数开辟栈空间,用于本次函数的调用中临时变量的保存、现场保护。
维护这块栈帧需要两个寄存器ebp、esp.函数调用时这两个寄存器存放了维护这块栈帧的栈底和栈顶指针。
操作系统首先调用mainCRTStartup()
函数,mainCRTStartup()
函数调用,再由__tmainCRTStartup()
函数,__tmainCRTStartup()
函数调用main()函数.
示例代码:
#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 ret = add(a, b);
printf("ret = %d\n", ret);
return 0;
}
main()函数运行前调用堆栈:
当程调试时,点击【调用堆栈】,如下图:
从main()函数开始的调用堆栈:
函数返回时调用堆栈:
总结:
- main()函数之前还有mainCRTStartup()、__tmainCRTStartup()函数的调用。
- 函数形参的实例化是先入栈右参数,再依次从右往左入栈。
- call指令执行时,先压栈call指令的下一条指令的地址,然后跳转至调用函数处。
- ret指令将栈顶的内容当作地址,并且程序将跳转至此处。