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);  // 调用成员函数addreturn 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,被调用函数负责平栈。
  • 栈与平栈:栈用于存储函数调用的局部变量和参数,平栈是清理栈的过程,不同的调用约定决定了由谁负责平栈。

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


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

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/900519.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

xv6-labs-2024 lab1

lab-1 注:实验环境在我的汇编随手记的末尾部分有搭建教程。 0.前置 第零章 xv6为我们提供了多种系统调用,其中,exec将从某个文件里读取内存镜像(这确实是一个好的说法),并且将其替换到调用它的内存空间,也就是这个…

属性修改器 (AttributeModifier)

主页面设置组件 import { MyButtonModifier } from ../datastore/MyButtonModifier;Entry ComponentV2 struct MainPage {// 支持用状态装饰器修饰,行为和普通的对象一致Local modifier: MyButtonModifier new MyButtonModifier();build() {Column() {Button(&quo…

【 <二> 丹方改良:Spring 时代的 JavaWeb】之 Spring Boot 中的监控:使用 Actuator 实现健康检查

<前文回顾> 点击此处查看 合集 https://blog.csdn.net/foyodesigner/category_12907601.html?fromshareblogcolumn&sharetypeblogcolumn&sharerId12907601&sharereferPC&sharesourceFoyoDesigner&sharefromfrom_link <今日更新> 一、引子&…

类和对象(下篇)(详解)

【本节目标】 1. 再谈构造函数 2. Static成员 3. 友元 4. 内部类 5. 再次理解封装 1. 再谈构造函数 1.1 构造函数体赋值 在创建对象时&#xff0c;编译器通过调用构造函数&#xff0c;给对象中各个成员变量一个合适的初始值。 #include <iostream> using name…

高精度算法

高精度加法 输入两个数&#xff0c;输出他们的和&#xff08;高精度&#xff09; 输入样例 111111111111111111111111111111 222222222222222222222222222222 输出样例 333333333333333333333333333333 #include <bits/stdc.h> using namespace std;string a,b; in…

Linux开发中注意哪些操作系统安全

在 Linux 开发中&#xff0c;确保操作系统的安全至关重要。以下是一些需要注意的方面&#xff1a; 用户管理与权限控制 合理设置用户权限&#xff1a;为不同的用户和用户组分配适当的权限&#xff0c;遵循最小权限原则。避免给普通用户过多的权限&#xff0c;以免他们误操作或…

x64dbg调试python解释器

可以先写个input()这会让dbg中断在ntdll模块中&#xff0c;查看调用堆栈在系统调用结束后的打断点 然后直接断到PyObject_Vectorcall函数

✅ Ultralytics YOLO验证(Val)时自动输出COCO指标(AP):2025最新配置与代码详解 (小白友好 + B站视频)

✅ YOLO获取COCO指标(3)&#xff1a;验证(Val) 启用 COCO API 评估&#xff08;自动输出AP指标&#xff09;| 发论文必看&#xff01; | Ultralytics | 小白友好 文章目录 一、问题定位二、原理分析三、解决方案与实践案例步骤 1: 触发 COCO JSON 保存步骤 2: 确保 self.is_coc…

【嵌入式学习3】基于python的tcp客户端、服务器

目录 1、tcp客户端 2、tcp服务器 3、服务器多次连接客户端、多次接收信息 1、tcp客户端 """ tcp:客户端 1. 导入socket模块 2. 创建socket套接字 3. 建立tcp连接(和服务端建立连接) 4. 开始发送数据(到服务端) 5. 关闭套接字 """ import soc…

Linux: network: 两台直连的主机业务不通

前提环境,有一个产品的设定是两个主机之间必须是拿网线直连。但是设备管理者可能误将设置配错,不是直连。 最近遇到一个问题,说一个主机发的包,没有到对端,一开始怀疑设定的bond设备的问题,检查了bond的设置状态,发现没有问题,就感觉非常的奇怪。后来就开始怀疑两个主机…

COMSOL固体力学接触

目录 一、接触非线性及接触面积计算 一、接触非线性及接触面积计算 COMSOL接触非线性及接触面积计算_哔哩哔哩_bilibili 形成联合体&#xff0c;在定义处右键选择“建立接触对” 位移dz使用参数化扫描。 接触选择定义中设置的接触对&#xff0c;选择罚函数&#xff0c;摩擦设置…

22.OpenCV轮廓匹配原理介绍与使用

OpenCV轮廓匹配原理介绍与使用 1. 轮廓匹配的基本概念 轮廓匹配&#xff08;Contour Matching&#xff09;是计算机视觉中的一种重要方法&#xff0c;主要用于比较两个轮廓的相似性。它广泛应用于目标识别、形状分析、手势识别等领域。 在 OpenCV 中&#xff0c;轮廓匹配主要…

oracle 快速创建表结构

在 Oracle 中快速创建表结构&#xff08;仅复制表结构&#xff0c;不复制数据&#xff09;可以通过以下方法实现&#xff0c;适用于需要快速复制表定义或生成空表的场景 1. 使用 CREATE TABLE AS SELECT (CTAS) 方法 -- 复制源表的全部列和数据类型&#xff0c;但不复制数据 C…

若依原理笔记

代码生成器 源码分析 查询数据库列表 导入表结构 生成代码 修改generator.yml配置文件 代码生成器增强 Velocity模版引擎 基础语法-变量 Lombok集成 E:\javaProject\dkd-parent\dkd-generator\src\main\resources\vm\java\domain.java.vm package ${packageName}.domain;#fo…

Ansible的使用

##### Ansible使用环境 - 控制节点 - 安装Ansible软件 - Python环境支持&#xff1a;Python>2.6 - 必要的模块&#xff1a;如PyYAML等 - 被控节点 - 启用SSH服务 - 允许控制节点登录&#xff0c;通常设置免密登录 - Python环境支持 http://www.ansible.com/ …

C++ 提高编程:模板与 STL 深度剖析

摘要&#xff1a;本文深入探讨 C 提高编程中的模板编程与标准模板库&#xff08;STL&#xff09;相关内容。详细阐述模板编程中函数模板和类模板的概念、语法、特性及应用案例&#xff1b;对 STL 的诞生背景、基本概念、六大组件进行剖析&#xff0c;并对常用容器、函数对象、常…

C++(类模板的运用)

使用vector实现一个简单的本地注册登录系统 注册&#xff1a;将账号密码存入vector里面&#xff0c;注意防重复判断 登录&#xff1a;判断登录的账号密码是否正确 #include <iostream> #include <vector> #include <fstream> #include <sstream> usi…

【大模型】DeepSeek+蓝耕MaaS平台+海螺AI生成高质量视频实战详解

目录 一、前言 二、蓝耘智能云MaaS平台介绍 2.1 蓝耘智算平台是什么 2.2 平台优势 2.3 平台核心能力 三、海螺AI视频介绍 3.1 海螺AI视频是什么 3.2 海螺AI视频主要功能 3.3 海螺AI视频应用场景 3.4 海螺AI视频核心优势 3.5 项目git地址 四、蓝耘MaaS平台DeepSeek海…

接口自动化学习二:session自动管理cookie

session自动管理cookie&#xff1a; cookie中的数据&#xff0c;都是session提供的 实现步骤&#xff1a; 1.创建session对象&#xff1b;my_sessionrequests.Session() 2.使用session实例&#xff0c;调用get方法&#xff0c;发送获取验证码请求&#xff08;不需要提取cookie&…

C++类型转换详解

目录 一、内置 转 内置 二、内置 转 自定义 三、自定义 转 内置 四、自定义 转 自定义 五、类型转换规范化 1.static_case 2.reinterpret_cast 3.const_cast 4.dynamic_cast 六、RTTI 一、内置 转 内置 C兼容C语言&#xff0c;在内置类型之间转换规则和C语言一样的&am…