一、函数堆栈调用
以以下简单的代码为例,分析函数调用的过程
#include<stdio.h>
int sum(int a,int b)
{
int tmp = a+b;
return tmp;
}
int main()
{
int a = 10;
int b = 20;
int rt = 0;
rt = sum(a,b);
printf("%d\n",rt);
return 0;
}
通过观察汇编语言,
第一步 就是首先开辟main函数的栈帧
第二步 就是将调用main函数-----MainCRTStartup的函数的栈底指针寄存器压入main栈帧中
第三步 让ebp 和 esp 指向相同的位置
第四步 esp向上开辟4C,并且用ccccccccc初始化
第五步 压入a,b,rt,用栈底指针偏移来做
第六步 rt = sum(a,b),即函数调用 (1) 将实参的值压入栈,即给形参开辟内存,从右向左
(2)call 指令 第一步 压入下一行指令的地址
第二步
接下来即跳转到sum函数的栈帧,与之前类似
1、将栈底指针寄存器函数压入栈(main)
2、ebp 指向与esp相同的位置
3、esp向上开辟4C,并且用ccccccccc初始化
4、tmp = a+b;将累加寄存器的值30赋给ebp-4(tmp)
5、return 将值赋给寄存器返回
清栈
esp 和 ebp在同一个位置
然后 pop ebp将ebp返回了main函数栈帧的栈底
然后清理形参内存
以上过程中的问题
1、rt = sum(a,b) ,将寄存器的值赋给rt
二、返回值规则
1、内置类型
如果小于4个字节, 一个寄存器带回
大于4字节小于等于8字节 两个寄存器带回
返回值字节>8 临时量,即就是返回值在内存上,寄存器只带出来地址
2、
类类型的返回值 ,不由寄存器带回,如果用寄存器带回生成返回对象就要调用构造函数,而寄存器没有地址,因此生成不了临时对象
所以由内存带回,一定会生成临时对象