⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 gdb-2.htm

📁 GDB相关资料,整理的真的很辛苦,希望大家好好珍惜
💻 HTM
字号:
<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0 Transitional//EN">
<!-- saved from url=(0059)http://www.linuxforum.net/books/LinuxFAQ/program-gdb-2.html -->
<HTML><HEAD><TITLE>BBS水木清华站∶精华区</TITLE>
<META http-equiv=Content-Type content="text/html; charset=gb2312">
<META content="MSHTML 6.00.2600.0" name=GENERATOR></HEAD>
<BODY>
<CENTER>
<H1>BBS水木清华站∶精华区</H1></CENTER>发信人: TJB (老六), 信区: Linux <BR>标&nbsp; 题: GDB (2) 
<BR>发信站: BBS 水木调试站 (Tue Jun&nbsp; 2 15:58:26 1998) 
<P>发 信 
人:System_Killer(大家一起来发呆)&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
信区名称:Linux[4614] <BR>信件提要:gdb(二) <BR>原发信站:中国科大BBS站(Sat, 28 Mar 1998 22:28:43) 
<P>这里是GDB的一个例子: <BR>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
原文中是使用一个叫m4的程序。但很遗憾我找不到这个程序的原代码, <BR>所以没有办法来按照原文来说明。不过反正是个例子,我就拿一个操作系统的 
<BR>进程调度原码来说明把,原代码我会附在后面。 <BR>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
首先这个程序叫os.c是一个模拟进程调度的原程序(也许是个老古董了:-))。 
<BR>先说明一下如何取得包括原代码符号的可执行代码。大家有心的话可以去看一下gcc的 <BR>man文件(在shell下打man gcc)。gcc -g 
&lt;原文件.c&gt; -o &lt;要生成的文件名&gt; <BR>-g 的意思是生成带原代码调试符号的可执行文件。 <BR>-o 
的意思是指定可执行文件名。 <BR>(gcc 的命令行参数有一大堆,有兴趣可以自己去看看。) <BR>(忍不住要加个注,现在应该用gcc 
-ggdb指定吧!因为有很多人都在问,因为除了gdb还有别的工具:-) 
<BR>反正在linux下把os.c用以上方法编译连接以后就产生了可供gdb使用的可执行文件。 <BR>我用gcc -g os.c -o 
os,产生的可执行文档叫os. <BR>然后打gdb os,就可进入gdb,屏幕提示: <BR>&nbsp;&nbsp;&nbsp;&nbsp; GDB is 
free software and you are welcome to distribute copies 
<BR>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; of it under certain conditions; type "show 
copying" to see <BR>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; the conditions. 
<BR>&nbsp;&nbsp;&nbsp;&nbsp; There is absolutely no warranty for GDB; type "show 
warranty" <BR>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; for details. 
<P>&nbsp;&nbsp;&nbsp;&nbsp; GDB 4.16, Copyright 1995 Free Software Foundation, 
Inc... <BR>&nbsp;(gdb) <BR>&nbsp; (gdb)是提示符,在这提示符下可以输入命令,直到退出。(退出命令是q/Q) 
<BR>为了尽量和原文档说明的命令相符,即使在本例子中没用的命令我也将演示。 <BR>首先我们可以设置gdb的屏幕大小。键入: 
<BR>&nbsp;(gdb)set width 70 <BR>就是把标准屏幕设为70列。 <BR>&nbsp; 
然后让我们来设置断点。设置方法很简单:break或简单打b后面加行号或函数名 <BR>比如我们可以在main 函数上设断点: 
<BR>&nbsp;(gdb)break main <BR>或(gdb)b main <BR>&nbsp;系统提示:Breakpoint 1 at 
0x8049552: file os.c, line 455. 
<BR>&nbsp;然后我们可以运行这个程序,当程序运行到main函数时程序就会停止返回到gdb的 
<BR>提示符下。运行的命令是run或r(gdb中有不少alias,可以看一下help,在gdb下打help) <BR>run 
后面可以跟参数,就是为程序指定命令行参数。 <BR>比如r abcd,则程序就会abcd以作为参数。(这里要说明的是可以用set args来指定参 
<BR>数)。打入r或run后,程序就开始运行直到进入main的入口停止,显示: <BR>Starting program: &lt;路径&gt;/os 
<P>Breakpoint 1, main () at os.c:455 
<BR>455&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
Initial(); <BR>这里455 Initial();是将要执行的命令或函数。 <BR>gdb提供两种方式:1.单步进入,step 
into就是跟踪到函数内啦。命令是step或s 
<BR>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
2.单步,next,就是简单的单步,不会进入函数。命令是next或n <BR>这两个命令还有别的用法以后再说。 <BR>我们用n命令,键入: 
<BR>(gdb)n <BR>Success forking process# 1 ,pid is 31474 
<P>Success forking process# 2 ,pid is 31475 
<P>Success forking process# 3 ,pid is 31476 
<P>Success forking process# 4 ,pid is 31477 
<P>Success forking process# 5 ,pid is 31478 
<P>Success forking process# 6 ,pid is 31479 
<P>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
Dispatching Algorithm : FIFO 
<BR>******************************************************************************** 

