您好,欢迎访问三七文档
当前位置:首页 > IT计算机/网络 > 数据库 > UNIX下命令行调试工具GDB使用教程
UNIX下命令行调试工具GDB使用教程GDB是GNU开源组织发布的一个强大的UNIX下的程序调试工具。它是一种强大的命令行调试工具。一般来说,调试器的功能:能够运行程序,设置所有能影响程序运行的参数;能够让程序在指定条件下停止运行;能够在程序停止时检查所有参数的情况;能够根据指定条件改变程序的运行。gdb调试源代码流程:1)进入GDB#gdbtest只需输入GDB和要调试的可执行文件即可,在GDB的启动画面中指出了GDB的版本号、遵循的许可等信息,接下来就进入了由(gdb)开头的命令行界面了;2)查看文件(gdb)l在GDB中键入l(list)就可以查看程序的源码了,GDB列出的源代码中明确地给出了对应的行号,方便代码的定位;3)设置断点(gdb)b6只需在b(break)后加入对应的行号即可,在GDB中利用行号设置断点是指代码运行到对应行之前暂停;设置断点可以使程序执行到某个位置时暂停,程序员在该位置处可以方便地查看变量的值、堆栈情况等;一般情况下,源代码中大家行号与用户书写程序的行号是一致的,但有时由于用户的某些编译选项会导致行号不一致的情况,因此,一定要查看在GDB中的行号;4)查看断点处情况(gdb)infob可以键入infob来查看断点处情况,可以设置多个断点;5)运行代码(gdb)rGDB默认从首行开始运行代码,键入r(run)即可;程序运行到断点处停止。6)看变量值(gdb)pn在程序暂停之后,程序员可以查看断点处的相关变量值,在GDB中只需键入p变量名(print)即可;GDB在显示变量值时都会在对应值之前加上$N标记,它是当前变量值的引用标记,以后若想再次引用此变量,就可以直接写作$N,而无需写冗长的变量名;7)观察变量(gdb)watchn在某一循环处,往往希望能够观察一个变量的变化情况,这时就可以键入命令watch来观察变量的变化情况,GDB在n设置了观察点;8)单步运行(gdb)n单步运行是指一次只运行一条语句,这样可以方便查看程序运行的结果,在此处只需键入n(next)即可;随着程序的单步运行,当变量n的值发生变化时,GDB就会自动显示出n的变化情况。9)程序继续运行(gdb)c命令c(continue)可以使程序继续往下运行,直到再次遇到断点或程序结束;10)退出GDB(gdb)q只需使用指令q(quit)即可。设置/删除断点命令格式例子作用break+设置断点的行号beakn用于在程序中对应行设置断点tbreak+行号或函数名tbreakn/func设置临时断点,到达后被自动删除break+filename+行号breakmain.c:10用于在指定文件对应行设置断点break+0x...break0x3400a用于在内存某一位置处暂停break+行号+if+条件break10ifi==3用于设置条件断点,在循环中使用非常方便infobreakpoints/watchpoints[n]infobreakn表示断点号,查看断点/观察点的情况clear+要清除的断点行号clear10用于清除对应行的断点,要给出断点的行号,清除时GDB会给出提示delete+要清除的断点编号delete3用于清除断点和自动显示的表达式的命令,要给出断点的编号,清除时GDB不会给出任何提示disable/enable+断点编号disable3让所设断点暂时失效/使能,如果要让多个编号处的断点失效/使能,可将编号之间用空格隔开awatch/watch+变量awatch/watchi设置一个观察点,当变量被读出或写入时程序被暂停rwatch+变量rwatchi设置一个观察点,当变量被读出时,程序被暂停catch设置捕捉点来补捉程序运行时的一些事件。如:载入共享库(动态链接库)或是C++的异常tcatch只设置一次捕捉点,当程序停住以后,应点被自动删除断点与观察点的区别:1.所有使用与breakpoint的操作都适用于watchpoint2.断点是CPU到某一地址取指令时中断,观察点是CPU到某一地址读写数据时中断。在多线程的程序中,观察点的作用很有限,GDB只能观察一个线程中的表达式的值,如果用户确信表达式只被当前线程所存取,那么使用观察点才有效,GDB无法检测一个非当前线程对表达式值的改变。数据相关命令display+表达式displaya用于显示表达式的值,每当程序运行到断点处都会显示表达式的值infodisplay用于显示当前所有要显示值的表达式的情况delete+display编号delete3用于删除一个要显示值的表达式,被删除的表达式将不被显示disable/enable+display编号disable/enable3使一个要显示值的表达式暂时失效/使能undisplay+display编号undisplay3用于结束某个表达式值的显示whatis+变量whatisi显示某个表达式的数据类型print(p)+变量/表达式pn用于打印变量或表达式的值set+变量=变量值seti=3改变程序中某个变量的值在使用print命令时,可以对变量按指定格式进行输出,其命令格式为print/变量名+格式其中常用的变量格式:x:十六进制;d:十进制;u:无符号数;o:八进制;c:字符格式;f:浮点数。调试运行环境相关命令setargssetargsarg1arg2设置运行参数showargsshowargs参看运行参数setwidth+数目setwidth70设置GDB的行宽cd+工作目录cd../切换工作目录runr/run程序开始执行step(s)s进入式(会进入到所调用的子函数中)单步执行,进入函数的前提是,此函数被编译有debug信息next(n)n非进入式(不会进入到所调用的子函数中)单步执行finishfinish一直运行到函数返回并打印函数返回时的堆栈地址和返回值及参数值等信息until+行数u3运行到函数某一行continue(c)c执行到下一个断点或程序结束return返回值return5改变程序流程,直接结束当前函数,并将指定值返回call+函数callfunc在当前位置执行所要运行的函数堆栈相关命令backtrace/btbt用来打印栈帧指针,也可以在该命令后加上要打印的栈帧指针的个数,查看程序执行到此时,是经过哪些函数呼叫的程序,程序“调用堆栈”是当前函数之前的所有已调用函数的列表(包括当前函数)。每个函数及其变量都被分配了一个“帧”,最近调用的函数在0号帧中(“底部”帧)frameframe1用于打印指定栈帧inforeginforeg查看寄存器使用情况infostackinfostack查看堆栈使用情况up/downup/down跳到上一层/下一层函数跳转执行一般来说,被调试程序会按照程序代码的运行顺序依次执行。GDB提供了乱序执行的功能,也就是说,GDB可以修改程序的执行顺序,可以让程序执行随意跳跃。这个功能可以由GDB的jump命令来完:jump指定下一条语句的运行点。可以是文件的行号,可以是file:line格式,可以是+num这种偏移量格式。表式着下一条运行语句从哪里开始。jump这里的是代码行的内存地址。注意,jump命令不会改变当前的程序栈中的内容,所以,当你从一个函数跳到另一个函数时,当函数运行完返回时进行弹栈操作时必然会发生错误,可能结果还是非常奇怪的,甚至于产生程序CoreDump。所以最好是同一个函数中进行跳转。熟悉汇编的人都知道,程序运行时,有一个寄存器用于保存当前代码所在的内存地址。所以,jump命令也就是改变了这个寄存器中的值。于是,你可以使用“set$pc”来更改跳转执行的地址。如:set$pc=0x485signal命令信号是一种软中断,是一种处理异步事件的方法,使用singal命令,可以产生一个信号量给被调试程序,如中断信号Ctrl+C。这非常方便于程序的调试,可以在程序运行的任意位置设置断点,并在该断点用GDB产生一个信号量,这种精确地在某处产生信号的方法非常有利于程序的调试,signal命令语法是:signalsignal,UNIX的系统信号量通常从1到15,所以signal取值也在这个范围。handle在GDB中定义一个信号处理。信号可以以SIG开头或不以SIG开头,可以用定义一个要处理信号的范围(如:SIGIO-SIGKILL,表示处理从SIGIO信号到SIGKILL的信号,其中包括SIGIO,SIGIOT,SIGKILL三个信号),也可以使用关键字all来标明要处理所有的信号。一旦被调试的程序接收到信号,运行程序马上会被GDB停住,以供调试。其可以是以下几种关键字的一个或多个。nostop/stop当被调试的程序收到信号时,GDB不会停住程序的运行,但会打出消息告诉你收到这种信号/GDB会停住你的程序print/noprint当被调试的程序收到信号时,GDB会显示出一条信息/GDB不会告诉你收到信号的信息passnoignore当被调试的程序收到信号时,GDB不处理信号。这表示,GDB会把这个信号交给被调试程序会处理。nopassignore当被调试的程序收到信号时,GDB不会让被调试程序来处理这个信号。infosignalsinfohandle可以查看哪些信号被GDB处理,并且可以看到缺省的处理方式single命令和shell的kill命令不同,系统的kill命令发信号给被调试程序时,是由GDB截获的,而single命令所发出一信号则是直接发给被调试程序的。GDB中运行UNIX的shell程序在gdb环境中,你可以执行UNIX的shell的命令,使用gdb的shell命令来完成:shell调用UNIX的shell来执行,环境变量SHELL中定义的UNIX的shell将会被执行,如果SHELL没有定义,那就使用UNIX的标准shell:/bin/sh。(在Windows中使用Command.com或cmd.exe)make可以在gdb中执行make命令来重新build自己的程序。这个命令等价于shellmake在GDB中运行程序当以gdb方式启动gdb后,gdb会在PATH路径和当前目录中搜索的源文件。如要确认gdb是否读到源文件,可使用l或list命令,看看gdb是否能列出源代码。在gdb中,运行程序使用r或是run命令。1、程序运行参数。setargs可指定运行时参数。(如:setargs1020304050)showargs命令可以查看设置好的运行参数。2、运行环境。path可设定程序的运行路径。showpaths查看程序的运行路径。setenvironmentvarname[=value]设置环境变量。如:setenvUSER=hchenshowenvironment[varname]查看环境变量。3、工作目录。cd相当于shell的cd命令。pwd显示当前的所在目录。4、程序的输入输出。infoterminal显示你程序用到的终端的模式。使用重定向控制程序输出。如:runoutfiletty命令可以指写输入输出的终端设备。如:tty/dev/ttyb调试已运行的程序两种方法:1、在UNIX下用ps查看正在运行的程序的PID(进程ID),然后用gdbPID格式挂接正在运行的程序。2、先用gdb关联上源代码,并进行gdb,在gdb中用attach命令来挂接进程的PID。并用detach来取消挂接的进程。暂停/恢复程序运行当进程被gdb停住时,你可以使用infoprogram来查看程序的是否在运行,进程号,被暂停的原因。在gdb中,我们可以有以下几种暂停方式:断点(BreakPoint)、观察点(WatchPoint)、捕捉点(CatchPoint)、信号(Signals)、线程停止(ThreadStops),如果要恢复程序运行,可以使用c或是continue命令。线程(ThreadStops)如果程序是多线程,可以定义断点是否在所有的线程上,或是在某个特定的线程。breakthreadbreakthreadif...
本文标题:UNIX下命令行调试工具GDB使用教程
链接地址:https://www.777doc.com/doc-2865081 .html