改變程序的執(zhí)行 ——————— 一旦使用GDB掛上被調(diào)試程序,,當(dāng)程序運(yùn)行起來后,,你可以根據(jù)自己的調(diào)試思路來動態(tài)地在GDB中更改當(dāng)前被調(diào)試程序的運(yùn)行線路或是其變量的值,這個強(qiáng)大的功能能夠讓你更好的調(diào)試你的程序,,比如,你可以在程序的一次運(yùn)行中走遍程序的所有分支,。 一,、修改變量值 修改被調(diào)試程序運(yùn)行時的變量值,在GDB中很容易實(shí)現(xiàn),,使用GDB的print命令即可完成,。如: (gdb) print x=4 x=4這個表達(dá)式是C/C++的語法,意為把變量x的值修改為4,,如果你當(dāng)前調(diào)試的語言是Pascal,,那么你可以使用Pascal的語法:x:=4。 在某些時候,,很有可能你的變量和GDB中的參數(shù)沖突,,如: (gdb) whatis width type = double (gdb) p width $4 = 13 (gdb) set width=47 Invalid syntax in expression. 因?yàn)椋瑂et width是GDB的命令,,所以,,出現(xiàn)了“Invalid syntax in expression”的設(shè)置錯誤,此時,,你可以使用set var命令來告訴GDB,,width不是你GDB的參數(shù),而是程序的變量名,,如: (gdb) set var width=47 另外,,還可能有些情況,GDB并不報告這種錯誤,,所以保險起見,,在你改變程序變量取值時,,最好都使用set var格式的GDB命令。 二,、跳轉(zhuǎn)執(zhí)行 一般來說,,被調(diào)試程序會按照程序代碼的運(yùn)行順序依次執(zhí)行。GDB提供了亂序執(zhí)行的功能,,也就是說,,GDB可以修改程序的執(zhí)行順序,可以讓程序執(zhí)行隨意跳躍,。這個功能可以由GDB的jump命令來完: jump <linespec> 指定下一條語句的運(yùn)行點(diǎn),。<linespce>可以是文件的行號,可以是file:line格式,,可以是+num這種偏移量格式,。表式著下一條運(yùn)行語句從哪里開始。 jump <address> 這里的<address>是代碼行的內(nèi)存地址,。 注意,,jump命令不會改變當(dāng)前的程序棧中的內(nèi)容,所以,,當(dāng)你從一個函數(shù)跳到另一個函數(shù)時,,當(dāng)函數(shù)運(yùn)行完返回時進(jìn)行彈棧操作時必然會發(fā)生錯誤,可能結(jié)果還是非常奇怪的,,甚至于產(chǎn)生程序Core Dump,。所以最好是同一個函數(shù)中進(jìn)行跳轉(zhuǎn)。 熟悉匯編的人都知道,,程序運(yùn)行時,,有一個寄存器用于保存當(dāng)前代碼所在的內(nèi)存地址。所以,,jump命令也就是改變了這個寄存器中的值,。于是,你可以使用“set $pc”來更改跳轉(zhuǎn)執(zhí)行的地址,。如: set $pc = 0x485 三,、產(chǎn)生信號量
使用singal命令,可以產(chǎn)生一個信號量給被調(diào)試的程序,。如:中斷信號Ctrl+C,。這非常方便于程序的調(diào)試,可以在程序運(yùn)行的任意位置設(shè)置斷點(diǎn),,并在該斷點(diǎn)用GDB產(chǎn)生一個信號量,,這種精確地在某處產(chǎn)生信號非常有利程序的調(diào)試。 語法是:signal <singal>,,UNIX的系統(tǒng)信號量通常從1到15,。所以<singal>取值也在這個范圍,。 single命令和shell的kill命令不同,系統(tǒng)的kill命令發(fā)信號給被調(diào)試程序時,,是由GDB截獲的,,而single命令所發(fā)出一信號則是直接發(fā)給被調(diào)試程序的。 四,、強(qiáng)制函數(shù)返回 如果你的調(diào)試斷點(diǎn)在某個函數(shù)中,,并還有語句沒有執(zhí)行完。你可以使用return命令強(qiáng)制函數(shù)忽略還沒有執(zhí)行的語句并返回,。 return return <expression> 使用return命令取消當(dāng)前函數(shù)的執(zhí)行,,并立即返回,如果指定了<expression>,,那么該表達(dá)式的值會被認(rèn)作函數(shù)的返回值,。 五、強(qiáng)制調(diào)用函數(shù) call <expr> 表達(dá)式中可以一是函數(shù),,以此達(dá)到強(qiáng)制調(diào)用函數(shù)的目的,。并顯示函數(shù)的返回值,如果函數(shù)返回值是void,,那么就不顯示,。 另一個相似的命令也可以完成這一功能——print,print后面可以跟表達(dá)式,,所以也可以用他來調(diào)用函數(shù),print和call的不同是,,如果函數(shù)返回void,,call則不顯示,print則顯示函數(shù)返回值,,并把該值存入歷史數(shù)據(jù)中,。 在不同語言中使用GDB —————————— GDB支持下列語言:C, C++, Fortran, PASCAL, Java, Chill, assembly, 和 Modula-2。一般說來,,GDB會根據(jù)你所調(diào)試的程序來確定當(dāng)然的調(diào)試語言,,比如:發(fā)現(xiàn)文件名后綴為“.c”的,GDB會認(rèn)為是C程序,。文件名后綴為“.C, .cc, .cp, .cpp, .cxx, .c++”的,,GDB會認(rèn)為是C++程序。而后綴是“.f, .F”的,,GDB會認(rèn)為是Fortran程序,,還有,后綴為如果是“.s, .S”的會認(rèn)為是匯編語言,。 也就是說,,GDB會根據(jù)你所調(diào)試的程序的語言,,來設(shè)置自己的語言環(huán)境,并讓GDB的命令跟著語言環(huán)境的改變而改變,。比如一些GDB命令需要用到表達(dá)式或變量時,,這些表達(dá)式或變量的語法,完全是根據(jù)當(dāng)前的語言環(huán)境而改變的,。例如C/C++中對指針的語法是*p,,而在Modula-2中則是p^。并且,,如果你當(dāng)前的程序是由幾種不同語言一同編譯成的,,那到在調(diào)試過程中,GDB也能根據(jù)不同的語言自動地切換語言環(huán)境,。這種跟著語言環(huán)境而改變的功能,,真是體貼開發(fā)人員的一種設(shè)計(jì)。 下面是幾個相關(guān)于GDB語言環(huán)境的命令:
show language 查看當(dāng)前的語言環(huán)境,。如果GDB不能識為你所調(diào)試的編程語言,,那么,C語言被認(rèn)為是默認(rèn)的環(huán)境,。 info frame 查看當(dāng)前函數(shù)的程序語言,。 info source 查看當(dāng)前文件的程序語言。 如果GDB沒有檢測出當(dāng)前的程序語言,,那么你也可以手動設(shè)置當(dāng)前的程序語言,。使用set language命令即可做到。 當(dāng)set language命令后什么也不跟的話,,你可以查看GDB所支持的語言種類: (gdb) set language The currently understood settings are: local or auto Automatic setting based on source file c Use the C language c++ Use the C++ language asm Use the Asm language chill Use the Chill language fortran Use the Fortran language java Use the Java language modula-2 Use the Modula-2 language pascal Use the Pascal language scheme Use the Scheme language 于是你可以在set language后跟上被列出來的程序語言名,,來設(shè)置當(dāng)前的語言環(huán)境。 后記 —— GDB是一個強(qiáng)大的命令行調(diào)試工具,。大家知道命令行的強(qiáng)大就是在于,,其可以形成執(zhí)行序列,形成腳本,。UNIX下的軟件全是命令行的,,這給程序開發(fā)提代供了極大的便利,命令行軟件的優(yōu)勢在于,,它們可以非常容易的集成在一起,,使用幾個簡單的已有工具的命令,就可以做出一個非常強(qiáng)大的功能,。 于是UNIX下的軟件比Windows下的軟件更能有機(jī)地結(jié)合,,各自發(fā)揮各自的長處,組合成更為強(qiáng)勁的功能,。而Windows下的圖形軟件基本上是各自為營,,互相不能調(diào)用,,很不利于各種軟件的相互集成。在這里并不是要和Windows做個什么比較,,所謂“寸有所長,,尺有所短”,圖形化工具還是有不如命令行的地方,。(看到這句話時,,希望各位千萬再也不要認(rèn)為我就是“鄙視圖形界面”,和我抬杠了 ) 我是根據(jù)版本為5.1.1的GDB所寫的這篇文章,,所以可能有些功能已被修改,,或是又有更為強(qiáng)勁的功能。而且,,我寫得非常倉促,,寫得比較簡略,并且,,其中我已經(jīng)看到有許多錯別字了(我用五筆,,所以錯字讓你看不懂),所以,,我在這里對我文中的差錯表示萬分的歉意,。 文中所羅列的GDB的功能時,我只是羅列了一些帶用的GDB的命令和使用方法,,其實(shí),,我這里只講述的功能大約只占GDB所有功能的60%吧,詳細(xì)的文檔,,還是請查看GDB的幫助和使用手冊吧,,或許,過段時間,,如果我有空,我再寫一篇GDB的高級使用,。 我個人非常喜歡GDB的自動調(diào)試的功能,,這個功能真的很強(qiáng)大,試想,,我在UNIX下寫個腳本,,讓腳本自動編譯我的程序,被自動調(diào)試,,并把結(jié)果報告出來,,調(diào)試成功,自動checkin源碼庫,。一個命令,,編譯帶著調(diào)試帶著checkin,,多爽啊。只是GDB對自動化調(diào)試目前支持還不是很成熟,,只能實(shí)現(xiàn)半自動化,,真心期望著GDB的自動化調(diào)試功能的成熟。 如果各位對GDB或是別的技術(shù)問題有興趣的話,,歡迎和我討論交流,。本人目前主要在UNIX下做產(chǎn)品軟件的開發(fā),所以,,對UNIX下的軟件開發(fā)比較熟悉,,當(dāng)然,不單單是技術(shù),,對軟件工程實(shí)施,,軟件設(shè)計(jì),系統(tǒng)分析,,項(xiàng)目管理我也略有心得,。歡迎大家找我交流,(QQ是:753640,,MSN是:[email protected])
|