当前位置: 首页 > news >正文

C/C++ 调用约定:深入理解栈与平栈

前言

在编程中,理解函数调用约定和栈的机制对于编写高效代码、调试程序以及进行逆向工程至关重要。本文将深入探讨 C 和 C++ 的调用约定,以及栈与平栈的相关知识。


在这里插入图片描述

C 调用约定

在 C 语言中,默认的调用约定是 cdeclcdecl 调用约定的特点如下:

  • 参数传递:参数从右向左依次压入栈中。
  • 栈清理:调用者负责清理栈(即调用者在函数返回后负责平栈)。
  • 返回值:返回值通常存放在 EAX 寄存器中。

示例

int add(int a, int b) {
    return a + b;
}

int main() {
    int result = add(3, 4);  // 调用add函数
    return 0;
}

在汇编层面,调用 add(3, 4) 的代码可能如下:

push 4        ; 第二个参数压栈
push 3        ; 第一个参数压栈
call add      ; 调用add函数
add esp, 8    ; 调用者平栈,清理8字节的栈空间

C++ 调用约定

C++ 调用约定与 C 调用约定有所不同,主要体现在以下几点:

名称修饰(Name Mangling)

  • C++ 编译器会对函数名进行修饰(Name Mangling),以支持函数重载、命名空间等特性。例如,函数 int add(int a, int b) 可能会被修饰为 _Z3addii
  • C 语言没有名称修饰,函数名在编译后保持不变。

thiscall 调用约定

在 C++ 中,非静态成员函数的调用约定通常是 thiscallthiscall 调用约定的特点:

  • this 指针this 指针通常通过 ECX 寄存器传递。
  • 参数传递:其他参数从右向左压入栈中。
  • 栈清理:被调用函数负责清理栈。

示例

class MyClass {
public:
    int add(int a, int b) {
        return a + b;
    }
};

int main() {
    MyClass obj;
    int result = obj.add(3, 4);  // 调用成员函数add
    return 0;
}

在汇编层面,调用 obj.add(3, 4) 的代码可能如下:

lea ecx, [obj]  ; 将this指针(即obj的地址)放入ECX寄存器
push 4          ; 第二个参数压栈
push 3          ; 第一个参数压栈
call ?add@MyClass@@QAEHHH@Z  ; 调用成员函数add

栈与平栈

在这里插入图片描述

栈的基本概念

  • 栈(Stack):栈是一种后进先出(LIFO)的数据结构,用于存储函数调用时的局部变量、参数、返回地址等信息。
  • 栈帧(Stack Frame):每个函数调用都会在栈上创建一个栈帧,用于存储该函数的局部变量、参数等信息。
  • 栈指针(ESP)ESP 寄存器指向当前栈顶的位置。

平栈(Stack Cleanup)

平栈是指在函数调用结束后,清理栈上的参数,使栈恢复到函数调用前的状态。不同的调用约定决定了由谁负责平栈:

  1. cdecl 调用约定

    • 调用者负责平栈:调用者在函数返回后使用 add esp, n 指令清理栈。
    • 示例
      push 4
      push 3
      call add
      add esp, 8  ; 调用者平栈,清理8字节的栈空间
      
  2. stdcall 调用约定

    • 被调用函数负责平栈:被调用函数在返回前使用 ret n 指令自动清理栈。
    • 示例
      push 4
      push 3
      call add
      ; 被调用函数内部:
      ret 8  ; 被调用函数平栈,清理8字节的栈空间
      
  3. fastcall 调用约定

    • 被调用函数负责平栈:被调用函数在返回前使用 ret n 指令自动清理栈。
    • 示例
      mov ecx, 3  ; 第一个参数通过ECX寄存器传递
      mov edx, 4  ; 第二个参数通过EDX寄存器传递
      call add
      ; 被调用函数内部:
      ret 0  ; 没有参数通过栈传递,无需清理栈
      
  4. thiscall 调用约定

    • 被调用函数负责平栈:被调用函数在返回前使用 ret n 指令自动清理栈。
    • 示例
      lea ecx, [obj]  ; this指针通过ECX寄存器传递
      push 4          ; 第二个参数压栈
      push 3          ; 第一个参数压栈
      call ?add@MyClass@@QAEHHH@Z
      ; 被调用函数内部:
      ret 8  ; 被调用函数平栈,清理8字节的栈空间
      

总结

  • C 调用约定:默认使用 cdecl,调用者负责平栈。
  • C++ 调用约定:默认使用 thiscall,被调用函数负责平栈。
  • 栈与平栈:栈用于存储函数调用的局部变量和参数,平栈是清理栈的过程,不同的调用约定决定了由谁负责平栈。

理解这些调用约定和栈的机制对于编写高效的代码、调试程序以及进行逆向工程都非常重要。希望本文能帮助你更好地掌握这些知识,提升编程技能!


如果你觉得这篇文章对你有帮助,请点赞、收藏并分享给你的朋友们!

相关文章:

  • OpenVLA-OFT——微调VLA时加快推理的三大关键设计:支持动作分块的并行解码、连续动作表示以及L1回归(含输入灵活化及对指令遵循的加强)
  • 第16届蓝桥杯单片机模拟试题Ⅱ
  • Java 大视界 -- Java 大数据机器学习模型在智能客服多轮对话系统中的优化策略(179)
  • 计算机系统---性能指标(3)续航与散热
  • 【C++篇】深入剖析C++ Vector底层源码及实现机制
  • 数据分享:汽车测评数据
  • ARM 汇编启动代码详解:从中断向量表到中断处理
  • 浪漫永恒怀旧婚礼结婚照户外摄影照片调色Lightroom预设 J SERIES PRESETS
  • Java反射实战-特殊嵌套格式JSON自定义解析装配
  • Exce格式化批处理工具详解:高效处理,让数据更干净!
  • C语言-查表法详解与实践
  • 网络游戏服务器如何构建全方位防御体系?DDoS与CC攻击实战防护指南
  • 鸿蒙开发_ARKTS快速入门_语法说明_自定义组件开发---纯血鸿蒙HarmonyOS5.0工作笔记011
  • 如何将/dev/ubuntu-vg/lv-data的空间扩展到/dev/ubuntu-vg/ubuntu-lv的空间上
  • 杂谈:模型训练参数是否存在临界点?
  • DNS服务(Linux)
  • 软考笔记9——数据库技术基础
  • python应用之使用pdfplumber 解析pdf文件内容
  • 第二篇:系统分析师——7-11章
  • 网站缓存怎么检查是否生效?
  • java 做网站/独立站怎么建站
  • 定西企业网站制作/泰州网站排名seo
  • vps主机可以做几个网站/陕西seo顾问服务
  • 帮别人做钓鱼网站/网店网络推广方案
  • a站下载安装/seo没什么作用了
  • 做网站建设的公司有哪些/外贸网站优化推广