为二进制文件添加.gnu_debugdata调试信息

前言

在使用gcc/g++编译二进制文件过程中,如果添加了-g参数,编译出来的二进制文件会带有debug信息,供调试使用。但是debug信息往往占用空间很大,导致二进制文件太大,在发布到生产环境时,一般会去掉调试信息,以减小二进制文件大小。如此一来,在出现问题后就无法直接使用gdb调试工具进行调试了。一种可行的做法是版本发布时保留debug信息,出现问题后使用debug信息进行问题分析调试,但这种做法相对比较麻烦。mini debuginfo的技术提供了一种有效的解决思路。

mini debuginfo的思路是在二进制ELF文件中添加一个section(.gnu_debugdata),section中仅保留最少的调试信息(如:只保留可调试的函数符号),并对调试进行进行压缩,以尽可能地减少调试信息大小。

下面以一个实际例子说明如何给二进制文件添加该类型的section。

实例

我们以一个简单的例子开始,以下是该例子的代码:

#include <stdio.h>
int a;
void func1()
{
        a = 1;
}

void func2()
{
        int test;
        a = 2;
        test = a;
}

int main()
{
        func1();
        func2();
        printf("a is %d\n", a);
        return 0;
}

编译二进制文件

使用gcc命令将代码编译成二进制可执行文件:

gcc -g -o test_mini_debug test_mini_debug.c

保存要保留的符号信息

然后使用nm命令查看该二进制文件中的符号信息,其中--defined-only参数指定只关注定义的信息:

nm --defined-only test_mini_debug

二进制文件中有很多符号,我们调试时关心的主要是函数和变量信息,函数位于ELF二进制文件的.TEXT段中,即上图中第二列为T/t的符号;变量分全局变量和局部变量,全局变量位于.BSS段(未初始化的全局变量)或.DATA段(初始化的全局变量)中,即上图中第二列为B/b,D/d的符号,局部变量一般保存在寄存器或栈中,此处看不到。

我们将函数和全局变量保存到一个文件中,后续会使用到,命令如下:

nm --defined-only test_mini_debug |awk '{ if($2 == "T" || $2 == "t" || $2 == "B" || $2 == "b" || $2 == "D" || $2 == "d") print $3 }' > test_mini_debug.syms

生成mini debuginfo信息

在上一步中保存了我们要保留的符号信息,接下来就可以根据这些信息生成minidebug信息了。首先使用objcopy命令将上面的二进制文件的debug信息剥离出来:

objcopy --only-keep-debug test_mini_debug test_mini_debug.debug

然后,对debug信息进行瘦身,只保留我们关注的函数和全局变量信息:

objcopy -S --keep-symbols=test_mini_debug.syms test_mini_debug.debug test_mini_debug.mini_debuginfo

接着,使用xz命令对debug信息进行压缩,进一步减小debug信息大小:

xz test_mini_debug.mini_debuginfo

经过以上一番操作,debug信息从原来的6K减少到了1K左右:

对二进制文件进行瘦身

在上面的步骤中,我们将debug信息从二进制文件中剥离了出来,但是原来的二进制文件还存在debug信息,如下图,使用readelf查看,这些debug信息(以.debug开头的段)还在二进制文件中:

readelf -S test_mini_debug

要先将该部分信息删除,直接使用objcopy -S命令即可。

objcopy -S test_mini_debug

再次使用readelf命令查看,已经没有debug信息了:

将mini debuginfo添加到二进制文件中

最后一步,将我们之前生成的mini debuginfo信息添加到二进制文件中,命令如下:

objcopy --add-section .gnu_debugdata=test_mini_debug.mini_debuginfo.xz test_mini_debug

使用readelf命令查看,二进制文件中已经有.gnu_debugdata段了:

验证

使用gdb命令调试该二进制文件,可以看到,已能够识别函数和全局变量了。

相关推荐

  1. mysql二进制文件恢复sql

    2024-07-13 13:28:01       42 阅读
  2. Vue将File二进制文件转换base64格式

    2024-07-13 13:28:01       55 阅读

最近更新

  1. docker php8.1+nginx base 镜像 dockerfile 配置

    2024-07-13 13:28:01       66 阅读
  2. Could not load dynamic library ‘cudart64_100.dll‘

    2024-07-13 13:28:01       70 阅读
  3. 在Django里面运行非项目文件

    2024-07-13 13:28:01       57 阅读
  4. Python语言-面向对象

    2024-07-13 13:28:01       68 阅读

热门阅读

  1. 【Android】在渲染生效前提前测量View大小

    2024-07-13 13:28:01       22 阅读
  2. 基于节点嵌入的链接预测(暂时这样吧)

    2024-07-13 13:28:01       19 阅读
  3. C#中where的约束

    2024-07-13 13:28:01       22 阅读
  4. ABP框架中的ISoftDelete与软删除

    2024-07-13 13:28:01       24 阅读
  5. 三级_网络技术_13_局域网技术基础及应用

    2024-07-13 13:28:01       22 阅读
  6. 服务器数据出现丢失该怎样恢复?

    2024-07-13 13:28:01       17 阅读
  7. React中使用usePrevious的意义是什么,为啥要用它

    2024-07-13 13:28:01       19 阅读
  8. Spring:SpringBoot为什么可以使用Jar包启动

    2024-07-13 13:28:01       18 阅读
  9. ubuntu安装k8s+docker运行英伟达gpu cuda

    2024-07-13 13:28:01       22 阅读
  10. 使用Python绘制百分比堆积柱形图

    2024-07-13 13:28:01       21 阅读
  11. Memcached负载均衡:揭秘高效缓存分发策略

    2024-07-13 13:28:01       21 阅读
  12. Mybatis-SQL注入讲解及#预处理与$的区别

    2024-07-13 13:28:01       19 阅读