前面几篇文章学习了链接器相关的内容。现在开始来学习GDB调试。我们的目的是通过这几篇文章将GDB调试完全学会。
1 为什么需要GDB
- 什么是GDB?
- GNU项目中的调试器(gnu debuger)
- 能够跟踪程序的执行,也能够恢复程序执行前的状态
- 为什么需要GDB?
- 软件不是一次性开发完的,是软件就一定有bug,所以需要调试工具来定位bug
- 调试是软件开发过程中不可或缺的部分
2 GDB 的常规应用
- 自定义程序启动的方式(指定影响程序运行的参数)
- 设置条件断点(在条件满足时暂停程序的执行,一般为循环中的语句或者递归调用中的语句)
- 回溯检查导致程序异常的原因(一般是通过分析核心转储文件-core文件)
- 动态改变程序执行流(定位问题的辅助方式)
GDB的启动方式有哪些?
- 直接启动
- gdb
- gdb test.out
- gdb test.out core
- 动态连接
- gdb test.out pid
在给出gdb具体的调试代码步骤之前,我们先来看看两个应用示例:
- GDB应用示例一
- 其中设置命令行参数这里,如果了解main函数的参数的话应该知道,如果不了解命令行参数,请参考这篇文章: main函数参数的意义
- 还有一点就是下面两种载入目标程序的命令是一样的效果
- 其他的用法都很好理解,我们不再赘述
- GDB应用示例二
- 上述各个命令的使用方法也很好理解,我们只需要注意下图中的两种启动方式同样是等效的:
3 GDB调试程序实例
下面就开始使用具体的例子来详细解读GDB调试的过程
我们调试的两个代码为:
- test.c
#include <stdio.h>
#include <unistd.h>
extern int* g_pointer; // 外部变量
extern void func(); // 外部函数
void test_1()
{
printf("test_1() : %p\n", test_1);
}
void test_2()
{
printf("test_2() : %p\n", test_2);
}
void test_3()
{
printf("test_3() : %p\n", test_3);
}
int main(int argc, char *argv[])
{
typedef void(TFunc)();
TFunc* fa[] = {test_1, test_2, test_3};
int i = 0;
printf("main() : begin...\n");
for(i=0; i<argc; i++) //argc代表命令行参数的个数
{
printf("argv[%d] = %s\n", i, argv[i]);
}
for(i=0; i<100; i++)
{
fa[i%3]();
sleep(argc > 1); // 如果argc大于1,则执行睡眠函数
}
printf("g_pointer = %p\n", g_pointer);
func();
printf("main() : end...\n");
return 0;
}
- func.c
#include <stdio.h>
int* g_pointer;
void func()
{
*g_pointer = (int)"D.T.Software"; //注意,这里是出错的地方,g_pointer是指向0地址,但是在这里却对0地址赋值
return;
}
- GDB使用初探-----下面我们暂时先不调试,先熟悉熟悉几个命令是如何使用的。
- 首先对上述程序编译并且运行:
- gcc func.c test.c -o test.out
- ./test.out
- 毫无疑问,程序肯定会产生错误,如下图:
- 这是在意料之中的,毕竟在func.c程序中,我们对0地址进行写内容了。
- 那么现在我们开始使用gdb来定位出错误,在开启gdb调试之前,需要在编译源程序的时候加上-g选项,并将程序的崩溃信息转储的core文件(这在【软件开发底层知识修炼】六 Binutils辅助工具之- addr2line与strip工具这篇文章中有讲解过)。
- gcc -g test.c func.c -o test.out //重新编译加上调试信息
- ulimit -c unlimited //让程序在崩溃时产生core文件
- ./test.out //重新运行看看是否产生core文件
- 很明显核心已转出,生成了core文件----注意,这是GDB调试需要用的文件
- 输入命令:gdb test.out core 进行调试:
- 从这里我们甚至都直接看到了产生段错误的地方就在func.c程序中的第7行,func函数中出的问题。
- 输入quit命令退出当前gdb调试
- 输入命令:gdb 进行调试:
- 输入gdb后再gdb调试模式下输入:file test.out
- 然后字gdb调试模式下输入run,显示结果最后部分如下:
- 通过上面的动态图我们很容易发现,程序执行的很快,瞬间就到了段错误那里。我们回到test.c程序中,会发现这段代码中:
for(i=0; i<100; i++)
{
fa[i%3] ();
sleep(argc > 1); // 如果argc大于1,则执行睡眠函数
}
由于sleep参数中argc的参数为1(只有./test.out这个参数),所以不会睡眠。- 但是我们可以在gdb中进行设置参数,输入命令set args D.T.SoftWare:
- 就像下面的动态图一样:
- 很明显,我们的程序运行起来变得慢很多,这是因为我们加了一个命令行参数D.T.SoftWare,现在命令行参数就有两个,一个是可执行程序test.out,一个是D.T.SoftWare(如果这里不明白命令行参数的意思,请参考这篇文章:main函数参数)
- 最后,输入ctrl + c可以终止程序的执行,再输入continue可以继续执行刚刚被终止的程序。
- gdb动态连接到一个正在执行的程序,然后对其进行调试
- 还是直接看下面的动图,更加好理解:
- 在右边的终端我们运行程序的时候加一个参数D.T.SoftWare这样可以让上述的for循环中的sleep开启,让程序执行的慢一点
- 程序执行起来后,在左边的终端首先输入ps aux查看我们的程序的pid
- 然后sudo gdb 开启gdb,这里加上sudo以root模式开启,是因为动态连接正在运行的程序的话就需要以root模式
- 开启gdb进入gdb模式后,使用attach pid (这里的pid根据你自己查到的pid写)连接到我们运行的程序。
- 在我们连接到程序的一瞬间,发现程序的执行停止了(使用continue可以继续程序的执行,当程序运行到段错误那里,gdb可以发现错误),说明已经连接到运行中的程序,现在可以使用gdb对它进行调试了。怎么调试随你意,上面我们也说了几种简单的调试方法。
4 总结
本文只是介绍GDB调试的几种简单用法。下一篇文章会学习GDB的断点调试。
本文章参考狄泰软件学院相关课程
想学习的可以加狄泰软件学院群,
群聊号码:199546072
学习探讨加个人(可以免费帮忙下载CSDN资源):
qq:1126137994
微信:liu1126137994