关闭

关闭

封号提示

内容

首页 GDB调试命令手册.pdf

GDB调试命令手册.pdf

GDB调试命令手册.pdf

上传者: 秋末 2012-04-02 评分 0 0 0 0 0 0 暂无简介 简介 举报

简介:本文档为《GDB调试命令手册pdf》,可适用于IT/计算机领域,主题内容包含GDB调试命令常用的gdb命令backtrace显示程序中的当前位置和表示如何到达当前位置的栈跟踪(同义词:where)breakpoint在程序中符等。

GDB调试命令常用的gdb命令backtrace显示程序中的当前位置和表示如何到达当前位置的栈跟踪(同义词:where)breakpoint在程序中设置一个断点cd改变当前工作目录clear删除刚才停止处的断点commands命中断点时列出将要执行的命令continue从断点开始继续执行delete删除一个断点或监测点也可与其他命令一起使用display程序停止时显示变量和表达时down下移栈帧使得另一个函数成为当前函数frame选择下一条continue命令的帧info显示与该程序有关的各种信息jump在源程序中的另一点开始运行kill异常终止在gdb控制下运行的程序list列出相应于正在执行的程序的原文件内容next执行下一个源程序行从而执行其整体中的一个函数print显示变量或表达式的值pwd显示当前工作目录pype显示一个数据结构(如一个结构或C类)的内容quit退出gdbreversesearch在源文件中反向搜索正规表达式run执行该程序search在源文件中搜索正规表达式setvariable给变量赋值signal将一个信号发送到正在运行的进程step执行下一个源程序行必要时进入下一个函数undisplaydisplay命令的反命令不要显示表达式until结束当前循环up上移栈帧使另一函数成为当前函数watch在程序中设置一个监测点(即数据断点)whatis显示变量或函数类型GDB调试命令GDB命令分类详解一:列文件清单二:执行程序三:显示数据四:断点(breakpoint)五.断点的管理六.变量的检查和赋值七.单步执行八.函数的调用九.机器语言工具十.信号十一原文件的搜索十二UNIX接口十三命令的历史十四GDB帮助十五GDB多线程十六GDB使用范例一:列文件清单.List(gdb)listline,line二:执行程序要想运行准备调试的程序可使用run命令在它后面可以跟随发给该程序的任何参数包括标准输入和标准输出说明符(<和>)和外壳通配符(*、?、、)在内。如果你使用不带参数的run命令gdb就再次使用你给予前一条run命令的参数这是很有用的。利用setargs命令就可以修改发送给程序的参数而使用showargs命令就可以查看其缺省参数的列表。(gdb)setargs–b–x(gdb)showargsbacktrace命令为堆栈提供向后跟踪功能。Backtrace命令产生一张列表包含着从最近的过程开始的所以有效过程和调用这些过程的参数。三:显示数据利用print命令可以检查各个变量的值。(gdb)printp(p为变量名)whatis命令可以显示某个变量的类型(gdb)whatisptype=int*print是gdb的一个功能很强的命令利用它可以显示被调试的语言中任何有效的表达式。表达式除了包含你程序中的变量外还可以包含以下内容:GDB调试命令l对程序中函数的调用(gdb)printfindentry(,)l数据结构和其他复杂对象(gdb)print*tablestart$={e=reference=’’,location=x,next=x}l值的历史成分(gdb)print$($为历史记录变量,在以后可以直接引用$的值)l人为数组人为数组提供了一种去显示存储器块(数组节或动态分配的存储区)内容的方法。早期的调试程序没有很好的方法将任意的指针换成一个数组。就像对待参数一样让我们查看内存中在变量h后面的个整数一个动态数组的语法如下所示:baselength因此要想显示在h后面的个元素可以使用h:(gdb)printh$=(,,,,,,,,,)四:断点(breakpoint)break命令(可以简写为b)可以用来在调试的程序中设置断点该命令有如下四种形式:lbreaklinenumber使程序恰好在执行给定行之前停止。lbreakfunctionname使程序恰好在进入指定的函数之前停止。lbreaklineorfunctionifconditio如果condition(条件)是真程序到达指定行或函数时停止。lbreakroutinename在指定例程的入口处设置断点如果该程序是由很多原文件构成的你可以在各个原文件中设置断点而不是在当前的原文件中设置断点其方法如下:(gdb)breakfilename:linenumber(gdb)breakfilename:functionname要想设置一个条件断点可以利用breakif命令如下所示:(gdb)breaklineorfunctionifexpr例:(gdb)breakiftestsize==从断点继续运行:countinue命令五.断点的管理.显示当前gdb的断点信息:(gdb)infobreak他会以如下的形式显示所有的断点信息:NumTypeDispEnbAddressWhatbreakpointkeepyxbcininitrandomatqsortc:breakpointkeepyxcininitorganatqsortc:(gdb)删除指定的某个断点:(gdb)deletebreakpoint该命令将会删除编号为的断点如果不带编号参数将删除所有的断点GDB调试命令(gdb)deletebreakpoint禁止使用某个断点(gdb)disablebreakpoint该命令将禁止断点,同时断点信息的(Enb)域将变为n.允许使用某个断点(gdb)enablebreakpoint该命令将允许断点,同时断点信息的(Enb)域将变为y.清除原文件中某一代码行上的所有断点(gdb)cleannumber注:number为原文件的某个代码行的行号六.变量的检查和赋值lwhatis:识别数组或变量的类型lptype:比whatis的功能更强他可以提供一个结构的定义lsetvariable:将值赋予变量lprint:除了显示一个变量的值外还可以用来赋值七.单步执行next不进入的单步执行step进入的单步执行finish如果已经进入了某函数而想退出该函数返回到它的调用函数中可使用命令finish八.函数的调用lcallname调用和执行一个函数(gdb)callgenandsork(,,)(gdb)callprintf(“abcd”)$=lfinish结束执行当前函数显示其返回值(如果有的话)九.机器语言工具有一组专用的gdb变量可以用来检查和修改计算机的通用寄存器gdb提供了目前每一台计算机中实际使用的个寄存器的标准名字:$pc程序计数器$fp帧指针(当前堆栈帧)$sp栈指针$ps处理器状态十.信号gdb通常可以捕捉到发送给它的大多数信号通过捕捉信号它就可决定对于正在运行的进程要做些什么工作。例如按CTRLC将中断信号发送给gdb通常就会终止gdb。但是你或许不想中断gdb真正的目的是要中断gdb正在运行的程序因此gdb要抓住该信号并停止它正在运行的程序这样就可以执行某些调试操作。Handle命令可控制信号的处理他有两个参数一个是信号名另一个是接受到信号时该作什么。GDB调试命令几种可能的参数是:nostop接收到信号时不要将它发送给程序也不要停止程序。stop接受到信号时停止程序的执行从而允许程序调试显示一条表示已接受到信号的消息(禁止使用消息除外)print接受到信号时显示一条消息noprint接受到信号时不要显示消息(而且隐含着不停止程序运行)pass将信号发送给程序从而允许你的程序去处理它、停止运行或采取别的动作。nopass停止程序运行但不要将信号发送给程序。例如假定你截获SIGPIPE信号以防止正在调试的程序接受到该信号而且只要该信号一到达就要求该程序停止并通知你。要完成这一任务可利用如下命令:(gdb)handleSIGPIPEstopprint请注意UNIX的信号名总是采用大写字母!你可以用信号编号替代信号名如果你的程序要执行任何信号处理操作就需要能够测试其信号处理程序为此就需要一种能将信号发送给程序的简便方法这就是signal命令的任务。该命令的参数是一个数字或者一个名字如SIGINT。假定你的程序已将一个专用的SIGINT(键盘输入或CTRLC信号)信号处理程序设置成采取某个清理动作要想测试该信号处理程序你可以设置一个断点并使用如下命令:(gdb)signalcontinuingwithsignalSIGINT()该程序继续执行但是立即传输该信号而且处理程序开始运行十一原文件的搜索Searchtext:该命令可显示在当前文件中包含text串的下一行。Reversesearchtext:该命令可以显示包含text的前一行。十二UNIX接口shell命令可启动UNIX外壳CTRLD退出外壳返回到gdb十三命令的历史为了允许使用历史命令可使用sethistoryexpansionon命令(gdb)sethistoryexpansionon十四GDB帮助在gdb提示符处键入help将列出命令的分类主要的分类有:*aliases:命令别名*breakpoints:断点定义*data:数据查看*files:指定并查看文件*internals:维护命令*running:程序执行*stack:调用栈查看*statu:状态查看*tracepoints:跟踪程序执行。键入help后跟命令的分类名可获得该类命令的详细清单。GDB调试命令十五GDB多线程先介绍一下GDB多线程调试的基本命令。linfothreads显示当前可调试的所有线程每个线程会有一个GDB为其分配的ID后面操作线程的时候会用到这个ID。前面有*的是当前调试的线程。lthreadID切换当前调试的线程为指定ID的线程。lbreakthreadtestc:threadall在所有线程中相应的行上设置断点lthreadapplyIDIDcommand让一个或者多个线程执行GDB命令command。lthreadapplyallcommand让所有被调试线程执行GDB命令command。lsetschedulerlockingoff|on|step估计是实际使用过多线程调试的人都可以发现在使用step或者continue命令调试当前被调试线程的时候其他线程也是同时执行的怎么只让被调试程序执行呢?通过这个命令就可以实现这个需求。off不锁定任何线程也就是所有线程都执行这是默认值。on只有当前被调试程序会执行。step在单步的时候除了next过一个函数的情况(熟悉情况的人可能知道这其实是一个设置断点然后continue的行为)以外只有当前线程会执行。在介绍完基本的多线程调试命令后大概介绍一下GDB多线程调试的实现思路。比较主要的代码是threadc前面介绍的几个命令等都是在其中实现。threadlist这个表存储了当前可调试的所有线程的信息。函数addthreadsilent或者addthread(不同版本GDB不同)用来向threadlist列表增加一个线程的信息。函数deletethread用来向threadlist列表删除一个线程的信息。上面提到的这个函数会被有线程支持的target调用用来增加和删除线程不同的OS对线程的实现差异很大这么实现比较好的保证了GDB多线程调试支持的扩展性。函数infothreadscommand是被命令infothreads调用的就是显示threadlist列表的信息。函数threadcommand是被命令thread调用切换当前线程最终调用的函数是switchtothread这个函数会先将当前调试线程变量inferiorptid然后对寄存器和frame缓冲进行刷新。函数threadapplycommand被命令threadapply调用这个函数的实际实现其实很简单就是先切换当前线为指定线程然后调用函数executecommand调用指定函数。比较特别的是setschedulerlocking没有实现在threadc中而是实现在控制被调试程序执行的文件infrunc中。对其的设置会保存到变量schedulermode中而实际使用这个变量的函数只有用来令被调试程序执行的函数resume。在默认情况下传递给targetresume的变量是resumeptid默认情况下其的值为RESUMEALL也就是告诉target程序执行的时候所有被调试线程都要被执行。而当schedulermode设置为只让当前线程执行的时候resumeptid将被设置为inferiorptid这就告诉target只有inferiorptid的线程会被执行。最后特别介绍一下Linux下多线程的支持基本的调试功能在linuxnatc中这里有对Linux轻量级别进程本地调试的支持。但是其在调试多线程程序的时候还需要对pthread调试的支持这个功能实现在linuxthreaddbc中。对pthread的调试要通过调用libthreaddb库来支持。这里有一个单独的target"multithread"这个target有点很特别:GDB调试命令第一一般target的装载是在调用相关toopen函数的时候调用pushtarget进行装载。而这个target则不同在其初始化的时候就注册了函数threaddbnewobjfile到库文件attach事件中。这样当GDB为调试程序的动态加载库时候attach库文件的时候就会调用这个函数threaddbnewobjfile。这样当GDB装载libpthread库的时候最终会装载target"multithread"。第二这个target并没有像大部分target那样自己实现了全部调试功能其配合linuxnatc的代码的功能这里有一个target多层结构的设计要介绍的比较多就不详细介绍了。最后介绍一下我最近遇见的一个多线程调试和解决。基本问题是在一个Linux环境中调试多线程程序不正常infothreads看不到多线程的信息。我先用命令maintenanceprinttargetstack看了一下target的装载情况发现target"multithread"没有被装载用GDB对GDB进行调试发现在函数checkforthreaddb在调用libthreaddb中的函数tdtanew的时候返回了TDNOLIBTHREAD所以没有装载target"multithread"。在时候我就怀疑是不是libpthread有问题于是检查了一下发现了问题这个环境中的libpthread是被strip过的我想可能就是以为这个影响了tdtanew对libpthread符号信息的获取。当我换了一个没有strip过的libpthread的时候问题果然解决了。最终我的解决办法是拷贝了一个debug版本的libpthread到lib目录中问题解决了。多线程如果dump多为段错误一般都涉及内存非法读写。可以这样处理使用下面的命令打开系统开关让其可以在死掉的时候生成core文件。ulimitcunlimited这样的话死掉的时候就可以在当前目录看到corepid(pid为进程号)的文件。接着使用gdb:gdbbincorepid进去后使用bt查看死掉时栈的情况在使用frame命令。还有就是里面某个线程停住也没死这种情况一般就是死锁或者涉及消息接受的超时问题(听人说的没有遇到过)。遇到这种情况可以使用:gcorepid(调试进程的pid号)手动生成core文件在使用pstack(linux下好像不好使)查看堆栈的情况。如果都看不出来就仔细查看代码看看是不是在ifreturnbreakcontinue这种语句操作是忘记解锁还有嵌套锁的问题都需要分析清楚了。十六GDB使用范例清单一个有错误的C源程序buggingc代码:#includestaticcharbuffstaticchar*stringintmain(){printf("Pleaseinputastring:")gets(string)printf("nYourstringis:sn",string)}GDB调试命令上面这个程序非常简单其目的是接受用户的输入然后将用户的输入打印出来。该程序使用了一个未经过初始化的字符串地址string因此编译并运行之后将出现SegmentFault错误:$gccobugginggbuggingc$buggingPleaseinputastring:asfdSegmentationfault(coredumped)为了查找该程序中出现的问题我们利用gdb并按如下的步骤进行:.运行gdbbugging命令装入bugging可执行文件.执行装入的bugging命令run.使用where命令查看程序出错的地方.利用list命令查看调用gets函数附近的代码.唯一能够导致gets函数出错的因素就是变量string。用print命令查看string的值.在gdb中我们可以直接修改变量的值只要将string取一个合法的指针值就可以了为此我们在第行处设置断点break.程序重新运行到第行处停止这时我们可以用setvariable命令修改string的取值.然后继续运行将看到正确的程序运行结果

职业精品

精彩专题

上传我的资料

热门资料

资料评价:

/ 8
所需积分:2 立即下载

意见
反馈

返回
顶部

Q