GDB使用说明文档
一、简介:
GDB能让你观察一个程序在执行时的内部活动,或程序出错时发生了什么。
GDB主要能为你做以下四件事,帮助你出程序中的错误。
1. 运行你的程序,设置所有的能影响程序运行的东西。
2. 保证你的程序在指定的条件下停止。
3. 当你程序停止时,让你检查发生了什么。
4. 改变你的程序。那样你可以试着修正某个bug引起的问题,然后继续查另一个bug.
二、GDB运行方式
1. 通常的调试可执行程序:gdb <可执行文档名>
2. 调试执行文件指定一个core文件:gdb <可执行文件名> core
3. 为执行的文件指定一个进程号:gdb <可执行文件名> <进程号>
三、GDB常用命令
下面先说明GDB的基本指令:(大部分命令使用时只要输入第一个字母就好了,同时支持TAB的自动补全,与shell相类似)
1. help:查看帮助
2. file:指定一个可执行文件进行调试,gdb将读取些文件的调试信息
3. list:列出程序源文件
4. run:装载完要调试的可执行文件后,可以用run命令运行可执行文件
5. break:设置断点breakpoint,如b 25,则在源程序的第25行设置一个断点,当程序执行到第25行时,就会产生中断;也可以使用b funcname,funcname为函数的名称,当程序调用些函数时,则产生中断
6. continue:c命令可以使中断的程序继续执行,直到下一个中断点或程序结束
7. print:输入某个变量的值,如程序定义了一个int aa的就是,p aa就会输出aa的当前值
8. next:程序执行到断点时中断执行,可以用n指令进行单步执行
9. step:程序执行到断点时中断执行,可以用s指令进行单步执行进某一函数,如果已经进入了某函数,而想退出该函数返回到它的调用函数中,可使用命令finish
10. attach:命令为attach PROCESS-ID,这个命令把一个已经运行的进程(在gdb外启动)连接入gdb,以便 调试。PROCESS-ID是进程号,当gdb接到attach命令后第一件事就是停止进程的运行
11. detach:与attach相对应,不多解释
12. thread:命令为thread THREADNO,把线程号为THREADNO的线程设为当前线程。命令行参数THREADNO是gdb内定的 线程号。你可以用info threads命令来查看gdb内设置的线程号
13. kill: 终止正在调试的程序
14. watch: 使你能监视一个变量的值而不管它何时改变, 当表达式的值被改变时GDB就使程序停止,还有rwatch是使程序暂停
15. clear:使用clear命令你可以删除指定位置的断点,如:clear FUNCTION, clear LINENUM,也可以使用delete命令通过断点号来指定要删去的断点或观察点,如果没有指定参数则删去程序中所有的断点
16. make: 使你能不退出gdb就可以重新产生可执行文件
17. shell:使你能不退出gdb就可以执行shell命令
18. info:用来显示你程序的状态,要获得详细的关于info的信息可通过help info查看
19. quit:退出GDB
20. 查看栈信息(bt frame up down )
当程序被停住了,你需要做的第一件事就是查看程序是在哪里停住的。当你的程序调用了一个函数,函数的地址,函数参数,函数内的局部变量都会被压入“栈”(Stack)中。你可
以用GDB命令来查看当前的栈中的信息。
下面是一些查看函数调用栈信息的GDB命令:
bt:打印当前的函数调用栈的所有信息。如:
(gdb) bt
#0 func (n=250) at tst.c:6
#1 0x08048524 in main (argc=1, argv=0xbffff674) at tst.c:30
#2 0x400409ed in __libc_start_main () from /lib/libc.so.6
从上可以看出函数的调用栈信息:__libc_start_main --> main() --> func()
bt <n>:n是一个正整数,表示只打印栈顶上n层的栈信息。
bt <-n>:-n表一个负整数,表示只打印栈底下n层的栈信息。
如果你要查看某一层的信息,你需要在切换当前的栈,一般来说,程序停止时,最顶层的栈就是当前栈,如果你要查看栈下面层的详细信息,首先要做的是切换当前栈。
f <n> :n是一个从0开始的整数,是栈中的层编号。比如:frame 0,表示栈顶,frame 1,表示栈的第二层。
up <n>:表示向栈的上面移动n层,可以不打n,表示向上移动一层。
down <n> :表示向栈的下面移动n层,可以不打n,表示向下移动一层。
查看当前栈层的信息,你可以用以下GDB命令:
frame 或 f :会打印出这些信息,栈的层编号,当前的函数名,函数参数值,函数所在文件及行号,函数执行到的语句。
info f :这个命令会打印出更为详细的当前栈层的信息,只不过,大多数都是运行时的内内地址。比如:函数地址,调用函数的地址,被调用函数的地址,目前的函数是由什么样的程序语言写成的、函数参数地址及值、局部变量的地址等等。如:
info args:打印出当前函数的参数名及其值。
info locals:打印出当前函数中所有局部变量及其值。
info catch:打印出当前的函数中的异常处理信息。
21. 查看内存
你可以使用examine命令(简写是x)来查看内存地址中的值。x命令的语法如下所示:
x/<n/f/u> <addr>
n、f、u是可选的参数。
n 是一个正整数,表示显示内存的长度,也就是说从当前地址向后显示几个地址的内容。
f 表示显示的格式,如果地址所指的是字符串,那么格式可以是s,如果地址所指的是指令地址,那么格式可以是i,还有x表示的是十六进制。
u 表示从当前地址往后请求的字节数,如果不指定的话,GDB默认是4个bytes。u参数可以用下面的字符来代替,b表示单字节,h表示双字节,w表示四字节,g表示八字节。当我们指定了字节长度后,GDB会从指定内存的内存地址开始,读写指定字节,并把其当作一个值取出来。
<addr>表示一个内存地址。
n/f/u三个参数可以一起使用。例如:
命令:x/3uh 0x54320 表示,从内存地址0x54320读取内容,h表示以双字节为一个单位,3表示三个单位,u表示按十六进制显示。
四、多进程如何gdb子进程
gdb对调试使用fork系统调用产生新进程的程序没有很多支持。当一个程序开始一个新进程时,gdb将继续对父进程进行调试,子进程将不受影响的运行。如果你在子进程可能会执行到的地方设了断点,那么子进程将收到SIGTRAP信号,如果子进程没有对这个信号进行处理的话那么缺省的处理就是使子进程终止。然而,如果你要一定要调试子进程的话,可
以在子进程被运行起来的开头几句语句前加上一个sleep命令。这在调试过程中并不会引起程序中很大的麻烦。然后再使用ps命令列出新开的子进程号,最后使用attach命令。这样就没有问题了。
五、简单的实例演示
源程序:tst.c
int func(int n)
{
int sum=0,i;
for(i=1; i<=n; ++i)
{
sum+=i;
}
return sum;
}
main()
{
int i;
long result = 0;
for(i=1; i<=100; i++)
{
result += i;
}
printf("result[1-100] = %d \n", result );
printf("result[1-250] = %d \n", func(250) );
}
编译生成执行文件:(Linux下)
cc -g tst.c -o tst
使用GDB调试:
gdb tst <---------- 启动GDB
GNU gdb 5.1.1
Copyright 2002 Free Software Foundation, Inc.
GDB is free software, covered by the GNU General Public License, and you are
welcome to change it and/or distribute copies of it under certain conditions.
Type "show copying" to see the conditions.
There is absolutely no warranty for GDB. Type "show warranty" for details.
This GDB was configured as "i386-suse-linux"...
(gdb) l <-------------------- l命令相当于list,从第一行开始例出原码。
1 #i nclude
2
3 int func(int n)
4 {
5 int sum=0,i;
6 for(i=1; i<=n; ++i)
7 {
8 sum+=i;
9 }
10 return sum;
(gdb) <-------------------- 直接回车表示,重复上一次命令
exited11 }
12
13
14 main()
15 {
16 int i;
17 long result = 0;
18 for(i=1; i<=100; i++)
19 {
20 result += i;
(gdb) break 16 <-------------------- 设置断点,在源程序第16行处。
Breakpoint 1 at 0x8048496: file tst.c, line 16.
(gdb) break func <-------------------- 设置断点,在函数func()入口处。
版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系QQ:729038198,我们将在24小时内删除。
发表评论