40.Linux應用調試-使用gdb和gdbserver


1.gdb和gdbserver調試原理

通過linux虛擬機里的gdb,來向開發板里的gdbserver發送命令,比如設置斷點,運行setp等,然后開發板上的gdbserver收到命令后,便會執行應用程序做相應的動作,來實現調試的功能

和之前學的裸板GDB調試 一樣,只不過之前學的是在win下的,本次是在linux里的gdb

1.1同樣,它們都會需要一個帶調試信息的編譯文件.

通過Makefile里的arm-linux-gcc  -g 來的,     -g:表示編譯文件里包含gdb調試信息

1.2為什么需要調試信息的編譯文件?

比如讀開發板的應用程序里的變量a:

首先gdb通過應用程序的帶調試信息的編譯文件,來找出變量a存的地址位置

然后將地址發送給開發板里的gdbserver,來讀出a地址的值

 

2.安裝gdb和gdbserver

首先進入官網下載gdb-7.4: http://ftp.gnu.org/gnu/gdb/ 

2.1在虛擬機上安裝GDB:  

# tar xjf gdb-7.4.tar.bz2                  //解壓
 # cd gdb-7.4/                              //進入gdb-7.4目錄
 #./configure --target=arm-linux //GDB需要在pc本機里運行,並調試開發板里的應用程序,所以--target設為arm-linux
 #make //編譯
 #mkdir tmp #make install prefix=$PWD/tmp              //安裝到./tmp目錄下
 sudo cp tmp/bin/arm-linux-gdb  /bin/       //復制到/bin目錄下

/bin/arm-linux-gdb  -v                     //-v: 確定一下gdb的版本VID,是否是7.4

2.2 在開發板上安裝GDBServer:

cd gdb/gdbserver/                                   //在gdb-7.4目錄下輸入
 ./configure --target=arm-linux --host=arm-linux              //設GDBServer的工作環境
 make //編譯

出現以下錯誤:

指在linux-arm-low.c里,沒有找到PTRACE_GETSIGINFO 定義

2.3 解決:

1) 

#echo $PATH     //來查看PATH環境變量

找到編譯器gcc位於/work/tools/gcc-3.4.5-glibc-2.3.6/bin

2)

#cd /work/tools/gcc-3.4.5-glibc-2.3.6/     # grep "PTRACE_GETSIGINFO"  *  -nR

在gcc根目錄下,搜索到在linux/ptrace.h中定義:

 

3)

#vi linux-arm-low.c

添加: #define PTRACE_GETSIGINFO 0x4202

4)最后重新make,生成gdbserver命令文件

然后將gdbserver命令文件,放入我們開發板的根目錄/bin中,便能使用了

cp  gdbserver  /nfs_root/bin/          //nfs_root:開發板的nfs系統根目錄

3.測試程序如下(test_debug.c)

#include <stdio.h>

void  C(int *p) {    *p = 0x12; } void  B(int *p) {   C(p); } void  A(int *p) {   B(p); } void  A2(int *p) {   C(p); } int  main(int argc, char **argv) {   int a;   int *p = NULL;   A2(&a);  // A2 > C   printf("a = 0x%x\n", a);   A(p); // A > B > C
  return 0; }

其中A2(&a)會調用A2()->C(),然后將a賦值為0x12.

A(p)會調用A()->B()->C(),由於p是個空指針,這里將會出錯.

接下來,我們便以這個應用程序為例.

4.編譯

#arm-linux-gcc -g -o test_debug test_debug.c   //-g:附帶調試信息

5.調試test_debug.c

在開發板上:

首先,需要讓gdbserver建立本地服務器,以及要測試的哪個文件:

#gdbserver 192.168.2.107:2345 ./test_debug //192.168.2.107:本地IP地址 //2345:端口號,用來讓gdb來連接用的 //./test_debug:要測試的哪個文件

在虛擬機上:

#/bin/arm-linux-gdb   ./test_debug    // 啟動gdb,指定調試文件為test_debug
 #target remote 192.168.2.107:2345    //與gdbserver建立連接

5.1連接成功,便使用gdb命令來調試

常用命令如下所示(參考http://blog.sciencenet.cn/blog-619295-813770.html):

l

列出所有源代碼

break [file]:[row]

打斷點,比如:

break test_debug.c:21     //在test_debug.c文件的第21行處打斷點

info br

查看斷點

 

info file

 

列出當前的文件,共享庫。

 

delete <num>

刪除第幾個斷點,如下圖所示:

 

c

啟動程序運行 

step

單步執行

next  

單步執行,和step不同的是,比如:當前行里有函數調用時,next直接執行下一句,step會進入函數

print a

打印a變量的值

quit

退出gdb

 

6.也可以通過gdb+coredump來調試test_debug.c

當程序運行出錯時,便會生成core文件,並將程序里的運行狀況存到core中,也就是coredump,供給gdb來調試

6.1首先,通過ulimit來查看coredump的資源大小     

ulimit命令(user limit),主要用來限制用戶的各個進程資源.

在開發板里,輸入

 

如上圖所示,可以看到coredump的資源大小為0,也就是說,當程序運行出錯時,不會生成core文件

6.2設置core文件

設置core文件的資源大小為無限制,輸入:

ulimit -c unlimited
 //-c:對應coredump

6.3生成core文件

執行:

#./test_debug

出現段錯誤,並生成core文件,如下圖所示:

 

6.4 進入虛擬機

core拷貝過來,然后執行:

#/bin/arm-linux-gdb ./test_debug ./core

然后輸入bt,便可查看調用關系:

 

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM