查看棧信息
當(dāng)程序被停住了,你需要做的第一件事就是查看程序是在哪里停住的。當(dāng)你的程序調(diào)用了一個(gè)函數(shù),函數(shù)的地址,函數(shù)參數(shù),函數(shù)內(nèi)的局部變量都會(huì)被壓入“棧”(Stack)中。你可以用GDB命令來(lái)查看當(dāng)前的棧中的信息。
下面是一些查看函數(shù)調(diào)用棧信息的GDB命令:
Backtrace,bt 打印當(dāng)前的函數(shù)調(diào)用棧的所有信息。如:
(gdb) bt
#0 func (n=250) at tst.c:6
#1 0x08048524 in main (argc=1, argv=0xbffff674) at tst.c:30
#2 0x400409ed in __libc_start_main () from /lib/libc.so.6
從上可以看出函數(shù)的調(diào)用棧信息:__libc_start_main --> main() --> func()
backtrace <n>, bt <n> n是一個(gè)正整數(shù),表示只打印棧頂上n層的棧信息。
backtrace <-n> ,bt <-n> -n表一個(gè)負(fù)整數(shù),表示只打印棧底下n層的棧信息。
如果你要查看某一層的信息,你需要在切換當(dāng)前的棧,一般來(lái)說(shuō),程序停止時(shí),最頂層的棧就是當(dāng)前棧,如果你要查看棧下面層的詳細(xì)信息,首先要做的是切換當(dāng)前棧。
frame <n>,f <n> n是一個(gè)從0開(kāi)始的整數(shù),是棧中的層編號(hào)。比如:frame 0,表示棧頂,frame 1,表示棧的第二層。
up <n> 表示向棧的上面移動(dòng)n層,可以不打n,表示向上移動(dòng)一層。
down <n> 表示向棧的下面移動(dòng)n層,可以不打n,表示向下移動(dòng)一層。
上面的命令,都會(huì)打印出移動(dòng)到的棧層的信息。如果你不想讓其打出信息。你可以使用這三個(gè)命令:
select-frame <n> 對(duì)應(yīng)于 frame 命令。
up-silently <n> 對(duì)應(yīng)于 up 命令。
down-silently <n> 對(duì)應(yīng)于 down 命令。
查看當(dāng)前棧層的信息,你可以用以下GDB命令:
frame 或 f 會(huì)打印出這些信息:棧的層編號(hào),當(dāng)前的函數(shù)名,函數(shù)參數(shù)值,函數(shù)所在文件及行號(hào),函數(shù)執(zhí)行到的語(yǔ)句。
info frame,info f 這個(gè)命令會(huì)打印出更為詳細(xì)的當(dāng)前棧層的信息,只不過(guò),大多數(shù)都是運(yùn)行時(shí)的內(nèi)內(nèi)地址。比如:函數(shù)地址,調(diào)用函數(shù)的地址,被調(diào)用函數(shù)的地址,目前的函數(shù)是由什么樣的程序語(yǔ)言寫成的、函數(shù)參數(shù)地址及值、局部變量的地址等等。如:
(gdb) info f
Stack level 0, frame at 0xbffff5d4:
eip = 0x804845d in func (tst.c:6); saved eip 0x8048524
called by frame at 0xbffff60c
source language c.
Arglist at 0xbffff5d4, args: n=250
Locals at 0xbffff5d4, Previous frame's sp is 0x0
Saved registers:
ebp at 0xbffff5d4, eip at 0xbffff5d8
info args 打印出當(dāng)前函數(shù)的參數(shù)名及其值。
info locals 打印出當(dāng)前函數(shù)中所有局部變量及其值。
info catch 打印出當(dāng)前的函數(shù)中的異常處理信息。