linux GDB详解
1 簡(jiǎn)介
2 生成調(diào)試信息
3 啟動(dòng)GDB 的方法
4 程序運(yùn)行上下文
????4.1 程序運(yùn)行參數(shù)
????4.2 工作目錄
????4.3 程序的輸入輸出
5 設(shè)置斷點(diǎn)
????5.1 簡(jiǎn)單斷點(diǎn)
????5.2 多文件設(shè)置斷點(diǎn)
????5.3 查詢所有斷點(diǎn)
6 觀察點(diǎn)
7 條件斷點(diǎn)
8 維護(hù)停止點(diǎn)
9 為停止點(diǎn)設(shè)定運(yùn)行命令
10 調(diào)試代碼
11 查看運(yùn)行時(shí)數(shù)據(jù)
12 程序變量
13 自動(dòng)顯示
14 歷史記錄
15 改變程序的執(zhí)行
????15.1 修改變量值
????15.2 跳轉(zhuǎn)執(zhí)行
????15.3 產(chǎn)生信號(hào)量
????15.4 強(qiáng)制函數(shù)返回
????15.5 強(qiáng)制調(diào)用函數(shù)
16 顯示源代碼
17 調(diào)試已運(yùn)行的進(jìn)程
18 線程
19 查看棧信息
20 信號(hào)
21 catch
22 指定源文件的路徑
1 簡(jiǎn)介
GDB(GNU Debugger)是GCC的調(diào)試工具。其功能強(qiáng)大,現(xiàn)描述如下:?
GDB主要幫忙你完成下面四個(gè)方面的功能:?
1.啟動(dòng)你的程序,可以按照你的自定義的要求隨心所欲的運(yùn)行程序。?
2.可讓被調(diào)試的程序在你所指定的調(diào)置的斷點(diǎn)處停住。(斷點(diǎn)可以是條件表達(dá)式)?
3.當(dāng)程序被停住時(shí),可以檢查此時(shí)你的程序中所發(fā)生的事。?
4.動(dòng)態(tài)的改變你程序的執(zhí)行環(huán)境。
2 生成調(diào)試信息
一般來(lái)說(shuō)GDB主要調(diào)試的是C/C++的程序。要調(diào)試C/C++的程序,首先在編譯時(shí),我們必須要把調(diào)試信息加到可執(zhí)行文件中。使用編譯器(cc/gcc/g++)的 -g 參數(shù)可以做到這一點(diǎn)。如:
gcc -g hello.c -o hello
g++ -g hello.cpp -o hello
如果沒(méi)有-g,你將看不見(jiàn)程序的函數(shù)名、變量名,所代替的全是運(yùn)行時(shí)的內(nèi)存地址。當(dāng)你用-g把調(diào)試信息加入之后,并成功編譯目標(biāo)代碼以后,讓我們來(lái)看看如何用gdb來(lái)調(diào)試他。
3 啟動(dòng)GDB?的方法
1、gdb program
program 也就是你的執(zhí)行文件,一般在當(dāng)前目錄下。
2、gdb program core
用gdb同時(shí)調(diào)試一個(gè)運(yùn)行程序和core文件,core是程序非法執(zhí)行后core dump后產(chǎn)生的文件。
3、gdb program 1234
如果你的程序是一個(gè)服務(wù)程序,那么你可以指定這個(gè)服務(wù)程序運(yùn)行時(shí)的進(jìn)程ID。gdb會(huì)自動(dòng)attach上去,并調(diào)試他。program應(yīng)該在PATH環(huán)境變量中搜索得到。
4 程序運(yùn)行上下文
4.1 程序運(yùn)行參數(shù)
set args 可指定運(yùn)行時(shí)參數(shù)。(如:set args 10 20 30 40 50 )
show args 命令可以查看設(shè)置好的運(yùn)行參數(shù)。
run (r) 啟動(dòng)程序
不指定運(yùn)行參數(shù) r
指定運(yùn)行參數(shù)r 10 20 30 40 50
4.2 工作目錄
cd 相當(dāng)于shell的cd命令。
pwd 顯示當(dāng)前的所在目錄。
4.3 程序的輸入輸出
info terminal 顯示你程序用到的終端的模式。
使用重定向控制程序輸出。如:run > outfile
tty命令可以設(shè)置輸入輸出使用的終端設(shè)備。如:tty /dev/tty1
5 設(shè)置斷點(diǎn)
5.1 簡(jiǎn)單斷點(diǎn)
break 設(shè)置斷點(diǎn),可以簡(jiǎn)寫(xiě)為b
b 10 設(shè)置斷點(diǎn),在源程序第10行
b func 設(shè)置斷點(diǎn),在func函數(shù)入口處
5.2 多文件設(shè)置斷點(diǎn)
在進(jìn)入指定函數(shù)時(shí)停住:
C++中可以使用class::function或function(type,type)格式來(lái)指定函數(shù)名。如果有名稱空間,可以使用namespace::class::function或者function(type,type)格式來(lái)指定函數(shù)名。
break filename:linenum?
在源文件filename的linenum行處停住?
break filename:function?
在源文件filename的function函數(shù)的入口處停住
break class::function或function(type,type)
在類(lèi)class的function函數(shù)的入口處停住
break namespace::class::function
在名稱空間為namespace的類(lèi)class的function函數(shù)的入口處停住
5.3 查詢所有斷點(diǎn)
info b
6 觀察點(diǎn)
watch 為表達(dá)式(變量)expr設(shè)置一個(gè)觀察點(diǎn)。當(dāng)表達(dá)式值有變化時(shí),馬上停住程序。
rwatch 表達(dá)式(變量)expr被讀時(shí),停住程序。
awatch 表達(dá)式(變量)的值被讀或被寫(xiě)時(shí),停住程序。
info watchpoints 列出當(dāng)前所設(shè)置了的所有觀察點(diǎn)。
7 條件斷點(diǎn)
一般來(lái)說(shuō),為斷點(diǎn)設(shè)置一個(gè)條件,我們使用if關(guān)鍵詞,后面跟其斷點(diǎn)條件。并且,條件設(shè)置好后,我們可以用condition命令來(lái)修改斷點(diǎn)的條件。并且,條件設(shè)置好后,我們可以用condition命令來(lái)修改斷點(diǎn)的條件。(只有break 和 watch命令支持if,catch目前暫不支持if)。
設(shè)置一個(gè)條件斷點(diǎn)
b test.c:8 if intValue == 5
condition 與break if類(lèi)似,只是condition只能用在已存在的斷點(diǎn)上
修改斷點(diǎn)號(hào)為bnum的停止條件為expression
condition bnum expression
清楚斷點(diǎn)號(hào)為bnum的停止條件
condition bnum
ignore 忽略停止條件幾次
表示忽略斷點(diǎn)號(hào)為bnum的停止條件count次
Ignore bnum count
8 維護(hù)停止點(diǎn)
clear 清除所有的已定義的停止點(diǎn)。
clear function 清除所有設(shè)置在函數(shù)上的停止點(diǎn)。
clear linenum 清除所有設(shè)置在指定行上的停止點(diǎn)。
clear filename:linenum 清除所有設(shè)置在指定文件:指定行上的停止點(diǎn)。
delete [breakpoints] [range...] 刪除指定的斷點(diǎn),breakpoints為斷點(diǎn)號(hào)。如果不指定斷點(diǎn)號(hào),則表示刪除所有的斷點(diǎn)。range 表示斷點(diǎn)號(hào)的范圍(如:3-7)。其簡(jiǎn)寫(xiě)命令為d。
比刪除更好的一種方法是disable停止點(diǎn),disable了的停止點(diǎn),GDB不會(huì)刪除,當(dāng)你還需要時(shí),enable即可,就好像回收站一樣。
disable [breakpoints] [range...]
disable所指定的停止點(diǎn),breakpoints為停止點(diǎn)號(hào)。如果什么都不指定,表示disable所有的停止點(diǎn)。簡(jiǎn)寫(xiě)命令是dis.
enable [breakpoints] [range...]
enable所指定的停止點(diǎn),breakpoints為停止點(diǎn)號(hào)。
enable [breakpoints] once range…
enable所指定的停止點(diǎn)一次,當(dāng)程序停止后,該停止點(diǎn)馬上被GDB自動(dòng)disable。
enable [breakpoints] delete range…
enable所指定的停止點(diǎn)一次,當(dāng)程序停止后,該停止點(diǎn)馬上被GDB自動(dòng)刪除。
9 為停止點(diǎn)設(shè)定運(yùn)行命令
我們可以使用GDB提供的command命令來(lái)設(shè)置停止點(diǎn)的運(yùn)行命令。也就是說(shuō),當(dāng)運(yùn)行的程序在被停止住時(shí),我們可以讓其自動(dòng)運(yùn)行一些別的命令,這很有利行自動(dòng)化調(diào)試。對(duì)基于GDB的自動(dòng)化調(diào)試是一個(gè)強(qiáng)大的支持。
commands [bnum]
… command-list …
end
為斷點(diǎn)號(hào)bnum指寫(xiě)一個(gè)命令列表。當(dāng)程序被該斷點(diǎn)停住時(shí),gdb會(huì)依次運(yùn)行命令列表中的命令。
例如:
break foo if x>0
commands
printf “x is %d “,x
continue
end
斷點(diǎn)設(shè)置在函數(shù)foo中,斷點(diǎn)條件是x>0,如果程序被斷住后,也就是,一旦x的值在foo函數(shù)中大于0,GDB會(huì)自動(dòng)打印出x的值,并繼續(xù)運(yùn)行程序。
如果你要清除斷點(diǎn)上的命令序列,那么只要簡(jiǎn)單的執(zhí)行一下commands命令,并直接在打個(gè)end就行了。
10 調(diào)試代碼
run 運(yùn)行程序,可簡(jiǎn)寫(xiě)為r
next 單步跟蹤,函數(shù)調(diào)用當(dāng)作一條簡(jiǎn)單語(yǔ)句執(zhí)行,可簡(jiǎn)寫(xiě)為n
step 單步跟蹤,函數(shù)調(diào)進(jìn)入被調(diào)用函數(shù)體內(nèi),可簡(jiǎn)寫(xiě)為s
finish 退出函數(shù)
until 在一個(gè)循環(huán)體內(nèi)單步跟蹤時(shí),這個(gè)命令可以運(yùn)行程序直到退出循環(huán)體,可簡(jiǎn)寫(xiě)為u。
continue 繼續(xù)運(yùn)行程序,可簡(jiǎn)寫(xiě)為c
stepi或si, nexti或ni 單步跟蹤一條機(jī)器指令,一條程序代碼有可能由數(shù)條機(jī)器指令完成,stepi和nexti可以單步執(zhí)行機(jī)器指令。
info program 來(lái)查看程序的是否在運(yùn)行,進(jìn)程號(hào),被暫停的原因。
11 查看運(yùn)行時(shí)數(shù)據(jù)
print 打印變量、字符串、表達(dá)式等的值,可簡(jiǎn)寫(xiě)為p?
p count 打印count的值?
p cou1+cou2+cou3 打印表達(dá)式值
print接受一個(gè)表達(dá)式,GDB會(huì)根據(jù)當(dāng)前的程序運(yùn)行的數(shù)據(jù)來(lái)計(jì)算這個(gè)表達(dá)式,表達(dá)式可以是當(dāng)前程序運(yùn)行中的const常量、變量、函數(shù)等內(nèi)容。但是GDB不能使用程序中定義的宏。
12 程序變量
在GDB中,你可以隨時(shí)查看以下三種變量的值:
1、全局變量(所有文件可見(jiàn)的)
2、靜態(tài)全局變量(當(dāng)前文件可見(jiàn)的)
3、局部變量(當(dāng)前Scope可見(jiàn)的)
如果你的局部變量和全局變量發(fā)生沖突(也就是重名),一般情況下是局部變量會(huì)隱藏全局變量,也就是說(shuō),如果一個(gè)全局變量和一個(gè)函數(shù)中的局部變量同名時(shí),如果當(dāng)前停止點(diǎn)在函數(shù)中,用print顯示出的變量的值會(huì)是函數(shù)中的局部變量的值。如果此時(shí)你想查看全局變量的值時(shí),你可以使用“::”操作符:
file::variable
function::variable
可以通過(guò)這種形式指定你所想查看的變量,是哪個(gè)文件中的或是哪個(gè)函數(shù)中的。例如,查看文件f2.c中的全局變量x的值:
p ‘f2.c’::x
當(dāng)然,“::”操作符會(huì)和C++中的發(fā)生沖突,GDB能自動(dòng)識(shí)別“::”是否C++的操作符,所以你不必?fù)?dān)心在調(diào)試C++程序時(shí)會(huì)出現(xiàn)異常。
4數(shù)組變量
有時(shí)候,你需要查看一段連續(xù)的內(nèi)存空間的值。比如數(shù)組的一段,或是動(dòng)態(tài)分配的數(shù)據(jù)的大小。你可以使用GDB的“@”操作符,“@”的左邊是第一個(gè)內(nèi)存的地址的值,“@”的右邊則你你想查看內(nèi)存的長(zhǎng)度。例如,你的程序中有這樣的語(yǔ)句:
int *array = (int *) malloc (len * sizeof (int));
于是,在GDB調(diào)試過(guò)程中,你可以以如下命令顯示出這個(gè)動(dòng)態(tài)數(shù)組的取值:
p *array@len
@的左邊是數(shù)組的首地址的值,也就是變量array所指向的內(nèi)容,右邊則是數(shù)據(jù)的長(zhǎng)度,其保存在變量len中。
13 自動(dòng)顯示
你可以設(shè)置一些自動(dòng)顯示的變量,當(dāng)程序停住時(shí),或是在你單步跟蹤時(shí),這些變量會(huì)自動(dòng)顯示。相關(guān)的GDB命令是display。
display expr
display/fmt expr
display/fmt addr
expr是一個(gè)表達(dá)式,fmt表示顯示的格式,addr表示內(nèi)存地址,當(dāng)你用display設(shè)定好了一個(gè)或多個(gè)表達(dá)式后,只要你的程序被停下來(lái),GDB會(huì)自動(dòng)顯示你所設(shè)置的這些表達(dá)式的值。
info display
查看display設(shè)置的自動(dòng)顯示的信息。
undisplay dnums…
delete display dnums…
刪除自動(dòng)顯示,dnums意為所設(shè)置好了的自動(dòng)顯式的編號(hào)。如果要同時(shí)刪除幾個(gè),編號(hào)可以用空格分隔,如果要?jiǎng)h除一個(gè)范圍內(nèi)的編號(hào),可以用減號(hào)表示(如:2-5)
disable display dnums…
enable display dnums…
disable和enalbe不刪除自動(dòng)顯示的設(shè)置,而只是讓其失效和恢復(fù)。
14 歷史記錄
當(dāng)你用GDB的print查看程序運(yùn)行時(shí)的數(shù)據(jù)時(shí),你每一個(gè)print都會(huì)被GDB記錄下來(lái)。GDB會(huì)以$1, $2, $3 …..這樣的方式為你每一個(gè)print命令編上號(hào)。于是,你可以使用這個(gè)編號(hào)訪問(wèn)以前的表達(dá)式,如$1。這個(gè)功能所帶來(lái)的好處是,如果你先前輸入了一個(gè)比較長(zhǎng)的表達(dá)式,如果你還想查看這個(gè)表達(dá)式的值,你可以使用歷史記錄來(lái)訪問(wèn),省去了重復(fù)輸入。
show values
Print the last ten values in the value history, with their item numbers. This is
like ‘p $$9’ repeated ten times, except that show values does not change the
history.
show values n
Print ten history values centered on history item number n.
show values +
Print ten history values just after the values last printed. If no more values are
available, show values + produces no display.
15 改變程序的執(zhí)行
一旦使用GDB掛上被調(diào)試程序,當(dāng)程序運(yùn)行起來(lái)后,你可以根據(jù)自己的調(diào)試思路來(lái)動(dòng)態(tài)地在GDB中更改當(dāng)前被調(diào)試程序的運(yùn)行線路或是其變量的值,這個(gè)強(qiáng)大的功能能夠讓你更好的調(diào)試你的程序,比如,你可以在程序的一次運(yùn)行中走遍程序的所有分支。
15.1 修改變量值
修改被調(diào)試程序運(yùn)行時(shí)的變量值,在GDB中很容易實(shí)現(xiàn),使用GDB的print命令即可完成。如:
(gdb) print x=4
x=4這個(gè)表達(dá)式是C/C++的語(yǔ)法,意為把變量x的值修改為4,如果你當(dāng)前調(diào)試的語(yǔ)言是Pascal,那么你可以使用Pascal的語(yǔ)法:x:=4。
在某些時(shí)候,很有可能你的變量和GDB中的參數(shù)沖突,如:
(gdb) whatis width
type = double
(gdb) p width
$4 = 13
(gdb) set width=47
Invalid syntax in expression.
因?yàn)?#xff0c;set width是GDB的命令,所以,出現(xiàn)了“Invalid syntax in expression”的設(shè)置錯(cuò)誤,此時(shí),你可以使用set var命令來(lái)告訴GDB,width不是你GDB的參數(shù),而是程序的變量名,如:
(gdb) set var width=47
另外,還可能有些情況,GDB并不報(bào)告這種錯(cuò)誤,所以保險(xiǎn)起見(jiàn),在你改變程序變量取值時(shí),最好都使用set var格式的GDB命令。
15.2 跳轉(zhuǎn)執(zhí)行
一般來(lái)說(shuō),被調(diào)試程序會(huì)按照程序代碼的運(yùn)行順序依次執(zhí)行。GDB提供了亂序執(zhí)行的功能,也就是說(shuō),GDB可以修改程序的執(zhí)行順序,可以讓程序執(zhí)行隨意跳躍。這個(gè)功能可以由GDB的jump命令來(lái)完:
jump linespec
指定下一條語(yǔ)句的運(yùn)行點(diǎn)??梢允俏募男刑?hào),可以是file:line格式,可以是+num這種偏移量格式。表示下一條運(yùn)行語(yǔ)句從哪里開(kāi)始。
jump *address
這里的是代碼行的內(nèi)存地址。
注意,jump命令不會(huì)改變當(dāng)前的程序棧中的內(nèi)容,所以,當(dāng)你從一個(gè)函數(shù)跳到另一個(gè)函數(shù)時(shí),當(dāng)函數(shù)運(yùn)行完返回時(shí)進(jìn)行彈棧操作時(shí)必然會(huì)發(fā)生錯(cuò)誤,可能結(jié)果還是非常奇怪的,甚至于產(chǎn)生程序Core Dump。所以最好是同一個(gè)函數(shù)中進(jìn)行跳轉(zhuǎn)。
熟悉匯編的人都知道,程序運(yùn)行時(shí),eip寄存器用于保存當(dāng)前代碼所在的內(nèi)存地址。所以,jump命令也就是改變了這個(gè)寄存器中的值。于是,你可以使用“set $pc”來(lái)更改跳轉(zhuǎn)執(zhí)行的地址。如:
set $pc = 0×485
15.3 產(chǎn)生信號(hào)量
使用singal命令,可以產(chǎn)生一個(gè)信號(hào)量給被調(diào)試的程序。如:中斷信號(hào)Ctrl+C。這非常方便于程序的調(diào)試,可以在程序運(yùn)行的任意位置設(shè)置斷點(diǎn),并在該斷點(diǎn)用GDB產(chǎn)生一個(gè)信號(hào)量,這種精確地在某處產(chǎn)生信號(hào)非常有利程序的調(diào)試。
語(yǔ)法是:
signal signal
UNIX的系統(tǒng)信號(hào)量通常從1到15。所以取值也在這個(gè)范圍。
single命令和shell的kill命令不同,系統(tǒng)的kill命令發(fā)信號(hào)給被調(diào)試程序時(shí),是由GDB截獲的,而single命令所發(fā)出一信號(hào)則是直接發(fā)給被調(diào)試程序的。
15.4 強(qiáng)制函數(shù)返回
如果你的調(diào)試斷點(diǎn)在某個(gè)函數(shù)中,并還有語(yǔ)句沒(méi)有執(zhí)行完。你可以使用return命令強(qiáng)制函數(shù)忽略還沒(méi)有執(zhí)行的語(yǔ)句并返回。
return
return expression
使用return命令取消當(dāng)前函數(shù)的執(zhí)行,并立即返回,如果指定了,那么該表達(dá)式的值會(huì)被認(rèn)作函數(shù)的返回值。
15.5?強(qiáng)制調(diào)用函數(shù)
call expr
表達(dá)式中可以一是函數(shù),以此達(dá)到強(qiáng)制調(diào)用函數(shù)的目的。并顯示函數(shù)的返回值,如果函數(shù)返回值是void,那么就不顯示。
print expr
另一個(gè)相似的命令也可以完成這一功能——print,print后面可以跟表達(dá)式,所以也可以用他來(lái)調(diào)用函數(shù),print和call的不同是,如果函數(shù)返回void,call則不顯示,print則顯示函數(shù)返回值,并把該值存入歷史數(shù)據(jù)中。
16 顯示源代碼
GDB 可以打印出所調(diào)試程序的源代碼,當(dāng)然,在程序編譯時(shí)一定要加上 –g 的參數(shù),把源程序信息編譯到執(zhí)行文件中。不然就看不到源程序了。當(dāng)程序停下來(lái)以后, GDB會(huì)報(bào)告程序停在了那個(gè)文件的第幾行上。你可以用list命令來(lái)打印程序的源代碼。默認(rèn)打印10行,還是來(lái)看一看查看源代碼的GDB命令吧。
list linenum
Print lines centered around line number linenum in the current source file.
list function
顯示函數(shù)名為function的函數(shù)的源程序。
list
顯示當(dāng)前行后面的源程序。
list -
顯示當(dāng)前行前面的源程序。
一般是打印當(dāng)前行的上5行和下5行,如果顯示函數(shù)是是上2行下8行,默認(rèn)是10行,當(dāng)然,你也可以定制顯示的范圍,使用下面命令可以設(shè)置一次顯示源程序的行數(shù)。
set listsize count
設(shè)置一次顯示源代碼的行數(shù)。(unless the list argument explicitly specifies some other number)
show listsize
查看當(dāng)前l(fā)istsize的設(shè)置。
17 調(diào)試已運(yùn)行的進(jìn)程
兩種方法:
1、在UNIX下用ps查看正在運(yùn)行的程序的PID(進(jìn)程ID),然后用gdb PID process-id 格式掛接正在運(yùn)行的程序。
2、先用gdb 關(guān)聯(lián)上源代碼,并進(jìn)行g(shù)db,在gdb中用attach process-id 命令來(lái)掛接進(jìn)程的PID。并用detach來(lái)取消掛接的進(jìn)程。
18 線程
如果你程序是多線程的話,你可以定義你的斷點(diǎn)是否在所有的線程上,或是在某個(gè)特定的線程。GDB很容易幫你完成這一工作。
break linespec thread threadno
break linespec thread threadno if …
linespec指定了斷點(diǎn)設(shè)置在的源程序的行號(hào)。threadno指定了線程的ID,注意,這個(gè)ID是GDB分配的,你可以通過(guò)“info threads”命令來(lái)查看正在運(yùn)行程序中的線程信息。如果你不指定‘thread threadno ’則表示你的斷點(diǎn)設(shè)在所有線程上面。你還可以為某線程指定斷點(diǎn)條件。如:
(gdb) break frik.c:13 thread 28 if bartab > lim
當(dāng)你的程序被GDB停住時(shí),所有的運(yùn)行線程都會(huì)被停住。這方便你你查看運(yùn)行程序的總體情況。而在你恢復(fù)程序運(yùn)行時(shí),所有的線程也會(huì)被恢復(fù)運(yùn)行。那怕是主進(jìn)程在被單步調(diào)試時(shí)。
19 查看棧信息
當(dāng)程序被停住了,你需要做的第一件事就是查看程序是在哪里停住的。當(dāng)你的程序調(diào)用了一個(gè)函數(shù),函數(shù)的地址,函數(shù)參數(shù),函數(shù)內(nèi)的局部變量都會(huì)被壓入“?!?#xff08;Stack)中。你可以用GDB命令來(lái)查看當(dāng)前的棧中的信息。
下面是一些查看函數(shù)調(diào)用棧信息的GDB命令:
breacktrace,簡(jiǎn)稱bt
打印當(dāng)前的函數(shù)調(diào)用棧的所有信息。如:
(gdb) bt
#0 func (n=250) at tst.c:6
#1 0x08048524 in main (argc=1, argv=0xbffff674) at tst.c:30
#2 0x400409ed in __libc_start_main () from /lib/libc.so.6
從上可以看出函數(shù)的調(diào)用棧信息:__libc_start_main –> main() –> func()
backtrace n
bt n
n是一個(gè)正整數(shù),表示只打印棧頂上n層的棧信息。
backtrace -n
bt -n
-n表一個(gè)負(fù)整數(shù),表示只打印棧底下n層的棧信息。
如果你要查看某一層的信息,你需要在切換當(dāng)前的棧,一般來(lái)說(shuō),程序停止時(shí),最頂層的棧就是當(dāng)前棧,如果你要查看棧下面層的詳細(xì)信息,首先要做的是切換當(dāng)前棧。
frame n
n是一個(gè)從0開(kāi)始的整數(shù),是棧中的層編號(hào)。比如:frame 0,表示棧頂,frame 1,表示棧的第二層。
frame addr
f addr Select the frame at address addr. This is useful mainly if the chaining of stack frames has been damaged by a bug, making it impossible for gdb to assign
numbers properly to all frames. In addition, this can be useful when your program has multiple stacks and switches between them.
up n
表示向棧的上面移動(dòng)n層,可以不打n,表示向上移動(dòng)一層。
down n
表示向棧的下面移動(dòng)n層,可以不打n,表示向下移動(dòng)一層。
上面的命令,都會(huì)打印出移動(dòng)到的棧層的信息。如果你不想讓其打出信息。你可以使用這三個(gè)命令:
select-frame 對(duì)應(yīng)于 frame 命令。
up-silently n 對(duì)應(yīng)于 up 命令。
down-silently n 對(duì)應(yīng)于 down 命令。
查看當(dāng)前棧層的信息,你可以用以下GDB命令:
frame 或 f
會(huì)打印出這些信息:棧的層編號(hào),當(dāng)前的函數(shù)名,函數(shù)參數(shù)值,函數(shù)所在文件及行號(hào),函數(shù)執(zhí)行到的語(yǔ)句。
info frame
info f
20 信號(hào)
信號(hào)是一種軟中斷,是一種處理異步事件的方法。一般來(lái)說(shuō),操作系統(tǒng)都支持許多信號(hào)。尤其是UNIX,比較重要應(yīng)用程序一般都會(huì)處理信號(hào)。UNIX定義了許多信號(hào),比如SIGINT表示中斷字符信號(hào),也就是Ctrl+C的信號(hào),SIGBUS表示硬件故障的信號(hào);SIGCHLD表示子進(jìn)程狀態(tài)改變信號(hào); SIGKILL表示終止程序運(yùn)行的信號(hào),等等。
調(diào)試程序的時(shí)候處理信號(hào):
handle signal [keywords...]
signal可以以SIG開(kāi)頭或不以SIG開(kāi)頭,可以用定義一個(gè)要處理信號(hào)的范圍(如:SIGIO-SIGKILL,表示處理從 SIGIO信號(hào)到SIGKILL的信號(hào),其中包括SIGIO,SIGIOT,SIGKILL三個(gè)信號(hào)),也可以使用關(guān)鍵字all來(lái)標(biāo)明要處理所有的信號(hào)。一旦被調(diào)試的程序接收到信號(hào),運(yùn)行程序馬上會(huì)被GDB停住,以供調(diào)試。
keywords列表如下:
nostop
當(dāng)被調(diào)試的程序收到信號(hào)時(shí),GDB不會(huì)停住程序的運(yùn)行,但會(huì)打出消息告訴你收到這種信號(hào)。
stop
當(dāng)被調(diào)試的程序收到信號(hào)時(shí),GDB會(huì)停住你的程序。This implies the print keyword as well.
當(dāng)被調(diào)試的程序收到信號(hào)時(shí),GDB會(huì)顯示出一條信息。
noprint
當(dāng)被調(diào)試的程序收到信號(hào)時(shí),GDB不會(huì)告訴你收到信號(hào)的信息。This implies the nostop keyword as well.
pass
noignore
當(dāng)被調(diào)試的程序收到信號(hào)時(shí),GDB不處理信號(hào)。這表示,GDB會(huì)把這個(gè)信號(hào)交給被調(diào)試程序處理 or else it may terminate if the signal is fatal and not handled.
nopass
ignore
當(dāng)被調(diào)試的程序收到信號(hào)時(shí),GDB不會(huì)讓被調(diào)試程序來(lái)處理這個(gè)信號(hào)。
info signals
info handle
查看有哪些信號(hào)在被GDB檢測(cè)中。
21catch
當(dāng)event發(fā)生時(shí),停住程序。event可以是下面的內(nèi)容:?
1、throw 一個(gè)C++拋出的異常。(throw為關(guān)鍵字)?
2、catch 一個(gè)C++捕捉到的異常。(catch為關(guān)鍵字)
22 指定源文件的路徑
某些時(shí)候,用-g編譯過(guò)后的執(zhí)行程序中只是包括了源文件的名字,沒(méi)有路徑名。GDB提供了可以讓你指定源文件的路徑的命令,以便GDB進(jìn)行搜索。
Directory dirname …
dir dirname …
加一個(gè)源文件路徑到當(dāng)前路徑的前面。如果你要指定多個(gè)路徑,UNIX下你可以使用“:”,Windows下你可以使用“;”。
directory
清除所有的自定義的源文件搜索路徑信息。
show directories
顯示定義了的源文件搜索路徑。
總結(jié)
以上是生活随笔為你收集整理的linux GDB详解的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。
- 上一篇: IIS状态代码的含义
- 下一篇: 实用make最佳实践