20150502 调试分析之 使用gdb远程调试ARM开发板

20150502 调试分析之 使用gdb远程调试ARM开发板

 

2015-05-02 Lover雪儿

 

今天我们要学习的是使用gdbgdbserver来远程调试开发板程序.

下面是本人的一些具体步骤:

下载gdb-7.9.tar.gz地址:

http://ftp.gnu.org/gnu/gdb/gdb-7.9.tar.gz

安装gdb

tar -jxvf gdb-7.9.tar.bz2

./configure -target=arm-none-linux-gnueabi --prefix=/home/study/nfs_home/rootfs_imx257/arm-linux-gdb –program-prefix=arm-none-linux-gnueabi-

make

make install

安装gdbserver

./configure -target=arm-none-linux-gnueabi --prefix=/home/study/nfs_home/rootfs_imx257/arm-linux-gdbserver --host=arm-none-linux-gnueabi

make

make install

具体安装步骤如下:

编译安装gdb

root@Lover:/home/study/nfs_home/system# tar -jxvf gdb-7.9.tar.bz2

root@Lover:/home/study/nfs_home/system# cd gdb-7.9

root@Lover:/home/study/nfs_home/system/gdb-7.9# mkdir ../../rootfs_imx257/arm-linux-gdb

root@Lover:/home/study/nfs_home/system/gdb-7.9#

./configure -target=arm-none-linux-gnueabi --prefix=/home/study/nfs_home/rootfs_imx257/arm-linux-gdb --program-prefix=arm-none-linux-gnueabi-

checking build system type... x86_64-unknown-linux-gnu

checking host system type... x86_64-unknown-linux-gnu

checking target system type... arm-none-linux-gnueabi

###########################

root@Lover:/home/study/nfs_home/system/gdb-7.9# make

root@Lover:/home/study/nfs_home/system/gdb-7.9# make install

安装完毕发现多了一些文件夹

root@Lover:/home/study/nfs_home/system/gdb-7.9# ll ../../rootfs_imx257/arm-linux-gdb/

总用量 24

drwxr-xr-x 6 root root 4096 52 06:34 ./

drwxrwxrwx 14 root root 4096 52 06:11 ../

drwxr-xr-x 2 root root 4096 52 06:34 bin/

drwxr-xr-x 3 root root 4096 52 06:34 include/

drwxr-xr-x 2 root root 4096 52 06:34 lib/

drwxr-xr-x 5 root root 4096 52 06:34 share/

root@Lover:/home/study/nfs_home/system/gdb-7.9#

root@Lover:/home/study/nfs_home/system/gdb-7.9# cd gdb

###########################

编译安装gdbserver

root@Lover:/home/study/nfs_home/system/gdb-7.9/gdb# cd gdbserver/

root@Lover:/home/study/nfs_home/system/gdb-7.9/gdb/gdbserver#

mkdir /home/study/nfs_home/rootfs_imx257/arm-linux-gdbserver

root@Lover:/home/study/nfs_home/system/gdb-7.9/gdb/gdbserver#

./configure -target=arm-none-linux-gnueabi --prefix=/home/study/nfs_home/rootfs_imx257/arm-linux-gdbserver --host=arm-none-linux-gnueabi

configure: WARNING: If you wanted to set the --build type, don't use --host.

If a cross compiler is detected then cross compile mode will be used.

checking whether to enable maintainer-specific portions of Makefiles... no

checking for arm-none-linux-gnueabi-gcc... arm-none-linux-gnueabi-gcc

checking for C compiler default output file name... a.out

###########################

root@Lover:/home/study/nfs_home/system/gdb-7.9# make

root@Lover:/home/study/nfs_home/system/gdb-7.9# make install


进入安装目录运行gdb查看版本

loverxueer@Lover:/home/study/nfs_home/rootfs_imx257$ ./arm-linux-gdb/bin/arm-none-linux-gnueabi-gdb

GNU gdb (GDB) 7.9

Copyright (C) 2015 Free Software Foundation, Inc.

License GPLv3+: GNU GPL version 3 or later <http://gnu.org/licenses/gpl.html>

This is free software: you are free to change and redistribute it.

