arm-linux的gdb移植

摘要:
还有一种是干脆把整个gdb移植成一个ARM的本地版。它gdbserver可以通过arm-linux-gdb直接在host上单步调试target的应用程序。如何交叉调试arm-linux的本地版一般只能做字符界面的。这里移植是gdb6.8.编译器是arm-linux-gcc4.3.3.操作系统是arm-linux2.30.4ncurse5.6。

转载于:http://blog.chinaunix.net/uid-23381466-id-309369.html

arm-linux的gdb移植分为两种情况.一种是交叉调试版。这一种模式是需要编译一个arm-linux版本gdbserver (GDB的stub模块).然后再编译一个X86版本交叉调试的gdb.为了与桌面版本身的gdb 区别开来,一般改名为 arm-linux-gdb。两者通过串口或者网络进行互联。
还有一种是干脆把整个gdb移植成一个ARM的本地版。在开发板上直接用gdb来调试。
前一种方法是比较正统的方法。它gdbserver可以通过arm-linux-gdb直接在host上单步调试target的应用程序。.并且可以与图形界面调试器配合进行图形界面调试。缺点就是target资源较少。因此单步调试的速度并不是太快。因此实用性不算太强。
如何交叉调试
arm-linux的本地版一般只能做字符界面的。界面没有支持GUI的交叉调试版友好。而且单步调试速度也不算快。但是有几大大优点:
1.定位程序退出所在函数.
2.判断程序退出的原因
3.通过条件断点分析异常情况时运行环境.
用本地版gdb运行程序时,当程序因段错误或其它原因退出程序时。可以通过gdb bt(即backtrace)来查看最后运行的堆栈。来判断出错时是在哪一个函数里退出。这样会大大加快定位错误的速度,这样有时需要几天的定位的错误,可能只需要几分钟即可定位,这在嵌入式开发里有很强实际效用。
因此在这里我们把两种版本的gdb都移植一下。

---------------------------------------------------------------------------------------------
1.本地版本的gdb的移植
ARM本地版需要ncurse的支持。这里移植是gdb 6.8.
编译器是 arm-linux-gcc 4.3.3 .
操作系统是 arm-linux 2.30.4
ncurse 5.6 。
按照移植规范,ncurse放入libs目录,gdb 解压在项目目录下
/home/hxy/gdb (项目目录)
|
+--output
| |_arm-linux
|-- gdb 6.8|
|--
|--libs
|-- ncurse
1.1ncurse库的移植
解压 tar xvzf ncurses-5.6.tar.gz
cd ncurses-5.6
生成makefile
./configure --host=arm-linux --prefix=$PWD/output/arm --without-ada --enable-termcap --with-shared
其中--enable-termcap比较关键gdb需要这个库
编译 make
安装 make install
libncurses.so 库应该在/home/hxy/gdb/output/arm-linux/lib
1.2 gdb本身的移植。
解压 tar xvjf gdb-6.8.tar.bz2
cd gdb-6.8
生成Makefile
./configure --host=arm-linux --enable-shared --prefix=$PWD/output/arm --without-x --disable-gdbtk --disable-tui --without-included-regex --without-included-gettext LDFLAGS="-L$PWD/../output/arm-linux/lib" CPPFLAGS="-I$PWD/../output/arm-linux/include"
脚本含义下:
--enable-shared 动态编译
--host=arm-linux 用arm-linux-gcc编译
--prefix="$PWD/../output/arm-linux" 安装目录
--without-x 取消x windows 支持
--disable-gdbtk 取消gdbtk,应该也是图形界面相关的
--disable-tui 取消tui 界面
--without-included-regex 关闭正则表达式库
--without-included-gettext 去掉用于多语言处理的 gettext库
正则表达式/gettext,暂时不需要,先去掉
CPPFLAGS/LDFLAGS是确保能找到ARM版的ncurses库
编译 make
如果gdb 6.7有一个警告会当成bug处理symtab.c: In function 'find_line_symtab':
symtab.c:2252: error: 'exact' may be used uninitialized in this function
只要简单把int exact;变成int exact =0 ; 即可编译通过
补充:在arm-linux-gcc 4.4.1 (s3c6410)下有如下编译错误 eval.c: In function 'evaluate_subexp_standard':
eval.c:1705: error: 'subscript_array' may be used uninitialized in this function
分析源码发现,是编译器认为这个数组定义后未初始化造成的(可能是bug).因此在在eval.c::1650 行加入进入初始操作.如此编译通过
memset(subscript_array,0,sizeof(int)*MAX_FORTRAN_DIMS);
安装 make install
最终的arm的程序gdb应该在home/hxy/gdb/output/arm-linux/bin下面,将其拷贝和libncurses.so拷贝到开发板上即可运行,象x86的GDB一样运行即可
注意这个样编译能同时把ARM版gdbserver 也同时编译出来并在安装目录看到.而且一般eabi的gcc编译器都自带了arm-linux-gdb.所以交叉版本的gdb移植很多时候可以省略.
刚刚编译出来的gdb尺寸相当大,10M多,因此必须要用arm-linux-strip gdb 来把尺寸减少.strip后大约2M多.

