BBS水木清华站∶精华区

发信人: TJB (老六), 信区: Linux 
标  题: GDB (2) 
发信站: BBS 水木调试站 (Tue Jun  2 15:58:26 1998) 
 
发 信 人:System_Killer(大家一起来发呆)           信区名称:Linux[4614] 
信件提要:gdb(二) 
原发信站:中国科大BBS站(Sat, 28 Mar 1998 22:28:43) 
 
这里是GDB的一个例子: 
        原文中是使用一个叫m4的程序。但很遗憾我找不到这个程序的原代码, 
所以没有办法来按照原文来说明。不过反正是个例子,我就拿一个操作系统的 
进程调度原码来说明把,原代码我会附在后面。 
        首先这个程序叫os.c是一个模拟进程调度的原程序(也许是个老古董了:-))。 
先说明一下如何取得包括原代码符号的可执行代码。大家有心的话可以去看一下gcc的 
man文件(在shell下打man gcc)。gcc -g <原文件.c> -o <要生成的文件名> 
-g 的意思是生成带原代码调试符号的可执行文件。 
-o 的意思是指定可执行文件名。 
(gcc 的命令行参数有一大堆,有兴趣可以自己去看看。) 
反正在linux下把os.c用以上方法编译连接以后就产生了可供gdb使用的可执行文件。 
我用gcc -g os.c -o os,产生的可执行文档叫os. 
然后打gdb os,就可进入gdb,屏幕提示: 
     GDB is free software and you are welcome to 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. 
 
     GDB 4.16, Copyright 1995 Free Software Foundation, Inc... 
 (gdb) 
  (gdb)是提示符,在这提示符下可以输入命令,直到退出。(退出命令是q/Q) 
为了尽量和原文档说明的命令相符,即使在本例子中没用的命令我也将演示。 
首先我们可以设置gdb的屏幕大小。键入: 
 (gdb)set width 70 
就是把标准屏幕设为70列。 
  然后让我们来设置断点。设置方法很简单:break或简单打b后面加行号或函数名 
比如我们可以在main 函数上设断点: 
 (gdb)break main 
或(gdb)b main 
 系统提示:Breakpoint 1 at 0x8049552: file os.c, line 455. 
 然后我们可以运行这个程序,当程序运行到main函数时程序就会停止返回到gdb的 
提示符下。运行的命令是run或r(gdb中有不少alias,可以看一下help,在gdb下打help) 
run 后面可以跟参数,就是为程序指定命令行参数。 
比如r abcd,则程序就会abcd以作为参数。(这里要说明的是可以用set args来指定参 
数)。打入r或run后,程序就开始运行直到进入main的入口停止,显示: 
Starting program: <路径>/os 
 
Breakpoint 1, main () at os.c:455 
455            Initial(); 
这里455 Initial();是将要执行的命令或函数。 
gdb提供两种方式:1.单步进入,step into就是跟踪到函数内啦。命令是step或s 
                 2.单步,next,就是简单的单步,不会进入函数。命令是next或n 
这两个命令还有别的用法以后再说。 
我们用n命令,键入: 
(gdb)n 
Success forking process# 1 ,pid is 31474 
 
Success forking process# 2 ,pid is 31475 
 
Success forking process# 3 ,pid is 31476 
 
Success forking process# 4 ,pid is 31477 
 
Success forking process# 5 ,pid is 31478 
 
Success forking process# 6 ,pid is 31479 
 
                Dispatching Algorithm : FIFO 
******************************************************************************** 
 
            PCB#        PID     Priority        PC      State 
            1           31474      24            0      WAITING 
            2           31475      19            0      WAITING 
            3           31476      16            0      WAITING 
            4           31477      23            0      WAITING 
            5           31478      22            0      WAITING 
            6           31479      20            0      WAITING 
 
****************************************************************************** 
 