There is NO WARRANTY, to the extent permitted by law. Type "show copying"

and "show warranty" for details.

This GDB was configured as "--host=x86_64-unknown-linux-gnu --target=arm-none-linux-gnueabi".

Type "show configuration" for configuration details.

For bug reporting instructions, please see:

<http://www.gnu.org/software/gdb/bugs/>.

Find the GDB manual and other documentation resources online at:

<http://www.gnu.org/software/gdb/documentation/>.

For help, type "help".

Type "apropos word" to search for commands related to "word".

(gdb) q

进入开发板目录,再开发板上运行gdbserver查看版本

root@EasyARM-iMX257 /mnt/nfs/rootfs_imx257# ./arm-linux-gdbserver/bin/arm-none-l

inux-gnueabi-gdbserver --version

GNU gdbserver (GDB) 7.9

Copyright (C) 2015 Free Software Foundation, Inc.

gdbserver is free software, covered by the GNU General Public License.

This gdbserver was configured as "arm-none-linux-gnueabi"

root@EasyARM-iMX257 /mnt/nfs/rootfs_imx257#

 

使用gdbgdbserver远程调试程序

 

附上应用程序gdb_test.c

 

 1 #include <stdio.h>
 2 
 3 char *p = "hello,this is test gdb & gdbserver ";
 4 
 5 void delay(int m)
 6 {
 7     int n;
 8     while(m--)
 9         for(n =0; n<100; n++);            
10 }
11 
12 void print(char *p, int cnt)
13 {
14     printf("%s cnt=%d\n",p,cnt);
15 }
16 
17 int  main(void)
18 {
19     int cnt = 0;
20     while(1)
21     {
22         cnt++;
23         print(p,cnt);
24         delay(100000);
25     }
26     return 0;
27 }
gdb_test.c

 

 

 

 

首先再开发板写好写程序.编译

root@Lover:/home/study/nfs_home/module/40_gdb_gdbserver_test# vi gdb_test.c

root@Lover:/home/study/nfs_home/module/40_gdb_gdbserver_test#

arm-none-linux-gnueabi-gcc -g gdb_test.c -o gdb_test

这里注意一定要加 -g 选项

进入我们安装的gdb的目录

root@Lover:/home/study/nfs_home/module/40_gdb_gdbserver_test#

root@Lover:/home/study/nfs_home/module/40_gdb_gdbserver_test#

cd /home/study/nfs_home/rootfs_imx257/arm-linux-gdb/bin/

root@Lover:/home/study/nfs_home/rootfs_imx257/arm-linux-gdb/bin# ./arm-none-linux-gnueabi-gdb

 

连接开发板,在开发板操作步骤如下:

 

root@EasyARM-iMX257 /mnt/nfs/rootfs_imx257#

cp arm-linux-gdbserver/bin/arm-none-linux-gnueabi-gdbserver /bin/

进入要运行的程序的目录

root@EasyARM-iMX257 /mnt/nfs/rootfs_imx257# cd ../module/40_gdb_gdbserver_test/

root@EasyARM-iMX257 /mnt/nfs/module/40_gdb_gdbserver_test# ls

gdb_test gdb_test.c make.sh

root@EasyARM-iMX257 /mnt/nfs/module/40_gdb_gdbserver_test#

获取开发板当前的ip

root@EasyARM-iMX257 /mnt/nfs/module/40_gdb_gdbserver_test# ifconfig

eth0 Link encap:Ethernet H

inet addr:192.168.31.181

Bcast:192.168.31.255 Mask:255.255.255.0

UP BROADCAST ric:1

RX packets:720 errors:0 dropped:0 overruns:0 frame:0

TX packets:105 errors:0 dropped:0 overruns:0 carrier:0

collisions:0 txqueuelen:1000

RX bytes:950466 (928.1 KiB) TX bytes:18426 (17.9 KiB)

Base address:0x2000

root@EasyARM-iMX257 /mnt/nfs/module/40_gdb_gdbserver_test#

开始运行gdbserver监控,监控端口随便写,此处为1234

root@EasyARM-iMX257 /mnt/nfs/module/40_gdb_gdbserver_test# arm-none-linux-gnueab

