利用GDB進行多執行緒除錯

2021-07-22 06:31:43 字數 4026 閱讀 9025

一、多執行緒除錯

多執行緒除錯重要就是下面幾個命令:

info thread 檢視當前程序的執行緒。 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下,我們無法print巨集定義,因為巨集是預編譯的。但是我們還是有辦法來除錯巨集,這個需要gcc的配合。

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

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

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

三、原始檔

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

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

下面給乙個除錯/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.

inls.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

四、條件斷點

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

五、命令列引數

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

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

六、gdb的變數

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

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

1

2

3

4

5

(gdb) set

$i = 0

(gdb) p a[$i++]

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

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

七、x命令

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

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

1

2

3

4

5

6

7

8

9

10

(gdb) break

func

breakpoint 1 at 0x3475678: file

test.c, line 12.

(gdb) command1

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的三個引數值打出來。

設定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)set scheduler-locking off|on|step 

估計是實際使用過多執行緒除錯的人都可以發現,在使用step或者continue命令除錯當前被除錯執行緒的時候,其他執行緒也是同時執行的,怎麼只讓被除錯程式執行呢?通過這個命令就可以實現這個需求。

off 不鎖定任何執行緒,也就是所有執行緒都執行,這是預設值。 

on 只有當前被除錯程式會執行。 

step 在單步的時候,除了next過乙個函式的情況(熟悉情況的人可能知道,這其實是乙個設定斷點然後continue的行為)以外,只有當前執行緒會執行。

//顯示執行緒堆疊資訊

(gdb) bt 

察看所有的呼叫棧

(gdb) f 3

呼叫框層次

(gdb) i locals  

顯示所有當前呼叫棧的所有變數

利用GDB進行多執行緒除錯

一 多執行緒除錯 多執行緒除錯重要就是下面幾個命令 info thread 檢視當前程序的執行緒。thread 切換除錯的執行緒為指定id的執行緒。break file.c 100 thread all 在file.c檔案第100行處為所有經過這裡的執行緒設定斷點。set scheduler loc...

利用GDB進行多執行緒除錯

一 多執行緒除錯 多執行緒除錯重要就是下面幾個命令 info thread 檢視當前程序的執行緒。thread 切換除錯的執行緒為指定id的執行緒。break file.c 100 thread all 在file.c檔案第100行處為所有經過這裡的執行緒設定斷點。set scheduler loc...

gdb多執行緒除錯

先介紹一下gdb多執行緒除錯的基本命令。info threads顯示當前可除錯的所有執行緒,每個執行緒會有乙個gdb為其分配的id,後面操作執行緒的時候會用到這個id。前面有 的是當前除錯的執行緒。thread id切換當前除錯的執行緒為指定id的執行緒。break thread test.c 12...