1. 查看文件中某變量的值:
file::variable
function::variable
可以通過這種形式指定你所想查看的變量,是哪個文件中的或是哪個函數中的。例如,查看文件f2.c中的全局變量x的值:
gdb) p 'f2.c'::x
查看數組的值
有時候,你需要查看一段連續的內存空間的值。比如數組的一段,或是動態分配的數據的大小。你可以使用GDB的“@”操作符,“@”的左邊是第一個內存的地址的值,“@”的右邊則你你想查看內存的長度。例如,你的程序中有這樣的語句:
int *array = (int *) malloc (len * sizeof (int));
于是,在GDB調試過程中,你可以以如下命令顯示出這個動態數組的取值:
p *array@len
如果是靜態數組的話,可以直接用print數組名,就可以顯示數組中所有數據的內容了。
2
.輸出格式
一般來說,GDB會根據變量的類型輸出變量的值。但你也可以自定義GDB的輸出的格式。例如,你想輸出一個整數的十六進制,或是二進制來查看這個整型變量的中的位的情況。要做到這樣,你可以使用GDB的數據顯示格式:
x 按十六進制格式顯示變量。
d 按十進制格式顯示變量。
u 按十六進制格式顯示無符號整型。
o 按八進制格式顯示變量。
t 按二進制格式顯示變量。
a 按十六進制格式顯示變量。
c 按字符格式顯示變量。
f 按浮點數格式顯示變量。
(gdb) p i
$21 = 101
(gdb) p/a i
$22 = 0x65
(gdb) p/c i
$23 = 101 'e'
(gdb) p/f i
$24 = 1.41531145e-43
(gdb) p/x i
$25 = 0x65
(gdb) p/t i
$26 = 1100101
3.查看內存
使用examine命令(簡寫是x)來查看內存地址中的值。x命令的語法如下所示:
x/
n、f、u是可選的參數。
n 是一個正整數,表示顯示內存的長度,也就是說從當前地址向后顯示幾個地址的內容。
f 表示顯示的格式,參見上面。如果地址所指的是字符串,那么格式可以是s,如果地十是指令地址,那么格式可以是i。
u 表示從當前地址往后請求的字節數,如果不指定的話,GDB默認是4個bytes。u參數可以用下面的字符來代替,b表示單字節,h表示雙字節,w表示四字節,g表示八字節。當我們指定了字節長度后,GDB會從指內存定的內存地址開始,讀寫指定字節,并把其當作一個值取出來。
n/f/u三個參數可以一起使用。例如:
命令:x/3uh 0x54320 表示,從內存地址0x54320讀取內容,h表示以雙字節為一個單位,3表示三個單位,u表示按十六進制顯示。
4. 查看堆棧: bt full
5.?
查看信息:info
info break 查看斷點信息
??? info locals 打印出當前函數中所有局部變量及其值
info stack 查看棧中信息
??? info frame 更詳細的棧層地址信息
info args 查看參數信息
info registers/info all-registers 查看(所有)寄存器信息
info sources 查看項目的源代碼信息
?
6. 多線程調試
?
多線程調試可能是問得最多的。其實,重要就是下面幾個命令:
- 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的行為)以外,只有當前線程會執行。
7、調試宏
這個問題超多。在GDB下,我們無法print宏定義,因為宏是預編譯的。但是我們還是有辦法來調試宏,這個需要GCC的配合。
在GCC編譯程序的時候,加上 -ggdb3 參數,這樣,你就可以調試宏了。
另外,你可以使用下述的GDB的宏調試命令?來查看相關的宏。
- info macro – 你可以查看這個宏在哪些文件里被引用了,以及宏定義是什么樣的。
- macro – 你可以查看宏展開的樣子。
?
8、源文件
這個問題問的也是很多的,太多的朋友都說找不到源文件。在這里我想提醒大家做下面的檢查:
- 編譯程序員是否加上了-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.
in
ls
.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
|
9、條件斷點
條件斷點是語法是:break ?[where] if [condition],這種斷點真是非常管用。尤其是在一個循環或遞歸中,或是要監視某個變量。注意,這個設置是在GDB中的,只不過每經過那個斷點時GDB會幫你檢查一下條件是否滿足。
10、命令行參數
有時候,我們需要調試的程序需要有命令行參數,很多朋友都不知道怎么設置調試的程序的命令行參數。其實,有兩種方法:
- gdb命令行的 –args 參數
- gdb環境中 set args命令。
11、gdb的變量
有時候,在調試程序時,我們不單單只是查看運行時的變量,我們還可以直接設置程序中的變量,以模擬一些很難在測試中出現的情況,比較一些出錯,或是switch的分支語句。使用set命令可以修改程序中的變量。
另外,你知道gdb中也可以有變量嗎?就像shell一樣,gdb中的變量以$開頭,比如你想打印一個數組中的個個元素,你可以這樣:
1
2
3
4
5
|
(gdb)
set
$i = 0
?
(gdb) p a[$i++]
?
... ?
#然后就一路回車下去了
|
當然,這里只是給一個示例,表示程序的變量和gdb的變量是可以交互的。
12、x命令
也許,你很喜歡用p命令。所以,當你不知道變量名的時候,你可能會手足無措,因為p命令總是需要一個變量名的。x命令是用來查看內存的,在gdb中 “help x” 你可以查看其幫助。
- x/x 以十六進制輸出
- x/d 以十進制輸出
- x/c 以單字符輸出
-
x/i ?反匯編 – 通常,我們會使用
?
x/10i $ip-20 來查看當前的匯編($ip是指令寄存器)
- x/s 以字符串輸出
13、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)
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的三個參數值打出來。
14.
(gdb) help disassemble
?
Disassemble a specified section of memory.
Default is the function surrounding the pc of the selected frame.
With a /m modifier, source lines are included (if available).
??????? 支持的版本可以打出每行源代碼和匯編對應關系
With a /r modifier, raw instructions in hex are included.
With a single argument, the function surrounding that address is dumped.
Two arguments (separated by a comma) are taken as a range of memory to dump,
? in the form of "start,end", or "start,+length".
更多文章、技術交流、商務合作、聯系博主
微信掃碼或搜索:z360901061

微信掃一掃加我為好友
QQ號聯系: 360901061
您的支持是博主寫作最大的動力,如果您喜歡我的文章,感覺我的文章對您有幫助,請用微信掃描下面二維碼支持博主2元、5元、10元、20元等您想捐的金額吧,狠狠點擊下面給點支持吧,站長非常感激您!手機微信長按不能支付解決辦法:請將微信支付二維碼保存到相冊,切換到微信,然后點擊微信右上角掃一掃功能,選擇支付二維碼完成支付。
【本文對您有幫助就好】元
