1. 程式人生 > >Linux gdb除錯多執行緒

Linux gdb除錯多執行緒

一、多執行緒除錯

多執行緒除錯可能是問得最多的。其實,重要就是下面幾個命令:

info thread 檢視當前程序的執行緒。 thread <ID> 切換除錯的執行緒為指定ID的執行緒。 break file.c:100 thread all  在file.c檔案第100行處為所有經過這裡的執行緒設定斷點。 set scheduler-locking off|on|step,這個是問得最多的。在使用step或者continue命令除錯當前被除錯執行緒的時候,其他執行緒也是同時執行的,怎麼只讓被除錯程式執行呢?通過這個命令就可以實現這個需求。 off 不鎖定任何執行緒,也就是所有執行緒都執行,這是預設值。 on 只有當前被除錯程式會執行。 step 在單步的時候,除了next過一個函式的情況(熟悉情況的人可能知道,這其實是一個設定斷點然後continue的行為)以外,只有當前執行緒會執行。 二、除錯巨集

這個問題超多。在GDB下,我們無法print巨集定義,因為巨集是預編譯的。但是我們還是有辦法來除錯巨集,這個需要GCC的配合。

在GCC編譯程式的時候,加上-ggdb3引數,這樣,你就可以除錯巨集了。

另外,你可以使用下述的GDB的巨集除錯命令 來檢視相關的巨集。

info macro – 你可以檢視這個巨集在哪些檔案裡被引用了,以及巨集定義是什麼樣的。 macro – 你可以檢視巨集展開的樣子。

三、原始檔

這個問題問的也是很多的,太多的朋友都說找不到原始檔。在這裡我想提醒大家做下面的檢查:

編譯程式設計師是否加上了-g引數以包含debug資訊。 路徑是否設定正確了。使用GDB的directory命令來設定原始檔的目錄。

下面給一個除錯/bin/ls的示例(ubuntu下)

1234567891011121314151617181920$ apt-get sourcecoreutils$ sudoapt-get installcoreutils-dbgsym$ gdb /bin/lsGNU gdb (GDB) 7.1-ubuntu(gdb) list main1192    ls.c: No such fileor directory.inls.c(gdb) directory ~/src/coreutils-7.4/src/Source directories searched: /home/hchen/src/coreutils-7.4:$cdir:$cwd(gdb) list main1192        }1193    }11941195    int1196    main (int argc, char **argv)1197    {1198      int i;1199      struct pending *thispend;1200      int n_files;1201

四、條件斷點

條件斷點是語法是:break  [where] if [condition],這種斷點真是非常管用。尤其是在一個迴圈或遞迴中,或是要監視某個變數。注意,這個設定是在GDB中的,只不過每經過那個斷點時GDB會幫你檢查一下條件是否滿足。

五、命令列引數

有時候,我們需要除錯的程式需要有命令列引數,很多朋友都不知道怎麼設定除錯的程式的命令列引數。其實,有兩種方法:

gdb命令列的 –args 引數 gdb環境中 set args命令。 六、gdb的變數

有時候,在除錯程式時,我們不單單只是檢視執行時的變數,我們還可以直接設定程式中的變數,以模擬一些很難在測試中出現的情況,比較一些出錯,或是switch的分支語句。使用set命令可以修改程式中的變數。

另外,你知道gdb中也可以有變數嗎?就像shell一樣,gdb中的變數以$開頭,比如你想列印一個數組中的個個元素,你可以這樣:

12345(gdb) set$i = 0  (gdb) p a[$i++]  ...  #然後就一路回車下去了

當然,這裡只是給一個示例,表示程式的變數和gdb的變數是可以互動的。

七、x命令

也許,你很喜歡用p命令。所以,當你不知道變數名的時候,你可能會手足無措,因為p命令總是需要一個變數名的。x命令是用來檢視記憶體的,在gdb中 “help x” 你可以檢視其幫助。

x/x 以十六進位制輸出 x/d 以十進位制輸出 x/c 以單字元輸出 x/i  反彙編 – 通常,我們會使用 x/10i $ip-20 來檢視當前的彙編($ip是指令暫存器)x/s 以字串輸出 八、command命令

有一些朋友問我如何自動化除錯。這裡向大家介紹command命令,簡單的理解一下,其就是把一組gdb的命令打包,有點像字處理軟體的“巨集”。下面是一個示例:

12345678910(gdb) breakfuncBreakpoint 1 at 0x3475678: filetest.c, line 12.(gdb) command1Type commands forwhen breakpoint 1 is hit, one per line.End with a line saying just "end".>print arg1>print arg2>print arg3>end(gdb)

當我們的斷點到達時,自動執行command中的三個命令,把func的三個引數值打出來。

(全文完)

設定core環境
uname -a 檢視機器引數
ulimit -a 檢視預設引數
ulimit -c 1024  設定core檔案大小為1024
ulimit -c unlimit 設定core檔案大小為無限


  多執行緒如果dump,多為段錯誤,一般都涉及記憶體非法讀寫。可以這樣處理,使用下面的命令開啟系統開關,讓其可以在死掉的時候生成
core檔案。   
ulimit -c unlimited

執行緒除錯命令
(gdb)info threads 
顯示當前可除錯的所有執行緒,每個執行緒會有一個GDB為其分配的ID,後面操作執行緒的時候會用到這個ID。 
前面有*的是當前除錯的執行緒。

(gdb)thread ID 
切換當前除錯的執行緒為指定ID的執行緒。

(gdb)thread apply ID1 ID2 command 
讓一個或者多個執行緒執行GDB命令command。
(gdb)thread apply all command 
讓所有被除錯執行緒執行GDB命令command。

(gdb)set scheduler-locking off|on|step 
估計是實際使用過多執行緒除錯的人都可以發現,在使用step或者continue命令除錯當前被除錯執行緒的時候,其他執行緒也是同時執行的,怎麼只讓被除錯程式執行呢?通過這個命令就可以實現這個需求。 
off 不鎖定任何執行緒,也就是所有執行緒都執行,這是預設值。 
on 只有當前被除錯程式會執行。 
step 在單步的時候,除了next過一個函式的情況(熟悉情況的人可能知道,這其實是一個設定斷點然後continue的行為)以外,只有當前執行緒會執行。

//顯示執行緒堆疊資訊
(gdb) bt 
察看所有的呼叫棧


(gdb) f 3
呼叫框層次

(gdb) i locals  
顯示所有當前呼叫棧的所有變數