【源码】RapidJSON 源码剖析(0.1):调试工具 GDB 的使用

正式开始源码阅读之前,有必要了解一下源码阅读中用到的调试工具 GDB

GDB(GNU Debugger) 是一种可以运行在多种类 Unix 系统上的,可移植的,适用于多种编程语言的调试器。
(The GNU Debugger (GDB) is a portable debugger that runs on many Unix-like systems and works for many programming languages.)

GDB: The GNU Project Debugger 中对 GDB 有如下描述:

GDB 可以提供以下四种操作来帮助你捕获 bug:
• 启动你的程序,指定可能影响程序运行行为的内容。
• 让你的程序在指定的条件下停止。
• 当你的程序停止时, 检测你的程序发生的事。
• 改变你的程序的内容, 你可以纠正一个 bug 的影响,以便继续研究下一个 bug
(gdb can do four main kinds of things (plus other things in support of these) to help you catch bugs in the act:
• Start your program, specifying anything that might affect its behavior.
• Make your program stop on specified conditions.
• Examine what has happened, when your program has stopped.
• Change things in your program, so you can experiment with correcting the effects of one bug and go on to learn about another.)

调试工具 GDB 的使用

0. 待调试程序

本文调试的样例程序来自于 RapidJSON 官方文档,具体源代码如下:

// rapidjson/example/simpledom/simpledom.cpp`
#include "rapidjson/document.h"
#include "rapidjson/writer.h"
#include "rapidjson/stringbuffer.h"
#include <iostream>

using namespace rapidjson;

int main() 
{
    // 1. 把 JSON 解析至 DOM。
    const char* json = "{\"project\":\"rapidjson\",\"stars\":10}";
    Document d;
    d.Parse(json);

    // 2. 利用 DOM 作出修改。
    Value& s = d["stars"];
    s.SetInt(s.GetInt() + 1);

    // 3. 把 DOM 转换(stringify)成 JSON。
    StringBuffer buffer;
    Writer<StringBuffer> writer(buffer);
    d.Accept(writer);

    // Output {"project":"rapidjson","stars":11}
    std::cout << buffer.GetString() << std::endl;
    return 0; 
}

1. 编译

要使编译生成的可执行文件带有可被 GDB 识别的调试信息,需要在编译时加上参数 -g。如:

g++ -g -std=c++11 simpledom.cpp -o simpledom.out

2. 用 GDB 启动你的程序

gdb <待调试软件名> 命令启动调试待调试软件。如:

gdb simpledom.out

3. 显示程序源码

当用 GDB 打开指定的程序后,可以用 listl 命令查看该程序的源码。GDB: The GNU Project Debugger 9. Examining Source Files 对该命令有详细地介绍,这里摘录可能会用到的用法:

command description
list linenum Print lines centered around line number linenum in the current source file.
list function Print lines centered around the beginning of function function.
list first,last Print lines from first to last.
list ,last Print lines ending with last.
list first, Print lines starting with first.
list filename:linenum Print lines centered around linenum in the source file filename.
list filename:function Print lines centered around the beginning of the function function in the file filename.

如,显示当前文件的 20 到 23 行:

list 20, 23

输出如下:

4. 设置断点,查看断点和取消断点

设置断点是 breakb 命令,查看断点用 info breakinfo b 命令,删除断点用 deleted 命令。具体用法如下表:

command description
break location Set a breakpoint at the given location, which can specify a function name, a line number, or an address of an instruction.
breakif cond Set a breakpoint with condition cond; evaluate the expression cond each time the breakpoint is reached, and stop only if the value is nonzero—that is, if cond evaluates as true.‘…’ stands for one of the possible arguments described above (or no argument) specifying where to break.
info break Print a table of all breakpoints, watchpoints, and catchpoints set and not deleted. For each breakpoint, following columns are printed: Breakpoint Numbers, Type, Disposition, Enabled(‘y’) or Disabled(‘n’), Address, What.
delete breakpoint number Delete the breakpoint specified by breakpoint number, If no argument is specified, delete all breakpoints.

如,在 14 行和 18 行设置断点, 然后查看断点信息,最后取消 18行断点,再查看断点信息:

break 14
break 18
info break
delete 2
info break

输出如下:

5. 开始单步调试,下一步, 继续到下个断点,进入函数,跳出当前函数,查看变量值,查看变量类型。

start 命令从 main 函数开始单步调试。nextn 命令进入下一步。continuec 命令继续执行程序到下一个断点。
steps 命令进入函数内部, finish 跳出当前函数。
print <变量名>p <变量名> 显示指定变量的值,ptype <变量名> 查看变量的类型。

如,

  • 通过 start 命令开始单步调试代码
  • 进入下一步
  • 查看变量 json 的值和类型
  • 在 14 行处设置一个断点
  • 执行到下一个断点(14 行处)
  • 跳转进 Parse 函数
  • 跳出 Parse 函数
  • 继续执行完整个程序

参考文献