---------------------------------------------------------------------------------------------
2.交叉版本的gdb移植
交叉版本中,arm-linux-gcc 3.3.2 只能成功编译gdb 5.2.1.gdb 6.x 需要更gcc 3.4.4以上版本。
eabi arm-linux-gcc 4.3.3 编译可以成功编译gdb 6.8.
因为交叉版等于要移植两个平台程序.(x86的arm-linux-gdb和arm 版的gdb server)
这里目录结构调整如下
/home/hxy/gdb
|
+--output
| |_arm-linux
| |_x86-linux
|-- gdb 6.8
| |-- cross-gdb #保存x86 gdb
| |- -gdb/gdbserver #原有目录,在此编译arm gdbserver,
|
|--libs
|-- ncurse
因其中cross-gdb是手工创建为了存放x86的目标代码
2.1编译 cross 调试的gdb 6.8
在/home/hxy/gdb/gdb-6.8 清除上一次结果 make distclean
创建/home/hxy/gdb/gdb-6.8/cross-gdb
在上述目录生成Makefile
../configure --target=arm-linux --enable-shared --prefix=$PWD/../../output/x86-linux --without-x --disable-gdbtk --without-included-regex --without-include-gettext
其中大部分参数跟本地版gdb含意类似。但是--target=arm-linux 表示target是arm-linux版的。而且是安装在X86-linux下
编译 make
安装 make install
应该是安装在/home/hxy/gdb/output/x86-linux/bin/,名字是arm-linux-gdb
2.2生成arm版gdbserver
生成Makefile 在/home/hxy/gdb/gdb-6.8/gdb/gdbserver执行如下脚本
./configure --host=arm-linux --prefix=$PWD/../../../output/arm-linux --without-include-regex --without-included-gettext
编译make
安装 make install
这里应该在/home/hxy/gdb/output/arm-linux/bin有gdbserver
至于如何使用交叉调试参见相关博文,附件是已经编译好直接在eabi库环境下使用的版本 文件:arm-gdb-6.x.zip

---------------------------------------------------------------------------------------------
3.测试ARM本机调试
我们用一个有段错误的源码来测试本地调试的性能,参见如下代码,这里在f2()产生段错误
/*
* Andrew Huang <bluedrum@163.com>
*/
#include <stdio.h>
#include <string.h>
void f2(char * str)
{
char buf[1024];
strcpy(buf,(const char *)str);
}
void f1()
{
int x = 0;
f2((char *)x);
}
void test1()
{
f1();
}
int main()
{
test1();
}
编译 arm-linux-gcc test.c -o test -g
在NFS路径上测试它的调用gdb和test.速度相当快。以下是它的测试结果
[root@EmbedSky bin]# ./run.sh
GNU gdb 6.8
Copyright (C) 2008 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 "arm-linux"...
(gdb) r
Starting program: /mnt/nfs/gdb/output/arm-linux/bin/test
Program received signal SIGSEGV, Segmentation fault.
0x4009852c in strcpy () from /lib/libc.so.6
(gdb) bt
#0 0x4009852c in strcpy () from /lib/libc.so.6
#1 0x000083a8 in f2 (str=0x0) at test.c:8
#2 0x000083d4 in f1 () at test.c:14
#3 0x000083ec in test1 () at test.c:19
#4 0x00008408 in main () at test.c:24
(gdb)
可以看到很快能测试出段错语的位置。我也试了一下单步调试程序,发现居然比交叉调试速度还快。这个倒出乎我的意料之外

免责声明:文章转载自《arm-linux的gdb移植》仅用于学习参考。如对内容有疑问,请及时联系本站处理。

上篇GCD与莫比乌斯反演的勾当HTTP.SYS远程代码执行漏洞测试(ms15-034)下篇

宿迁高防,2C2G15M,22元/月;香港BGP,2C5G5M,25元/月 雨云优惠码:MjYwNzM=

相关文章

Redis在Linux下的安装

一、下载地址 ①redis中文网下载地址:http://www.redis.cn/ ②百度云网盘下载地址:https://pan.baidu.com/s/1UQcF9V3lwA0fxquM_JFMZw 提取码:lnwk 二、编译软件安装 yum -y install gcc gcc-c++ make 三、安装 解压后进入目录,编译安装命令: make ma...

LINUX-文件的特殊属性

chattr +a file1 只允许以追加方式读写文件 chattr +c file1 允许这个文件能被内核自动压缩/解压 chattr +d file1 在进行文件系统备份时,dump程序将忽略这个文件 chattr +i file1 设置成不可变的文件,不能被删除、修改、重命名或者链接 chattr +s file1 允许一个文件被安全地删除 cha...

LINUX进程组调度机制分析【转】

转自:https://oenhan.com/task-group-sched 又碰到一个神奇的进程调度问题,在系统重启过程中,发现系统挂住了,过了30s后才重新复位,真正系统复位的原因是硬件看门狗重启的系统,而非原来正常的reboot流程。硬件狗记录的复位时间,将不喂狗的时间向前推30s分析串口记录日志,当时的日志就打印了一句话:“sched: RT th...

Linux shell脚本,按顺序批量启动多个jar包,批量启动spring cloud的jar包

Linux shell脚本,按顺序批量启动多个jar包,批量启动spring cloud的jar包 一. 手动一个一个启动的方式: nohup java -jar eurekaserver.jar > ../logs/eurekaserver.log 2>&1 & nohup java -jar configserver.jar...

(转载)linux中shell变量

(转载)http://blog.csdn.net/zahuopuboss/article/details/8633891 为使shell编程更有效,系统提供了一些shell变量。shell变量可以保存诸如路径名、文件名或者一个数字这样的变量名。shell将其中任何设置都看做文本字符串。有两种变量,本地和环境。严格地说可以有4种,但其余两种是只读的,可以认为...

linux安装后需要进行的一些基本设置

修改网络: 在终端中输入:vi /etc/sysconfig/network-scripts/ifcfg-ens33    然后重启网络服务:systemctl restart network.service 更新该Liunx系统的内核版本:   yum update 关闭防火墙:   1:查看防火状态       systemctl status fir...