i-gdbserver 192.168.31.181:1234 ./gdb_test

Process ./gdb_test created; pid = 1816

Listening on port 1234

######################################################################################

当宿主机连接成功后

root@EasyARM-iMX257 /mnt/nfs/module/40_gdb_gdbserver_test# arm-none-linux-gnueab

i-gdbserver 192.168.31.181:1234 ./gdb_test

Process ./gdb_test created; pid = 1816

Listening on port 1234

Remote debugging from host 192.168.31.179 远程连接来自 192.168.31.179

######################################################################################

当宿主机敲完命令后

root@EasyARM-iMX257 /mnt/nfs/module/40_gdb_gdbserver_test# arm-none-linux-gnueab

i-gdbserver 192.168.31.181:1234 ./gdb_test

Process ./gdb_test created; pid = 1816

Listening on port 1234

Remote debugging from host 192.168.31.179

hello,this is test gdb & gdbserver cnt=1

hello,this is test gdb & gdbserver cnt=2

hello,this is test gdb & gdbserver cnt=3

 

接着在 宿主机上运行gdb来连接gdbserver,进行调试.

 

步骤如下:

 

注意gdb的目录

运行gdb

loverxueer@Lover:/home/study/nfs_home/rootfs_imx257$ ./arm-linux-gdb/bin/arm-none-linux-gnueabi-gdb

GNU gdb (GDB) 7.9

Copyright (C) 2015 Free Software Foundation, Inc.

Type "apropos word" to search for commands related to "word".

(gdb)

开始连接gdbserver,IP:端口号

(gdb) target remote 192.168.31.181:1234

Remote debugging using 192.168.31.181:1234

0x400007e0 in ?? ()

(gdb) l

没有符号表被读取。请使用 "file" 命令。

(gdb) c

######################################################################################

大家有没有发现,这样是连通了,也可以使用C命令运行,但是却无法使用listbreak命令

答案:

因为调试的信息是从本地的文件中读取的,所以,宿主目录和开发板要进入同一个程序目录下,进行调试

所以再开发部上进入我的程序目录

root@Lover:/home/study/nfs_home/rootfs_imx257/arm-linux-gdb/bin#

cd /home/study/nfs_home/module/40_gdb_gdbserver_test/

注意gdb的目录

loverxueer@Lover:/home/study/nfs_home/module/40_gdb_gdbserver_test$

/home/study/nfs_home/rootfs_imx257/arm-linux-gdb/bin/arm-none-linux-gnueabi-gdb

GNU gdb (GDB) 7.9

Copyright (C) 2015 Free Software Foundation, Inc.

(gdb) target remote 192.168.31.181:1234

Remote debugging using 192.168.31.181:1234

0x400007e0 in ?? ()

(gdb) l

没有符号表被读取。请使用 "file" 命令。

######################################################################################

(gdb) symbol-file ./gdb_test

加载本地目录下的文件的信号表

Reading symbols from ./gdb_test...done.

(gdb) l

5 void delay(int m)

6 {

7 int n;

8 while(m--)

9 for(n =0; n<100; n++);

10 }

11

12 void print(char *p, int cnt)

