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

c/c++ 汇编码中的.cfi 指令有什么用途?


author: hjjdebug
date: 2025年 06月 12日 星期四 14:24:40 CST
descrip: c/c++ 汇编码中的.cfi 指令有什么用途?


文章目录

  • 1. 几个简写词.
  • 2. 看一个简单的测试代码:
  • 3. 生成汇编代码:
  • 4. 分析.cfi 指令
  • 5. 小结:

1. 几个简写词.

cfi(call frame info) 调用帧信息, 名词. 描述的是调用栈的信息
cfa(call frame address) 调用帧地址, 就是当你执行 call xxx 时,堆栈esp的地址, 这个地址很重要.
它是一个固定的地址,该地址处,执行call时, 首先会存如一个8字节的返回地址.(x86_64)

cfi 作用: 异常时stack 回滚.
当程序崩溃或 C++ 异常抛出时,函数后面的代码就不执行了, 那堆栈如何恢复?
如何找到异常接受代码, 并恢复上一级甚至上上一级堆栈
CFI 信息就保留着各栈帧之间的信息.
eh_frame 节中的 CFI 指令, 记录了各调用栈信息,处理异常代码可据此正确回滚调用栈.

2. 看一个简单的测试代码:

这里不讲回滚细节, 这里之间.cfi 指令是什么, 记录了什么信息.

$ cat main.cpp
#include <stdio.h>
int Add(int i, int j)
{return i+j;
}int main()
{int i=Add(2,3);printf("i:%d\n",i);return 0;
}

3. 生成汇编代码:

g++ -S -o 1.S main.cpp

分析一下Add 函数的汇编代码.

Add(int, int):
.LFB0:.cfi_startprocendbr64pushq	%rbp.cfi_def_cfa_offset 16.cfi_offset 6, -16movq	%rsp, %rbp.cfi_def_cfa_register 6movl	%edi, -4(%rbp)movl	%esi, -8(%rbp)movl	-4(%rbp), %edxmovl	-8(%rbp), %eaxaddl	%edx, %eaxpopq	%rbp.cfi_def_cfa 7, 8ret.cfi_endproc

4. 分析.cfi 指令

前面已经说过了,cfi 开头的指令都是为了堆栈回滚而存在的,为了形成eh_frame节信息.
.cfi_startproc ;代表栈帧的开始
.cfi_endproc ;代表栈帧的结束

cfi_def_cfa_offset 16
.cfi_def_cfa_offset 16
def 是 define 的简写

定义调用帧地址偏移 为 16
就是说当你调用了pushq %rbp 后, 这个地方的堆栈地址比调用时的堆栈地址偏移了16bytes
这样定义显然是正确的, 因为此时堆栈中保存了8bytes 返回地址, 8bytes ebp值.
cfi 指令就是要跟踪堆栈的变化.

.cfi_offset 6, -16
6号寄存器 值为CFA-16

.cfi_def_cfa_register 6
定义cfa寄存器是6号寄存器
在movq %rsp,%rbp 后调用定义该cfi指令, 说明6号寄存器是%rbp

.cfi_def_cfa 7, 8
7号寄存器 值为CFA+8
在popq %rbp 后调用该指令, 此时堆栈中还有8字节的返回地址,可知.
此处的7号寄存器对应esp
定义cfa 的几号寄存器可能是保留cfi信息的一个必要手段,这里没有具体分析其与eh_frame的对应关系.

5. 小结:

总之, cfi 指令是为了形成eh_frame节而定义的指令,用来跟踪堆栈地址.以备随时可以返回调用帧.
由于函数执行过程中也可能使用堆栈,所以无非就是跟踪定义栈的偏移, 栈的偏移.
具体怎样触发恢复正确的栈帧,这个细节我们没有分析,这应该分析eh_frame 中保留的数据.
这里只需要知道汇编中.cfi 的大体作用就可以了.
由此我们也可以断定,删除所有.cfi指令,不会影响代码的执行. 如果代码没有异常的话.
只有代码出现异常,才会使用到.cfi信息. 这样就解除了对汇编码中.cfi 指令的疑惑.

从另一个角度看,.cfi指令是伪指令,它并不对应cpu的指令代码,但给编译器提供了一些元信息,
就是辅助信息,能够让编译器了解每个frame对应的堆栈信息. 以备不时之需.

http://www.xdnf.cn/news/13762.html

相关文章:

  • (LeetCode 每日一题) 3423. 循环数组中相邻元素的最大差值 (数组)
  • Java面试避坑指南:牛客网最新高频考点+答案详解
  • Mac电脑-Office 2024 长期支持版 PPT、Excel、Word(Mac中文)
  • RabbitMQ实现异步消息监听机制
  • Makefile 学习笔记
  • 无外接物理显示器的Ubuntu系统的远程桌面连接(升级版)
  • C#学习第30天: 匹配模式
  • 大模型技术30讲-5-利用数据来减少过拟合现象
  • Next.js + Supabase = 快速开发 = 高速公路
  • 怎样下载某个SCI期刊的endnote style?答:直接去endnote官网搜索期刊名称并下载即可
  • JMeter + 命令行服务器端压测全流程详解
  • 风控系统中,要调用第三方服务获取信息,很慢,如何解决?
  • vue3项目移动端实现进度条可手动滑动控制进度和点击控制进度
  • Docker入门篇--从安装到使用
  • 【Linux手册】从「程序」到「进程」:计算机世界的运行机制
  • 智慧养老与数字健康:科技赋能老年生活,构建全方位养老体系
  • Redis核心数据结构详解与应用
  • arduino通过控制器,精准控制24V电动轮毂转动
  • 解锁Scrapy爬虫:从入门到实战的Python秘籍
  • 图像分割技术:像素级的精准识别(superior哥深度学习系列第12期)
  • 关于MySql深分页的问题及优化方案
  • 软件测试的艺术与科学:构建商业级产品的优雅草卓伊凡
  • 微信小程序渗透测试指北(附案例)
  • ATM 模拟器 Golang 程序--示例
  • 【二分答案1-----切木棒】
  • 基于YOLOv11与单目测距的实战教程:从目标检测到距离估算
  • 嵌入式通信模块实战新范式:基于虚拟仿真平台的NB-IoT核心技能训练——零硬件损耗的全栈式实验方案,重构物联网通信教学逻辑
  • 基于多面体模型的编译优化技术
  • Ubuntu 绑定Conda
  • 在 Vue 3 中修改 el-select 组件接收的 prop 值