2022年

2022年发布的文章
  • GDB准备调试阶段

    在真正使用 GDB 调试一个程序前,我们需要做一些准备工作。主要包括调获取试信息,以及启动可执行程序。

    编译产生调试信息

    GDB 调试程序需要在源文件编译阶段产生调试信息。调试信息存储在目标文件中,主要包括行号、变量的类型和作用域、函数名字、函数参数和函数的作用域等源文件的特性。

    编译源文件的时候,使用-g选项可以产生调试信息,较早以前的 C 语言编译器也允许使用-gg选项来产生调试信息,但是现在版本的 GDB 不再支持这种格式产生的调试信息,所以不建议使用-gg选项。

    在 Linux 中 GCC 编译器使用-g选项编译文件时,编译器还会做出以下额外的操作:

    1. 创建符号表,符号表包含了程序中使用的变量名称的列表。
    2. 关闭所有的优化机制,以便执行程序过程中样按照原来的C代码进行。

    编译器会把优化机制自动的关闭,是不是意味着编译程序不能使用-O选项?其实这个与使用的编译器有关。有的编译器不支持-g-O参数选项同时处理,所以得不到带有调试信息并且优化过的可执行程序。GCC 编译器是支持这两个参数同时使用,所以 GDB 可以调试经过编译优化后并且带有调试信息的程序。

    注意:如果使用 GCC 编译时使用了优化选项-O,那么优化器会重新编排已经编写好的代码。调试器显示的是真正编译生成的代码,在执行路径和编写的源代码不一致是常见的。例如,源文件中定义了一个变量,但从来这个变量没有使用过。优化编译后使用 GDB 调试会发现没有这个变量,这就是优化器把变量给去除了。

    GDB 启动程序

    产生调试信息以后,就可以使用 GDB 来启动程序。使用 GDB 启动程序一般有以下三种方式:

    1) 直接启动可执行程序

    直接启动可执行程序的方法为:

    gdb fileName

    fileName 为可执行程序的名字。下面是一个简单的例子:

    gdb test

    test 就是我么生成的可执行文件的名字,这是最常见最简单的启动方式。启动时指定对象,进入 GDB 中可以直接对文件进行调试。

    2) 启动 core 文件

    启动可执行文件产生的 core 文件的方法为:

    gdb fileName file_core

    fileName 是可执行文件的名字,file_core 是 fileName 运行产生的 core 文件。下面是一个简单的例子:

    gdb test test_core

    test_core 为 test 运行时产生的 core 文件。当可执行程序运行时产生段错误,系统就会产生 core 文件(开启系统的 core dump 功能),gdb 对产生的 core 文件进行分析,可以还原系统发生段错误时刻的堆栈情况,这对于我们发现程序bug很有帮助。

    3) 启动进程

    启动进程的方式:

    gdb fileName PID

    fileName 是可执行文件的名字,PID 进程号。下面是一个简单的例子:

    gdb test 1234

    GDB 将调试程序 test 附加到进程 1234 上进行调试。

更多...

加载中...