七、八年前寫(xiě)過(guò)一篇《用GDB調(diào)試程序》,于是,從那以后,很多朋友在MSN上以及給我發(fā)郵件詢(xún)問(wèn)我關(guān)于GDB的問(wèn)題,一直到今天,還有人在問(wèn)GDB的相關(guān)問(wèn)題。這么多年來(lái),有一些問(wèn)題是大家反復(fù)在問(wèn)的,一方面,我覺(jué)得我以前的文章可能沒(méi)有說(shuō)清楚,另一方面,我覺(jué)得大家常問(wèn)的問(wèn)題正是最有用的,所以,在這里羅列出來(lái)。希望大家補(bǔ)充。
一、多線(xiàn)程調(diào)試
多線(xiàn)程調(diào)試可能是問(wèn)得最多的。其實(shí),重要就是下面幾個(gè)命令:
- info thread 查看當(dāng)前進(jìn)程的線(xiàn)程。
- thread <ID> 切換調(diào)試的線(xiàn)程為指定ID的線(xiàn)程。
- break file.c:100 thread all 在file.c文件第100行處為所有經(jīng)過(guò)這里的線(xiàn)程設(shè)置斷點(diǎn)。
- set scheduler-locking off|on|step,這個(gè)是問(wèn)得最多的。在使用step或者continue命令調(diào)試當(dāng)前被調(diào)試線(xiàn)程的時(shí)候,其他線(xiàn)程也是同時(shí)執(zhí)行的,怎么只讓被調(diào)試程序執(zhí)行呢?通過(guò)這個(gè)命令就可以實(shí)現(xiàn)這個(gè)需求。
- off 不鎖定任何線(xiàn)程,也就是所有線(xiàn)程都執(zhí)行,這是默認(rèn)值。
- on 只有當(dāng)前被調(diào)試程序會(huì)執(zhí)行。
- step 在單步的時(shí)候,除了next過(guò)一個(gè)函數(shù)的情況(熟悉情況的人可能知道,這其實(shí)是一個(gè)設(shè)置斷點(diǎn)然后continue的行為)以外,只有當(dāng)前線(xiàn)程會(huì)執(zhí)行。
二、調(diào)試宏
這個(gè)問(wèn)題超多。在GDB下,我們無(wú)法print宏定義,因?yàn)楹晔穷A(yù)編譯的。但是我們還是有辦法來(lái)調(diào)試宏,這個(gè)需要GCC的配合。
在GCC編譯程序的時(shí)候,加上-ggdb3參數(shù),這樣,你就可以調(diào)試宏了。
另外,你可以使用下述的GDB的宏調(diào)試命令 來(lái)查看相關(guān)的宏。
- info macro – 你可以查看這個(gè)宏在哪些文件里被引用了,以及宏定義是什么樣的。
- macro – 你可以查看宏展開(kāi)的樣子。
三、源文件
這個(gè)問(wèn)題問(wèn)的也是很多的,太多的朋友都說(shuō)找不到源文件。在這里我想提醒大家做下面的檢查:
- 編譯程序員是否加上了-g參數(shù)以包含debug信息。
- 路徑是否設(shè)置正確了。使用GDB的directory命令來(lái)設(shè)置源文件的目錄。
下面給一個(gè)調(diào)試/bin/ls的示例(ubuntu下)
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 | $ apt-get source coreutils
$ sudo apt-get install coreutils-dbgsym
$ gdb /bin/ls
GNU gdb (GDB) 7.1-ubuntu
(gdb) list main
1192 ls .c: No such file or directory.
in ls .c
(gdb) directory ~ /src/coreutils-7 .4 /src/
Source directories searched: /home/hchen/src/coreutils-7 .4:$cdir:$cwd
(gdb) list main
1192 }
1193 }
1194
1195 int
1196 main (int argc, char **argv)
1197 {
1198 int i;
1199 struct pending *thispend;
1200 int n_files;
1201
|
四、條件斷點(diǎn)
條件斷點(diǎn)是語(yǔ)法是:break [where] if [condition],這種斷點(diǎn)真是非常管用。尤其是在一個(gè)循環(huán)或遞歸中,或是要監(jiān)視某個(gè)變量。注意,這個(gè)設(shè)置是在GDB中的,只不過(guò)每經(jīng)過(guò)那個(gè)斷點(diǎn)時(shí)GDB會(huì)幫你檢查一下條件是否滿(mǎn)足。
五、命令行參數(shù)
有時(shí)候,我們需要調(diào)試的程序需要有命令行參數(shù),很多朋友都不知道怎么設(shè)置調(diào)試的程序的命令行參數(shù)。其實(shí),有兩種方法:
- gdb命令行的 –args 參數(shù)
- gdb環(huán)境中 set args命令。
六、gdb的變量
有時(shí)候,在調(diào)試程序時(shí),我們不單單只是查看運(yùn)行時(shí)的變量,我們還可以直接設(shè)置程序中的變量,以模擬一些很難在測(cè)試中出現(xiàn)的情況,比較一些出錯(cuò),或是switch的分支語(yǔ)句。使用set命令可以修改程序中的變量。
另外,你知道gdb中也可以有變量嗎?就像shell一樣,gdb中的變量以$開(kāi)頭,比如你想打印一個(gè)數(shù)組中的個(gè)個(gè)元素,你可以這樣:
1 2 3 4 5 | (gdb) set $i = 0
(gdb) p a[$i++]
... #然后就一路回車(chē)下去了
|
當(dāng)然,這里只是給一個(gè)示例,表示程序的變量和gdb的變量是可以交互的。
七、x命令
也許,你很喜歡用p命令。所以,當(dāng)你不知道變量名的時(shí)候,你可能會(huì)手足無(wú)措,因?yàn)閜命令總是需要一個(gè)變量名的。x命令是用來(lái)查看內(nèi)存的,在gdb中 “help x” 你可以查看其幫助。
- x/x 以十六進(jìn)制輸出
- x/d 以十進(jìn)制輸出
- x/c 以單字符輸出
- x/i 反匯編 – 通常,我們會(huì)使用
x/10i $ip-20 來(lái)查看當(dāng)前的匯編($ip是指令寄存器)
- x/s 以字符串輸出
八、command命令
有一些朋友問(wèn)我如何自動(dòng)化調(diào)試。這里向大家介紹command命令,簡(jiǎn)單的理解一下,其就是把一組gdb的命令打包,有點(diǎn)像字處理軟件的“宏”。下面是一個(gè)示例:
1 2 3 4 5 6 7 8 9 10 | (gdb) break func
Breakpoint 1 at 0x3475678: file test .c, line 12.
(gdb) command 1
Type commands for when breakpoint 1 is hit, one per line.
End with a line saying just "end" .
>print arg1
>print arg2
>print arg3
>end
(gdb)
|
當(dāng)我們的斷點(diǎn)到達(dá)時(shí),自動(dòng)執(zhí)行command中的三個(gè)命令,把func的三個(gè)參數(shù)值打出來(lái)。
(全文完)
轉(zhuǎn)自:http://coolshell.cn/articles/3643.html