原文地址:
http://blog.csdn.net/u010587742/article/details/22786661
首先为该函数打断点,打完断点后程序运行直到断点处,在gdb中输入命令bt即可打出断点函数的调用栈,从上到下理清该函数的完整调用流程,个人今天觉着这功能挺牛逼强大,分享一下便于日后查阅。
后面内容为转载:
下面先说明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命令 文档冲亿季,好礼乐相随mini ipad移动硬盘拍立得百度书包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层的栈信息。 bt <-n>:-n表一个负整数,表示只打印栈底下n层的栈信息。 如果你要查看某一层的信息,你需要在切换当前的栈,一般来说,程序停止时,最顶层的栈就是当前栈,如果你要查看栈下面层的详细信息,首先要做的是切换当前栈。 f :n是一个从0开始的整数,是栈中的层编号。比如:frame 0,表示栈顶,frame 1,表示栈的第二层。 up :表示向栈的上面移动n层,可以不打n,表示向上移动一层。 down :表示向栈的下面移动n层,可以不打n,表示向下移动一层。 查看当前栈层的信息,你可以用以下GDB命令: frame 或 f :会打印出这些信息,栈的层编号,当前的函数名,函数参数值,函数所在文件及行号,函数执行到的语句。 info f :这个命令会打印出更为详细的当前栈层的信息,只不过,大多数都是运行时的内内地址。比如:函数地址,调用函数的地址,被调用函数的地址,目前的函数是由什么样的程序语言写成的、函数参数地址及值、局部变量的地址等等。如: info args:打印出当前函数的参数名及其值。 info locals:打印出当前函数中所有局部变量及其值。 info catch:打印出当前的函数中的异常处理信息。 21. 查看内存 你可以使用examine命令(简写是x)来查看内存地址中的值。x命令的语法如下所示: x/ n、f、u是可选的参数。 n 是一个正整数,表示显示内存的长度,也就是说从当前地址向后显示几个地址的内容。 f 表示显示的格式,如果地址所指的是字符串,那么格式可以是s,如果地址所指的是指令地址,那么格式可以是i,还有x表示的是十六进制。 u 表示从当前地址往后请求的字节数,如果不指定的话,GDB默认是4个bytes。u参数可以用下面的字符来代替,b表示单字节,h表示双字节,w表示四字节,g表示八字节。当我们指定了字节长度后,GDB会从指定内存的内存地址开始,读写指定字节,并把其当作一个值取出来。 表示一个内存地址。 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) <-------------------- 直接回车表示,重复上一次命令 11 } 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()入口处。 Breakpoint 2 at 0x8048456: file tst.c, line 5. (gdb) info break <-------------------- 查看断点信息。 Num Type Disp Enb Address What 1 breakpoint keep y 0x08048496 in main at tst.c:16 2 breakpoint keep y 0x08048456 in func at tst.c:5 (gdb) r <--------------------- 运行程序,run命令简写 Starting program: /home/hchen/test/tst Breakpoint 1, main () at tst.c:17 <---------- 在断点处停住。 17 long result = 0; (gdb) n <--------------------- 单条语句执行,next命令简写。 18 for(i=1; i<=100; i++) (gdb) n 20 result += i; (gdb) n 18 for(i=1; i<=100; i++) (gdb) n 20 result += i; (gdb) c <--------------------- 继续运行程序,continue命令简写。 Continuing. result[1-100] = 5050 <----------程序输出。 Breakpoint 2, func (n=250) at tst.c:5 5 int sum=0,i; (gdb) n 6 for(i=1; i<=n; i++) (gdb) p i <--------------------- 打印变量i的值,print命令简写。 $1 = 134513808 (gdb) n 8 sum+=i; (gdb) n 6 for(i=1; i<=n; i++) (gdb) p sum $2 = 1 (gdb) n 8 sum+=i; (gdb) p i $3 = 2 (gdb) n 6 for(i=1; i<=n; i++) (gdb) p sum $4 = 3 (gdb) bt <--------------------- 查看函数堆栈。 #0 func (n=250) at tst.c:5 #1 0x080484e4 in main () at tst.c:24 #2 0x400409ed in __libc_start_main () from /lib/libc.so.6 (gdb) finish <--------------------- 退出函数。 Run till exit from #0 func (n=250) at tst.c:5 0x080484e4 in main () at tst.c:24 24 printf("result[1-250] = %d \n", func(250) ); returned is $6 = 31375 (gdb) c <--------------------- 继续运行。 Continuing. result[1-250] = 31375 <----------程序输出。 Program exited with code 027. <--------程序退出,调试结束。 (gdb) q <--------------------- 退出gdb。
阅读(1865) | 评论(0) | 转发(0) |