CPU  :  NO process running 
IO :  No process 
Waiting CPU!!!  31474   31475   31476   31477   31478   31479 
Waiting  IO    NONE 
456            State=WAITING; 
最后的一行就是下一句要执行的命令。我们现在在另一个函数上加断点。注意我们 
可以用l/list命令来显示原代码。这里我们键入 
(gdb)l 
451     main() 
452     { 
453             int message; 
454 
455            Initial(); 
456            State=WAITING; 
457            printf("Use Control-C to halt \n"); 
458            signal(SIGALRM,AlarmMessage); 
459            signal(SIGINT,InteruptMessage); 
460            signal(SIGUSR2,IoMessage); 
(gdb) l 
461            alarm(TimeSlot); 
462            for(;;) 
463             { 
464             message=GetMessage(); 
465                   switch(message) 
466                     { 
467                             case INTERRUPT :        printf("Use Control-C t; 
 
468                                                     break; 
469                             case CHILD_IO:          WaitingIo(); 
470                                                     break; 
显示了原代码,现在在AlarmMessage上加断点。 
(gdb) b AlarmMessage 
Breakpoint 2 at 0x8048ee3: file os.c, line 259. 
(gdb) 
然后我们继续运行程序。 
(gdb)c 
c或continue命令让我们继续被中断的程序。 显示: 
Continuing. 
Use Control-C to halt 
 
Breakpoint 2, AlarmMessage () at os.c:259 
259             ClearSignal(); 
注意我们下一句语句就是ClearSignal(); 
我们用s/step跟踪进入这个函数看看它是干什么的。 
(gdb) s 
ClearSignal () at os.c:227 
227             signal(SIGINT,SIG_IGN); 
用l命令列出原代码: 
(gdb) l 
222     } 
223 
224 
225     void ClearSignal()    /* Clear other signals */ 
226     { 
227             signal(SIGINT,SIG_IGN); 
228             signal(SIGALRM,SIG_IGN); 
229             signal(SIGUSR2,SIG_IGN); 
230     } 
231 
(gdb) 
我们可以用s命令继续跟踪。现在让我们来试试bt或backtrace命令。这个命令可以 
显示栈中的内容。 
(gdb) bt 
#0  ClearSignal () at os.c:227 
#1  0x8048ee8 in AlarmMessage () at os.c:259 
#2  0xbffffaec in ?? () 
#3  0x80486ae in ___crt_dummy__ () 
(gdb) 
大家一定能看懂显示的意思。栈顶是AlarmMessage,接下来的函数没有名字--就是 
没有原代码符号。这显示了函数调用的嵌套。 
好了,我们跟踪了半天还没有检查过变量的值呢。检查表达式的值的命令是p或print 
格式是p <表达式> 
444444让我们来找一个变量来看看。:-) 
(gdb)l 1 
还记得l的作用吗?l或list显示原代码符号,l或list加<行号>就显示从<行号>开始的 
原代码。好了找到一个让我们来看看WaitingQueue的内容 
(gdb) p WaitingQueue 
$1 = {1, 2, 3, 4, 5, 6, 0} 
(gdb) 
WaitingQueue是一个数组,gdb还支持结构的显示, 
(gdb) p Pcb 
$2 = {{Pid = 0, State = 0, Prior = 0, pc = 0}, {Pid = 31474, State = 2, 
    Prior = 24, pc = 0}, {Pid = 31475, State = 2, Prior = 19, pc = 0}, { 
    Pid = 31476, State = 2, Prior = 16, pc = 0}, {Pid = 31477, State = 2, 
    Prior = 23, pc = 0}, {Pid = 31478, State = 2, Prior = 22, pc = 0}, { 
    Pid = 31479, State = 2, Prior = 20, pc = 0}} 
(gdb) 
这里可以对照原程序看看。 
原文档里是一个调试过程,不过我想这里我已经把gdb的常用功能介绍了一遍,基本上 
可以用来调试程序了。:-) 
 
-- 
※ 来源: 中国科大BBS站 [bbs.ustc.edu.cn] 
 
 
 
-- 
※ 来源:·BBS 水木调试站 Leeward.lib.tsinghua.edu.cn·[FROM: 202.200.37.100] 

BBS水木清华站∶精华区