簡述
一 列文件清單
二:執(zhí)行程序
三:顯示數(shù)據(jù)
四:斷點(breakpoint)
五.?dāng)帱c的管理
六.變量的檢查和賦值
七. 單步執(zhí)行
八.函數(shù)的調(diào)用
九.機器語言工具
十.信號
GDB的使用方法
簡述
一 列文件清單
* List
(gdb) list line1,line2
二:執(zhí)行程序
要想運行準(zhǔn)備調(diào)試的程序,可使用run命令,在它后面可以跟隨發(fā)給該程序的任何參數(shù),包括標(biāo)準(zhǔn)輸入和標(biāo)準(zhǔn)輸出說明符(<和> )和外殼通配符(*、?、[、])在內(nèi)。如果你使用不帶參數(shù)的run命令,gdb就再次使用你給予前一條run命令的參數(shù),這是很有用的。利用set args 命令就可以修改發(fā)送給程序的參數(shù),而使用show args 命令就可以查看其缺省參數(shù)的列表。
(gdb)set args –b –x
(gdb) show args
backtrace命令為堆棧提供向后跟蹤功能。
Backtrace 命令產(chǎn)生一張列表,包含著從最近的過程開始的所以有效過程和調(diào)用這些過程的參數(shù)。
三:顯示數(shù)據(jù)
* 利用print 命令可以檢查各個變量的值。
(gdb) print p (p為變量名)
print 是gdb的一個功能很強的命令,利用它可以顯示被調(diào)試的語言中任何有效的表達(dá)式。表達(dá)式除了包含你程序中的變量外,還可以包含以下內(nèi)容:
1. 對程序中函數(shù)的調(diào)用
(gdb) print find_entry(1,0)
2. 數(shù)據(jù)結(jié)構(gòu)和其他復(fù)雜對象
(gdb) print *table_start
$8={e=reference=’\000’,location=0x0,next=0x0}
3. 值的歷史成分
(gdb)print $1 ($1為歷史記錄變量,在以后可以直接引用 $1 的值)
4. 人為數(shù)組
人為數(shù)組提供了一種去顯示存儲器塊(數(shù)組節(jié)或動態(tài)分配的存儲區(qū))內(nèi)容的方法。早期的調(diào)試程序沒有很好的方法將任意的指針換成一個數(shù)組。就像對待參數(shù)一樣,讓我們查看內(nèi)存中在變量h后面的10個整數(shù),一個動態(tài)數(shù)組的語法如下所示:
base@length
因此,要想顯示在h后面的10個元素,可以使用h@10:
(gdb)print h@10
$13=(-1,345,23,-234,0,0,0,98,345,10)
* whatis 命令可以顯示某個變量的類型
(gdb) whatis p
type = int *
四:斷點(breakpoint)
break命令(可以簡寫為b)可以用來在調(diào)試的程序中設(shè)置斷點,該命令有如下四種形式:
* break line-number 使程序恰好在執(zhí)行給定行之前停止。
* break function-name 使程序恰好在進(jìn)入指定的函數(shù)之前停止。
* break line-or-function if condition 如果condition(條件)是真,程序到達(dá)指定行或函數(shù)時停止。
* break routine-name 在指定例程的入口處設(shè)置斷點
如果該程序是由很多原文件構(gòu)成的,你可以在各個原文件中設(shè)置斷點,而不是在當(dāng)前的原文件中設(shè)置斷點,其方法如下:
(gdb) break filename:line-number
(gdb) break filename:function-name
要想設(shè)置一個條件斷點,可以利用break if命令,如下所示:
(gdb) break line-or-function if expr
例:
(gdb) break 46 if testsize==100
從斷點繼續(xù)運行:countinue 命令
五.?dāng)帱c的管理
1.顯示當(dāng)前gdb的斷點信息:
(gdb) info break
他會以如下的形式顯示所有的斷點信息:
Num Type Disp Enb Address What
1 breakpoint keep y 0x000028bc in init_random at qsort2.c:155
2 breakpoint keep y 0x0000291c in init_organ at qsort2.c:168
1. 刪除指定的某個斷點:
(gdb) delete breakpoint 1
該命令將會刪除編號為1的斷點,如果不帶編號參數(shù),將刪除所有的斷點
(gdb) delete breakpoint
2. 禁止使用某個斷點
(gdb) disable breakpoint 1
該命令將禁止斷點 1,同時斷點信息的 (Enb)域?qū)⒆優(yōu)?n
3. 允許使用某個斷點
(gdb) enable breakpoint 1
該命令將允許斷點 1,同時斷點信息的 (Enb)域?qū)⒆優(yōu)?y
4. 清除原文件中某一代碼行上的所有斷點
(gdb)clean number
注:number 為原文件的某個代碼行的行號
六.變量的檢查和賦值
* whatis:識別數(shù)組或變量的類型
* ptype:比whatis的功能更強,他可以提供一個結(jié)構(gòu)的定義
* set variable:將值賦予變量
* print 除了顯示一個變量的值外,還可以用來賦值
七. 單步執(zhí)行
* next 不進(jìn)入的單步執(zhí)行
* step 進(jìn)入的單步執(zhí)行如果已經(jīng)進(jìn)入了某函數(shù),而想退出該函數(shù)返回到它的調(diào)用函數(shù)中,可使用命令finish
八.函數(shù)的調(diào)用
* call name 調(diào)用和執(zhí)行一個函數(shù)
(gdb) call gen_and_sork( 1234,1,0 )
(gdb) call printf(“abcd”)
$1=4
* finish 結(jié)束執(zhí)行當(dāng)前函數(shù),顯示其返回值(如果有的話)
九.機器語言工具
有一組專用的gdb變量可以用來檢查和修改計算機的通用寄存器,gdb提供了目前每一臺計算機中實際使用的4個寄存器的標(biāo)準(zhǔn)名字:
* $pc : 程序計數(shù)器
* $fp : 幀指針(當(dāng)前堆棧幀)
* $sp : 棧指針
* $ps : 處理器狀態(tài)
十.信號
gdb 通??梢圆蹲降桨l(fā)送給它的大多數(shù)信號,通過捕捉信號,它就可決定對于正在運行的進(jìn)程要做些什么工作。例如,按CTRL-C將中斷信號發(fā)送給gdb,通常就會終止gdb。但是你或許不想中斷gdb,真正的目的是要中斷gdb正在運行的程序,因此,gdb要抓住該信號并停止它正在運行的程序,這樣就可以執(zhí)行某些調(diào)試操作。
Handle命令可控制信號的處理,他有兩個參數(shù),一個是信號名,另一個是接受到信號時該作什么。幾種可能的參數(shù)是:
* nostop 接收到信號時,不要將它發(fā)送給程序,也不要停止程序。
* stop 接受到信號時停止程序的執(zhí)行,從而允許程序調(diào)試;顯示一條表示已接受到信號的消息(禁止使用消息除外)
* print 接受到信號時顯示一條消息
* noprint 接受到信號時不要顯示消息(而且隱含著不停止程序運行)
* pass 將信號發(fā)送給程序,從而允許你的程序去處理它、停止運行或采取別的動作。
* nopass 停止程序運行,但不要將信號發(fā)送給程序。
例如,假定你截獲SIGPIPE信號,以防止正在調(diào)試的程序接受到該信號,而且只要該信號一到達(dá),就要求該程序停止,并通知你。要完成這一任務(wù),可利用如下命令:
(gdb) handle SIGPIPE stop print
請注意,UNIX的信號名總是采用大寫字母!你可以用信號編號替代信號名如果你的程序要執(zhí)行任何信號處理操作,就需要能夠測試其信號處理程序,為此,就需要一種能將信號發(fā)送給程序的簡便方法,這就是signal命令的任務(wù)。該命令的參數(shù)是一個數(shù)字或者一個名字,如SIGINT。假定你的程序已將一個專用的 SIGINT(鍵盤輸入,或CTRL-C;信號2)信號處理程序設(shè)置成采取某個清理動作,要想測試該信號處理程序,你可以設(shè)置一個斷點并使用如下命令:
(gdb) signal 2
continuing with signal SIGINT(2)
該程序繼續(xù)執(zhí)行,但是立即傳輸該信號,而且處理程序開始運行.
GDB的使用方法
GDB是一個強大的命令行調(diào)試工具。大家知道命令行的強大就是在于,其可以形成
執(zhí)行序列,形成腳本。UNIX下的軟件全是命令行的,這給程序開發(fā)提代供了極大的
便利,命令行軟件的優(yōu)勢在于,它們可以非常容易的集成在一起,使用幾個簡單的已
有工具的命令,就可以做出一個非常強大的功能。
于是UNIX下的軟件比Windows下的軟件更能有機地結(jié)合,各自發(fā)揮各自的長處,組合
成更為強勁的功能。而Windows下的圖形軟件基本上是各自為營,互相不能調(diào)用,很
不利于各種軟件的相互集成。在這里并不是要和Windows做個什么比較,所謂“寸有
所長,尺有所短”,圖形化工具還是有不如命令行的地方。
用GDB調(diào)試程序
GDB概述
————
GDB是GNU開源組織發(fā)布的一個強大的UNIX下的程序調(diào)試工具。或許,各位比較喜歡
那種圖形界面方式的,像VC、BCB等IDE的調(diào)試,但如果你是在UNIX平臺下做軟件,你
會發(fā)現(xiàn)GDB這個調(diào)試工具有比VC、BCB的圖形化調(diào)試器更強大的功能。
所謂“寸有所
長,尺有所短”就是這個道理。
一般來說,GDB主要幫忙你完成下面四個方面的功能:
1、啟動你的程序,可以按照你的自定義的要求隨心所欲的運行程序。
2、可讓被調(diào)試的程序在你所指定的調(diào)置的斷點處停住。(斷點可以是條件表達(dá)式)
3、當(dāng)程序被停住時,可以檢查此時你的程序中所發(fā)生的事。
4、動態(tài)的改變你程序的執(zhí)行環(huán)境。
從上面看來,GDB和一般的調(diào)試工具沒有什么兩樣,基本上也是完成這些功能,不過
在細(xì)節(jié)上,你會發(fā)現(xiàn)GDB這個調(diào)試工具的強大,大家可能比較習(xí)慣了圖形化的調(diào)試工
具,但有時候,命令行的調(diào)試工具卻有著圖形化工具所不能完成的功能。讓我們一
一看來。
一個調(diào)試示例
——————
源程序:tst.c
1 #include
2
3 int func(int n)
4 {
5 int sum=0,i;
6 for(i=0; i
7 {
8 sum+=i;
9 }
10 return sum;
11 }
12
13
14 main()
15 {
16 int i;
17 long result = 0;
18 for(i=1; i<=100; i++)
19 {
20 result += i;
21 }
22
23 printf("result[1-100] = %d \n", result );
24 printf("result[1-250] = %d \n", func(250) );
25 }
編譯生成執(zhí)行文件:(Linux下)
hchen/test> cc -g tst.c -o tst
使用GDB調(diào)試:
hchen/test> gdb tst <---------- 啟動GDB
GNU gdb 5.1.1
Copyright 2002 Free Software Foundation, Inc.
GDB is free software, covered by the GNU General Public License, and you are
welcome to change it and/or 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.
This GDB was configured as "i386-suse-linux"...
(gdb) l <-------------------- l命令相當(dāng)于list,從第一行開始例出原碼。
1 #include
2
3 int func(int n)
4 {
5 int sum=0,i;
6 for(i=0; i
7 {
8 sum+=i;
9 }
10 return sum;
(gdb) <-------------------- 直接回車表示,重復(fù)上一次命令
11 }
12
13
14 main()
15 {
16 int i;
17 long result = 0;
18 for(i=1; i<=100; i++)
19 {
20 result += i;
(gdb) break 16 <-------------------- 設(shè)置斷點,在源程序第16行處。
Breakpoint 1 at 0x8048496: file tst.c, line 16.
(gdb) break func <-------------------- 設(shè)置斷點,在函數(shù)func()入口處。
Breakpoint 2 at 0x8048456: file tst.c, line 5.
(gdb) info break <-------------------- 查看斷點信息。
Num Type Disp Enb Address What
1 breakpoint keep y 0x08048496 in main at tst.c:16
2 breakpoint keep y 0x08048456 in func at tst.c:5
(gdb) r <--------------------- 運行程序,run命令簡寫
Starting program: /home/hchen/test/tst
Breakpoint 1, main () at tst.c:17 <---------- 在斷點處停住。
17 long result = 0;
(gdb) n <--------------------- 單條語句執(zhí)行,next命令簡寫。
18 for(i=1; i<=100; i++)
(gdb) n
20 result += i;
(gdb) n
18 for(i=1; i<=100; i++)
(gdb) n
20 result += i;
(gdb) c <--------------------- 繼續(xù)運行程序,continue命令簡寫。
Continuing.
result[1-100] = 5050 <----------程序輸出。
Breakpoint 2, func (n=250) at tst.c:5
5 int sum=0,i;
(gdb) n
6 for(i=1; i<=n; i++)
(gdb) p i <--------------------- 打印變量i的值,print命令簡寫。
$1 = 134513808
(gdb) n
8 sum+=i;
(gdb) n
6 for(i=1; i<=n; i++)
(gdb) p sum
$2 = 1
(gdb) n
8 sum+=i;
(gdb) p i
$3 = 2
(gdb) n
6 for(i=1; i<=n; i++)
(gdb) p sum
$4 = 3
(gdb) bt <--------------------- 查看函數(shù)堆棧。
#0 func (n=250) at tst.c:5
#1 0x080484e4 in main () at tst.c:24
#2 0x400409ed in __libc_start_main () from /lib/libc.so.6
(gdb) finish <--------------------- 退出函數(shù)。
Run till exit from #0 func (n=250) at tst.c:5
0x080484e4 in main () at tst.c:24
24 printf("result[1-250] = %d \n", func(250) );
Value returned is $6 = 31375
(gdb) c <--------------------- 繼續(xù)運行。
Continuing.
result[1-250] = 31375 <----------程序輸出。
Program exited with code 027. <--------程序退出,調(diào)試結(jié)束。
(gdb) q <--------------------- 退出gdb。
hchen/test>
好了,有了以上的感性認(rèn)識,還是讓我們來系統(tǒng)地認(rèn)識一下gdb吧。
使用GDB
————
一般來說GDB主要調(diào)試的是C/C++的程序。要調(diào)試C/C++的程序,首先在編譯時,我們
必須要把調(diào)試信息加到可執(zhí)行文件中。使用編譯器(cc/gcc/g++)的 -g 參數(shù)可以
做到這一點。如:
> cc -g hello.c -o hello
> g++ -g hello.cpp -o hello
如果沒有-g,你將看不見程序的函數(shù)名、變量名,所代替的全是運行時的內(nèi)存地址。
當(dāng)你用-g把調(diào)試信息加入之后,并成功編譯目標(biāo)代碼以后,讓我們來看看如何用
gdb來調(diào)試他。
啟動GDB的方法有以下幾種:
1、gdb
program也就是你的執(zhí)行文件,一般在當(dāng)然目錄下。
2、gdb core
用gdb同時調(diào)試一個運行程序和core文件,core是程序非法執(zhí)行后core dump后產(chǎn)生的文件。
3、gdb
如果你的程序是一個服務(wù)程序,那么你可以指定這個服務(wù)程序運行時的進(jìn)程ID。
gdb會自動attach上去,并調(diào)試他。program應(yīng)該在PATH環(huán)境變量中搜索得到。
GDB啟動時,可以加上一些GDB的啟動開關(guān),詳細(xì)的開關(guān)可以用gdb -help查看。我在下面只例舉一些比較常用的參數(shù):
-symbols
-s
從指定文件中讀取符號表。
-se file
從指定文件中讀取符號表信息,并把他用在可執(zhí)行文件中。
-core
-c
調(diào)試時core dump的core文件。
-directory
-d
加入一個源文件的搜索路徑。默認(rèn)搜索路徑是環(huán)境變量中PATH所定義的路徑。
GDB的命令概貌
———————
啟動gdb后,就你被帶入gdb的調(diào)試環(huán)境中,就可以使用gdb的命令開始調(diào)試程序了,gdb的命令可以使用help命令來查看,如下所示:
/home/hchen> gdb
GNU gdb 5.1.1
Copyright 2002 Free Software Foundation, Inc.
GDB is free software, covered by the GNU General Public License, and you are
welcome to change it and/or 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.
This GDB was configured as "i386-suse-linux".
(gdb) help
List of classes of commands:
aliases -- Aliases of other commands
breakpoints -- Making program stop at certain points
data -- Examining data
files -- Specifying and examining files
internals -- Maintenance commands
obscure -- Obscure features
running -- Running the program
stack -- Examining the stack
status -- Status inquiries
support -- Support facilities
tracepoints -- Tracing of program execution without stopping the program
user-defined -- User-defined commands
Type "help" followed by a class name for a list of commands in that class.
Type "help" followed by command name for full documentation.
Command name abbreviations are allowed if unambiguous.
(gdb)
gdb的命令很多,gdb把之分成許多個種類。help命令只是例出gdb的命令種類,如果
要看種類中的命令,可以使用help 命令,如:help breakpoints,查看設(shè)置斷點的所
有命令。也可以直接help 來查看命令的幫助。
gdb中,輸入命令時,可以不用打全命令,只用打命令的前幾個字符就可以了,當(dāng)然,
命令的前幾個字符應(yīng)該要標(biāo)志著一個唯一的命令,在Linux下,你可以敲擊兩次TAB
鍵來補齊命令的全稱,如果有重復(fù)的,那么gdb會把其例出來。
示例一:在進(jìn)入函數(shù)func時,設(shè)置一個斷點??梢郧萌隻reak func,