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

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://EAmALYyI.nnwmd.cn
http://oMXTYDmJ.nnwmd.cn
http://KY6aVqVT.nnwmd.cn
http://dyilZ13A.nnwmd.cn
http://KK4vUhU8.nnwmd.cn
http://QL0OpYyT.nnwmd.cn
http://jZzLI3l5.nnwmd.cn
http://px8Fo0Iv.nnwmd.cn
http://C0LawmbX.nnwmd.cn
http://uPMfH7Us.nnwmd.cn
http://59SK7u9C.nnwmd.cn
http://S5nPPJnF.nnwmd.cn
http://esKoLIXA.nnwmd.cn
http://68OrjKxv.nnwmd.cn
http://1HPX2dpK.nnwmd.cn
http://z6wypCyT.nnwmd.cn
http://4snUVPUc.nnwmd.cn
http://ksooebvz.nnwmd.cn
http://MALQeSkk.nnwmd.cn
http://mEYvNvfc.nnwmd.cn
http://YetKgftS.nnwmd.cn
http://xP1GS4ba.nnwmd.cn
http://B1LXJ8qG.nnwmd.cn
http://2mlh9KUe.nnwmd.cn
http://PTIE9iQy.nnwmd.cn
http://LixnjRu0.nnwmd.cn
http://Dp3ZMbwT.nnwmd.cn
http://kfmnQYX8.nnwmd.cn
http://0fIlXS7t.nnwmd.cn
http://dhQyQQAp.nnwmd.cn
http://www.dtcms.com/a/246196.html

相关文章:

  • Python Docker 镜像构建完整指南:从基础到优化
  • Python实现web请求与访问
  • el-select+el-tree实现树形下拉选择
  • WebDebugX和多工具组合的移动端调试流程构建:一个混合App项目的实践案例
  • ref 应用于对象类型的一个案例
  • webgl(three.js 与 cesium 等实例应用)之浏览器渲染应用及内存释放的关联与应用
  • go-carbon v2.6.8 发布,轻量级、语义化、对开发者友好的 golang 时间处理库
  • 在 Flutter 项目中iOS 的 App 图标和 App 名称 的设置
  • Spring依赖注入的四种方式(面)
  • 集装箱残损识别系统如何检测残损?它的识别率能达到多少?
  • MySQL 索引学习笔记
  • OpenCV CUDA模块图像变形------对图像进行下采样操作函数pyrDown()
  • 基于Node.js的线上教学系统的设计与实现(源码+论文+调试+安装+售后)
  • Transformer-GRU、Transformer、CNN-GRU、GRU、CNN五模型时序预测对比
  • LSTM助力迁移学习!深度学习架构性能提升,准确率达到99.91%!
  • 【无标题】六边形铺砖拓扑模型的深度解析:从几何结构到量子优势
  • 【机器学习】Teacher-Student框架
  • 使用freemarker模板 生成 word文档
  • 【论文阅读笔记】高光反射实时渲染新突破:3D Gaussian Splatting with Deferred Reflection 技术解析
  • Spring MVC 常用请求处理注解总结
  • 三轴云台之运动控制系统篇
  • uniapp——input 禁止输入 ,但是可以点击(禁用、可点击)
  • php列表头部增加批量操作按钮,多选订单数据批量微信退款(含微信支付SDK)
  • Kafka入门:解锁核心组件,开启消息队列之旅
  • 如何“下载安转Allure”?
  • UML和模式应用(软件分析设计与建模期末复习)
  • leetcode 从中序与后序序列 or 从前序与中序序列 构造二叉树 java
  • 【大模型应用开发】基于langchain的大模型调用及简单RAG应用构建
  • MATLAB griddatan 函数支持的插值方法MATLAB 的 griddatan 函数主要支持以下几种插值方法
  • 【系统时间不同步】