📄 program-gdb-2.html
字号:
<!doctype html public "-//w3c//dtd html 4.0 transitional//en">
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=gb2312">
<meta name="GENERATOR" content="Mozilla/4.05 [zh-CN] (X11; I; Linux 2.1.127 i686) [Netscape]">
<title>BBS水木清华站∶精华区</title>
</head>
<body>
<center>
<h1>
BBS水木清华站∶精华区</h1></center>
发信人: TJB (老六), 信区: Linux
<br>标 题: GDB (2)
<br>发信站: BBS 水木调试站 (Tue Jun 2 15:58:26 1998)
<p>发 信 人:System_Killer(大家一起来发呆)
信区名称:Linux[4614]
<br>信件提要:gdb(二)
<br>原发信站:中国科大BBS站(Sat, 28 Mar 1998 22:28:43)
<p>这里是GDB的一个例子:
<br> 原文中是使用一个叫m4的程序。但很遗憾我找不到这个程序的原代码,
<br>所以没有办法来按照原文来说明。不过反正是个例子,我就拿一个操作系统的
<br>进程调度原码来说明把,原代码我会附在后面。
<br> 首先这个程序叫os.c是一个模拟进程调度的原程序(也许是个老古董了:-))。
<br>先说明一下如何取得包括原代码符号的可执行代码。大家有心的话可以去看一下gcc的
<br>man文件(在shell下打man gcc)。gcc -g <原文件.c> -o <要生成的文件名>
<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> GDB is free software and you are welcome to
distribute copies
<br> of it under certain conditions; type
"show copying" to see
<br> the conditions.
<br> There is absolutely no warranty for GDB; type
"show warranty"
<br> for details.
<p> GDB 4.16, Copyright 1995 Free Software Foundation,
Inc...
<br> (gdb)
<br> (gdb)是提示符,在这提示符下可以输入命令,直到退出。(退出命令是q/Q)
<br>为了尽量和原文档说明的命令相符,即使在本例子中没用的命令我也将演示。
<br>首先我们可以设置gdb的屏幕大小。键入:
<br> (gdb)set width 70
<br>就是把标准屏幕设为70列。
<br> 然后让我们来设置断点。设置方法很简单:break或简单打b后面加行号或函数名
<br>比如我们可以在main 函数上设断点:
<br> (gdb)break main
<br>或(gdb)b main
<br> 系统提示:Breakpoint 1 at 0x8049552: file os.c, line 455.
<br> 然后我们可以运行这个程序,当程序运行到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: <路径>/os
<p>Breakpoint 1, main () at os.c:455
<br>455
Initial();
<br>这里455 Initial();是将要执行的命令或函数。
<br>gdb提供两种方式:1.单步进入,step into就是跟踪到函数内啦。命令是step或s
<br>
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>
Dispatching Algorithm : FIFO
<br>********************************************************************************
<p> PCB#
PID Priority
PC State
<br>
1 31474
24 0
WAITING
<br>
2 31475
19 0
WAITING
<br>
3 31476
16 0
WAITING
<br>
4 31477
23 0
WAITING
<br>
5 31478
22 0
WAITING
<br>
6 31479
20 0
WAITING
<p>******************************************************************************
<p>CPU : NO process running
<br>IO : No process
<br>Waiting CPU!!! 31474 31475 31476
31477 31478 31479
<br>Waiting IO NONE
<br>456
State=WAITING;
<br>最后的一行就是下一句要执行的命令。我们现在在另一个函数上加断点。注意我们
<br>可以用l/list命令来显示原代码。这里我们键入
<br>(gdb)l
<br>451 main()
<br>452 {
<br>453
int message;
<br>454
<br>455
Initial();
<br>456
State=WAITING;
<br>457
printf("Use Control-C to halt \n");
<br>458
signal(SIGALRM,AlarmMessage);
<br>459
signal(SIGINT,InteruptMessage);
<br>460
signal(SIGUSR2,IoMessage);
<br>(gdb) l
<br>461
alarm(TimeSlot);
<br>462
for(;;)
<br>463
{
<br>464
message=GetMessage();
<br>465
switch(message)
<br>466
{
<br>467
case INTERRUPT : printf("Use
Control-C t;
<p>468
break;
<br>469
case CHILD_IO: WaitingIo();
<br>470
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
ClearSignal();
<br>注意我们下一句语句就是ClearSignal();
<br>我们用s/step跟踪进入这个函数看看它是干什么的。
<br>(gdb) s
<br>ClearSignal () at os.c:227
<br>227
signal(SIGINT,SIG_IGN);
<br>用l命令列出原代码:
<br>(gdb) l
<br>222 }
<br>223
<br>224
<br>225 void ClearSignal() /*
Clear other signals */
<br>226 {
<br>227
signal(SIGINT,SIG_IGN);
<br>228
signal(SIGALRM,SIG_IGN);
<br>229
signal(SIGUSR2,SIG_IGN);
<br>230 }
<br>231
<br>(gdb)
<br>我们可以用s命令继续跟踪。现在让我们来试试bt或backtrace命令。这个命令可以
<br>显示栈中的内容。
<br>(gdb) bt
<br>#0 ClearSignal () at os.c:227
<br>#1 0x8048ee8 in AlarmMessage () at os.c:259
<br>#2 0xbffffaec in ?? ()
<br>#3 0x80486ae in ___crt_dummy__ ()
<br>(gdb)
<br>大家一定能看懂显示的意思。栈顶是AlarmMessage,接下来的函数没有名字--就是
<br>没有原代码符号。这显示了函数调用的嵌套。
<br>好了,我们跟踪了半天还没有检查过变量的值呢。检查表达式的值的命令是p或print
<br>格式是p <表达式>
<br>444444让我们来找一个变量来看看。:-)
<br>(gdb)l 1
<br>还记得l的作用吗?l或list显示原代码符号,l或list加<行号>就显示从<行号>开始的
<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> Prior = 24, pc = 0}, {Pid = 31475, State = 2, Prior
= 19, pc = 0}, {
<br> Pid = 31476, State = 2, Prior = 16, pc = 0}, {Pid
= 31477, State = 2,
<br> Prior = 23, pc = 0}, {Pid = 31478, State = 2, Prior
= 22, pc = 0}, {
<br> Pid = 31479, State = 2, Prior = 20, pc = 0}}
<br>(gdb)
<br>这里可以对照原程序看看。
<br>原文档里是一个调试过程,不过我想这里我已经把gdb的常用功能介绍了一遍,基本上
<br>可以用来调试程序了。:-)
<p>--
<br>※ 来源: 中国科大BBS站 [bbs.ustc.edu.cn]
<br>
<br>
<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 + -