GDB除錯方法精粹

2021-07-07 09:33:40 字數 4119 閱讀 7730

除錯命令

說明info threads

檢視當前程序的執行緒。 gdb會為每個執行緒分配乙個id, 後面操作執行緒的時候會用到這個id. 前面有*的是當前除錯的執行緒.

thread

切換除錯的執行緒為指定id的執行緒。

break file.c:100 thread all

在file.c檔案第100行處為所有經過這裡的執行緒設定斷點。

set scheduler-locking off/on/step

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

讓乙個或者多個執行緒執行gdb命令command

讓所有被除錯執行緒執行gdb命令command。

執行緒產生通知:在產生新的執行緒時, gdb會給出提示資訊

(gdb) r

starting program: /root/thread

[new thread 1073951360 (lwp 12900)]

[new thread 1082342592 (lwp 12907)]---以下三個為新產生的執行緒

[new thread 1090731072 (lwp 12908)]

[new thread 1099119552 (lwp 12909)]

檢視執行緒:使用info threads可以檢視執行的執行緒。

(gdb) info threads

4 thread 1099119552 (lwp 12940) 0xffffe002 in ?? ()

3 thread 1090731072 (lwp 12939) 0xffffe002 in ?? ()

2 thread 1082342592 (lwp 12938) 0xffffe002 in ?? ()

* 1 thread 1073951360 (lwp 12931) main (argc=1, argv=0xbfffda04) at thread.c:21

(gdb)

注意,行首為gdb分配的執行緒id號,對執行緒進行切換時,使用該id號碼。

另外,行首的星號標識了當前活動的執行緒

切換執行緒:

使用 thread threadnumber 進行切換,threadnumber 為上文提到的執行緒id號。

下例顯示將活動執行緒從 1 切換至 4。

(gdb) info threads

4 thread 1099119552 (lwp 12940) 0xffffe002 in ?? ()

3 thread 1090731072 (lwp 12939) 0xffffe002 in ?? ()

2 thread 1082342592 (lwp 12938) 0xffffe002 in ?? ()

* 1 thread 1073951360 (lwp 12931) main (argc=1, argv=0xbfffda04) at thread.c:21

(gdb) thread 4

[switching to thread 4 (thread 1099119552 (lwp 12940))]#0 0xffffe002 in ?? ()

(gdb) info threads

* 4 thread 1099119552 (lwp 12940) 0xffffe002 in ?? ()

3 thread 1090731072 (lwp 12939) 0xffffe002 in ?? ()

2 thread 1082342592 (lwp 12938) 0xffffe002 in ?? ()

1 thread 1073951360 (lwp 12931) main (argc=1, argv=0xbfffda04) at thread.c:21

(gdb)

以上即為使用gdb提供的對多執行緒進行除錯的一些基本命令。

另外,gdb也提供對執行緒的斷點設定以及對指定或所有執行緒發布命令的命令

在gdb下, 我們無法print巨集定義,因為巨集是預編譯的。

但是我們還是有辦法來除錯巨集,這個需要gcc的配合。

在gcc編譯程式的時候,加上

-ggdb3 引數,這樣,你就可以除錯巨集了。

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

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

macro 檢視巨集展開的樣子。

gdb時,提示找不到原始檔。

需要做下面的檢查:

編譯程式員是否加上了 -g引數 以包含debug資訊。

路徑是否設定正確了。

使用gdb的directory命令來設定原始檔的目錄。

$ apt-get source coreutils

$ sudo apt-get install coreutils-dbgsym

$ gdb /bin/ls

gnu gdb (gdb) 7.1-ubuntu

(gdb) list main

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}}

intmain (int argc, char **argv)

{int i;

struct pending *thispend;

int n_files;

條件斷點是語法是:

break  [where] if [condition]
這種斷點真是非常管用。

尤其是在乙個迴圈或遞迴中,或是要監視某個變數。

注意,這個設定是在gdb中的,只不過每經過那個斷點時gdb會幫你檢查一下條件是否滿足。

有時候,我們需要除錯的程式需要有命令列引數, 有三種方法:

gdb命令列的 -args 引數

gdb環境中 set args命令。

gdb環境中 run 引數

有時候,在除錯程式時,我們不單單只是檢視執行時的變數,

我們還可以直接設定程式中的變數,以模擬一些很難在測試中出現的情況,比較一些出錯,

或是switch的分支語句。使用set命令可以修改程式中的變數。

另外,你知道gdb中也可以有變數嗎?

就像shell一樣,gdb中的變數以$開頭,比如你想列印乙個陣列中的個個元素,你可以這樣:

(gdb) set $i = 0

(gdb) p a[$i++]

... #然後就一路回車下去了

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

x/x 以十六進製制輸出

x/d 以十進位制輸出

x/c 以單字元輸出

x/i 反彙編 – 通常,我們會使用 x/10i $ip-20 來檢視當前的彙編($ip是指令暫存器)

x/s 以字串輸出

(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)

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

GDB除錯精粹及使用例項

gdb除錯精粹及使用例項 一 列檔案清單 1 list gdb list line1,line2 二 執行程式 要想執行準備除錯的程式,可使用run命令,在它後面可以跟隨發給該程式的任何引數,包括標準輸入和標準輸出說明符 和外殼萬用字元 在內。如果你使用不帶引數的run命令,gdb就再次使用你給予前...

GDB除錯精粹及使用例項

一 列檔案清單 1 list gdb list line1,line2 二 執行程式 要想執行準備除錯的程式,可使用run命令,在它後面可以跟隨發給該程式的任何引數,包括標準輸入和標準輸出說明符 和外殼萬用字元 在內。如果你使用不帶引數的run命令,gdb就再次使用你給予前一條run命令的引數,這是...

GDB除錯精粹及使用例項

一 列檔案清單 1 list gdb list line1,line2 二 執行程式 要想執行準備除錯的程式,可使用run命令,在它後面可以跟隨發給該程式的任何引數,包括標準輸入和標準輸出說明符 和 和外殼萬用字元 在內。如果你使用不帶引數的run命令,gdb就再次使用你給予前一條run命令的引數,...