13 {

14 printf("%s cnt=%d\n",p,cnt);

(gdb) b 22

Breakpoint 1 at 0x83fc: file gdb_test.c, line 22.

(gdb)

(gdb) c

Continuing.


Breakpoint 1, main () at gdb_test.c:22

22 cnt++;

(gdb) c

Continuing.


Breakpoint 1, main () at gdb_test.c:22

22 cnt++;

(gdb) c

Continuing.


Breakpoint 1, main () at gdb_test.c:22

22 cnt++;

(gdb) c

Continuing.


Breakpoint 1, main () at gdb_test.c:22

22 cnt++;

(gdb) print cnt

$1 = 3

(gdb) target exec

A program is being debugged already. Kill it? (y or n) y

No executable file now.

(gdb)

此时再看我们的开发板的串口窗口

######################################################################################

######################################################################################


如下所示:

root@EasyARM-iMX257 /mnt/nfs/module/40_gdb_gdbserver_test# arm-none-linux-gnueab

i-gdbserver 192.168.31.181:1234 ./gdb_test

Process ./gdb_test created; pid = 1816

Listening on port 1234

Remote debugging from host 192.168.31.179

hello,this is test gdb & gdbserver cnt=1

hello,this is test gdb & gdbserver cnt=2

hello,this is test gdb & gdbserver cnt=3

Killing all inferiors

root@EasyARM-iMX257 /mnt/nfs/module/40_gdb_gdbserver_test#

 

总结命令:

 

开发板的命令:

root@EasyARM-iMX257 /mnt/nfs/module/40_gdb_gdbserver_test#

arm-none-linux-gnueabi-gdbserver 192.168.31.181:1234 ./gdb_test


宿主机的命令:

loverxueer@Lover:/home/study/nfs_home/module/40_gdb_gdbserver_test$

/home/study/nfs_home/rootfs_imx257/arm-linux-gdb/bin/arm-none-linux-gnueabi-gdb


(gdb) target remote 192.168.31.181:1234


(gdb) symbol-file ./gdb_test 注意这个是调试程序的文件所在

(gdb) l

(gdb) b 20

(gdb) c

(gdb) target exec

 

配置内核输出应用程序的段错误信息

 

 1 arch/arm/mm/fault.c
 2 
 3 static void
 4 __do_user_fault(struct task_struct *tsk, unsigned long addr,
 5         unsigned int fsr, unsigned int sig, int code,
 6         struct pt_regs *regs)
 7 {
 8     struct siginfo si;
 9 
10 #ifdef CONFIG_DEBUG_USER    //1.配置内核
11     if (user_debug & UDBG_SEGV) {
12         printk(KERN_DEBUG "%s: unhandled page fault (%d) at 0x%08lx, code 0x%03x\n",
13                tsk->comm, sig, addr, fsr);
14         show_pte(tsk->mm, addr);
15         show_regs(regs);
16     }
17 #endif
18 #############################################################
19 arch/alpha/kernel/traps.c
20 
21 2.uboot: set bootargs console=ttySAC0 user_debug=0xff   //全部输出debug信息

 

 

 

配置的具体步骤:

root@Lover:/home/study/nfs_home/system/linux-2.6.31# make menuconfig

输入/搜索 DEBUG_USER

结果:

Symbol: DEBUG_USER [=n] │

Prompt: Verbose user fault messages │

Defined at arch/arm/Kconfig.debug:29 │

Location: │

-> Kernel hacking │


首先选中

[*] Kernel debugging

[*] Verbose user fault messages //显示用户的错误信息

重新编译

root@Lover:/home/study/nfs_home/system/linux-2.6.31# make uImage

烧写内核:



烧写完毕后,重新启动

进入开发板的u-boot

原来的参数保留,增加一项 user_debug=0xff

set bootargs user_debug=0xff

bootm 80800000 启动内核


执行应用程序,则会打印全部的错误信息.

当发生错误的时候,打印出栈信息

 1 在arch/arm/mm/fault.c中增加
 2 
 3 static void
 4 __do_user_fault(struct task_struct *tsk, unsigned long addr,
 5         unsigned int fsr, unsigned int sig, int code,
 6         struct pt_regs *regs)
 7 {
 8     struct siginfo si;
 9 
10 #ifdef CONFIG_DEBUG_USER    //1.配置内核
11     if (user_debug & UDBG_SEGV) {
12         printk(KERN_DEBUG "%s: unhandled page fault (%d) at 0x%08lx, code 0x%03x\n",
13                tsk->comm, sig, addr, fsr);
14         show_pte(tsk->mm, addr);
15         show_regs(regs);
16     }
17 #endif
18 
19 int i=0;
20 ////////////////////////////////////////////////////////////
21     while(i<1024)
22     {
23         if(copy_from_user(&val, (const void __user *)(regs->ARM_sp + i*4),4))
24             break;
25         printk("%08x",val);
26         i++;
27         if(i % 8 == 0)
28             printk("\n");
29     }
30     printk("\n End of Statck\n");
31 /////////////////////////////////////////////////////////////

 


 

你可能感兴趣的:(远程调试)