<P>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
PCB#&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; PID&nbsp;&nbsp;&nbsp;&nbsp; 
Priority&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
PC&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; State 
<BR>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
1&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
31474&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
24&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
0&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; WAITING 
<BR>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
2&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
31475&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
19&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
0&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; WAITING 
<BR>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
3&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
31476&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
16&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
0&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; WAITING 
<BR>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
4&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
31477&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
23&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
0&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; WAITING 
<BR>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
5&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
31478&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
22&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
0&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; WAITING 
<BR>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
6&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
31479&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
20&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
0&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; WAITING 
<P>****************************************************************************** 

<P>CPU&nbsp; :&nbsp; NO process running <BR>IO :&nbsp; No process <BR>Waiting 
CPU!!!&nbsp; 31474&nbsp;&nbsp; 31475&nbsp;&nbsp; 31476&nbsp;&nbsp; 
31477&nbsp;&nbsp; 31478&nbsp;&nbsp; 31479 <BR>Waiting&nbsp; IO&nbsp;&nbsp;&nbsp; 
NONE <BR>456&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
State=WAITING; <BR>最后的一行就是下一句要执行的命令。我们现在在另一个函数上加断点。注意我们 
<BR>可以用l/list命令来显示原代码。这里我们键入 <BR>(gdb)l <BR>451&nbsp;&nbsp;&nbsp;&nbsp; main() 
<BR>452&nbsp;&nbsp;&nbsp;&nbsp; { 
<BR>453&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
int message; <BR>454 
<BR>455&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
Initial(); 
<BR>456&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
State=WAITING; 
<BR>457&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
printf("Use Control-C to halt \n"); 
<BR>458&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
signal(SIGALRM,AlarmMessage); 
<BR>459&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
signal(SIGINT,InteruptMessage); 
<BR>460&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
signal(SIGUSR2,IoMessage); <BR>(gdb) l 
<BR>461&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
alarm(TimeSlot); 
<BR>462&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
for(;;) 
<BR>463&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
{ 
<BR>464&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
message=GetMessage(); 
<BR>465&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
switch(message) 
<BR>466&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
{ 
<BR>467&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
case INTERRUPT :&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; printf("Use Control-C 
t; 
<P>468&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
break; 
<BR>469&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
case CHILD_IO:&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
WaitingIo(); 
<BR>470&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
break; <BR>显示了原代码,现在在AlarmMessage上加断点。 <BR>(gdb) b AlarmMessage <BR>Breakpoint 2 
at 0x8048ee3: file os.c, line 259. <BR>(gdb) <BR>然后我们继续运行程序。 <BR>(gdb)c 
<BR>c或continue命令让我们继续被中断的程序。 显示: <BR>Continuing. <BR>Use Control-C to halt 
<P>Breakpoint 2, AlarmMessage () at os.c:259 
<BR>259&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
ClearSignal(); <BR>注意我们下一句语句就是ClearSignal(); <BR>我们用s/step跟踪进入这个函数看看它是干什么的。 
<BR>(gdb) s <BR>ClearSignal () at os.c:227 
<BR>227&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
signal(SIGINT,SIG_IGN); <BR>用l命令列出原代码: <BR>(gdb) l 
<BR>222&nbsp;&nbsp;&nbsp;&nbsp; } <BR>223 <BR>224 
<BR>225&nbsp;&nbsp;&nbsp;&nbsp; void ClearSignal()&nbsp;&nbsp;&nbsp; /* Clear 
other signals */ <BR>226&nbsp;&nbsp;&nbsp;&nbsp; { 
<BR>227&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
signal(SIGINT,SIG_IGN); 
<BR>228&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
signal(SIGALRM,SIG_IGN); 
<BR>229&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; 
signal(SIGUSR2,SIG_IGN); <BR>230&nbsp;&nbsp;&nbsp;&nbsp; } <BR>231 <BR>(gdb) 
<BR>我们可以用s命令继续跟踪。现在让我们来试试bt或backtrace命令。这个命令可以 <BR>显示栈中的内容。 <BR>(gdb) bt 
<BR>#0&nbsp; ClearSignal () at os.c:227 <BR>#1&nbsp; 0x8048ee8 in AlarmMessage 
() at os.c:259 <BR>#2&nbsp; 0xbffffaec in ?? () <BR>#3&nbsp; 0x80486ae in 
___crt_dummy__ () <BR>(gdb) <BR>大家一定能看懂显示的意思。栈顶是AlarmMessage,接下来的函数没有名字--就是 
<BR>没有原代码符号。这显示了函数调用的嵌套。 <BR>好了,我们跟踪了半天还没有检查过变量的值呢。检查表达式的值的命令是p或print <BR>格式是p 
&lt;表达式&gt; <BR>444444让我们来找一个变量来看看。:-) <BR>(gdb)l 1 
<BR>还记得l的作用吗?l或list显示原代码符号,l或list加&lt;行号&gt;就显示从&lt;行号&gt;开始的 
<BR>原代码。好了找到一个让我们来看看WaitingQueue的内容 <BR>(gdb) p WaitingQueue <BR>$1 = {1, 2, 3, 
4, 5, 6, 0} <BR>(gdb) <BR>WaitingQueue是一个数组,gdb还支持结构的显示, <BR>(gdb) p Pcb <BR>$2 
= {{Pid = 0, State = 0, Prior = 0, pc = 0}, {Pid = 31474, State = 2, 
<BR>&nbsp;&nbsp;&nbsp; Prior = 24, pc = 0}, {Pid = 31475, State = 2, Prior = 19, 
pc = 0}, { <BR>&nbsp;&nbsp;&nbsp; Pid = 31476, State = 2, Prior = 16, pc = 0}, 
{Pid = 31477, State = 2, <BR>&nbsp;&nbsp;&nbsp; Prior = 23, pc = 0}, {Pid = 
31478, State = 2, Prior = 22, pc = 0}, { <BR>&nbsp;&nbsp;&nbsp; Pid = 31479, 
State = 2, Prior = 20, pc = 0}} <BR>(gdb) <BR>这里可以对照原程序看看。 
<BR>原文档里是一个调试过程,不过我想这里我已经把gdb的常用功能介绍了一遍,基本上 <BR>可以用来调试程序了。:-) 
<P>-- <BR>※ 来源: 中国科大BBS站 [bbs.ustc.edu.cn] <BR>&nbsp; <BR>&nbsp; 
<P>-- <BR>※ 来源:·BBS 水木调试站 Leeward.lib.tsinghua.edu.cn·[FROM: 202.200.37.100] 
<CENTER>
<H1>BBS水木清华站∶精华区</H1></CENTER></BODY></HTML>

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -