c++++ 函数调用约定包括 cdecl 和 stdcall,分别用于参数压栈和返回值处理。此外,栈帧管理涉及帧指针 ebp 和栈指针 esp,局部变量存储在 ebp 和 esp 之间的区域,参数基于 esp 的偏移量存放,返回地址压栈到 ebp 之下。
C++ 函数调用约定与栈帧的管理方式
函数调用约定
函数调用约定是函数调用时,调用方和被调用方在栈帧布局、参数传递和返回值处理方面的约定。C++ 中主要有以下几种函数调用约定:
- cdecl 调用约定:参数从右向左压栈,函数返回值压栈。
- stdcall 调用约定:参数从右向左压栈,函数返回值由寄存器 eax 传递。
栈帧的管理
当函数被调用时,系统会在栈上分配一块内存区域,称为栈帧。栈帧用于存储函数执行期间的数据,包括参数、局部变量和临时寄存器。
栈帧的管理方式如下:
立即学习“C++免费学习笔记(深入)”;
- 帧指针 (EBP):EBP 寄存器指向当前栈帧的基地址。
- 栈指针 (ESP):ESP 寄存器指向当前栈顶。
- 局部变量:局部变量分配在 ESP 和 EBP 之间的区域。
- 参数:函数参数从 ESP 减去偏移量的位置开始存放。
- 返回地址:当函数被调用时,返回地址被压栈到 EBP 之下。
实战案例
以下代码展示了 cdecl 调用约定和 stdcall 调用约定下的函数调用和栈帧布局:
cdecl 调用约定:
int sum(int a, int b) {
return a + b;
}
void main() {
int result = sum(10, 20);
}
栈帧布局:
|-----------------------------|
| 栈顶 (ESP) |
| ... |
| 局部变量 b |
| 局部变量 a |
| 返回地址 |
|-----------------------------|
| 帧指针 (EBP) |
| 栈底 |
stdcall 调用约定:
__stdcall int sum(__stdcall int a, __stdcall int b) {
return a + b;
}
void main() {
int result = sum(10, 20);
}
栈帧布局:
|-----------------------------|
| 栈顶 (ESP) |
| ... |
| 参数 b |
| 参数 a |
| 返回地址 |
|-----------------------------|
| 帧指针 (EBP) |
| 栈底 |