首页 > 文章列表 > C++ 函数调用约定与栈帧的管理方式

C++ 函数调用约定与栈帧的管理方式

函数调用约定 栈帧管理
376 2025-03-22

C++ 函数调用约定包括 cdecl 和 stdcall,分别用于参数压栈和返回值处理。此外,栈帧管理涉及帧指针 EBP 和栈指针 ESP,局部变量存储在 EBP 和 ESP 之间的区域,参数基于 ESP 的偏移量存放,返回地址压栈到 EBP 之下。

C++ 函数调用约定与栈帧的管理方式

C++ 函数调用约定与栈帧的管理方式

函数调用约定

函数调用约定是函数调用时,调用方和被调用方在栈帧布局、参数传递和返回值处理方面的约定。C++ 中主要有以下几种函数调用约定:

  • cdecl 调用约定:参数从右向左压栈,函数返回值压栈。
  • stdcall 调用约定:参数从右向左压栈,函数返回值由寄存器 eax 传递。

栈帧的管理

当函数被调用时,系统会在栈上分配一块内存区域,称为栈帧。栈帧用于存储函数执行期间的数据,包括参数、局部变量和临时寄存器。

栈帧的管理方式如下:

  1. 帧指针 (EBP):EBP 寄存器指向当前栈帧的基地址。
  2. 栈指针 (ESP):ESP 寄存器指向当前栈顶。
  3. 局部变量:局部变量分配在 ESP 和 EBP 之间的区域。
  4. 参数:函数参数从 ESP 减去偏移量的位置开始存放。
  5. 返回地址:当函数被调用时,返回地址被压栈到 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)                |
| 栈底                       |