应用调试(二)GDB

时间:2024-10-23 19:37:50

title: 应用调试(二)GDB
date: 2019/1/17 21:00:10

toc: true

应用调试(二)GDB

gdb下载工具安装交叉工具链设置GDB介绍编译GDBtarget/host/build编译主机GDB编译单板GDBserver调试程序CoreDumpGDB命令杂项附录源码

gdb下载

官网

这里我下载了gdb-7.6.tar.gz编译成功,下载了8.x的需要c++11,后来没有试了

里面的".sig"为文件名的分离签名文件,用来校验下载资源的完整性,.xz为另外格式的压缩包

工具安装

# 高版本的gdb需要这个东西
sudo apt-get install texinfo

交叉工具链设置

  • --program-prefix=arm-linux-是指生成的可执行文件的前缀,比如arm-linux-gdb
  • -prefix是指生成的可执行文件安装在哪个目录
  • 可以指定工具连 make CC=arm-none-linux-gnueabi-gcc https://blog.****.net/hanq4998/article/details/84675718

GDB介绍

GDB调试分为两个部分

  • 单板运行gdbserver,作为主进程,他来创建所调试程序为子进程,使用系统调用ptrace跟踪
  • 主机运行arm-linux-gdb,调试远程的单板程序

应用调试(二)GDBmark

编译GDB

target/host/build

这里的这些其实就是编译工具的指定

  • build
  • host 编译出来的程序在哪里运行,其实就是编译工具
  • target 这个选项一般为编译器所拥有,一般的程序不需要,意思是编译出来的编译器它编译的东西在哪里运行

应用调试(二)GDBmark

编译主机GDB

具体流程如下

tar xzf gdb-7.6.tar.gz
cd gdb-7.6/
# 忽略这种函数命名不规范导致的错误
./configure --target=arm-linux --disable-werror
make
mkdir tmp 
make install prefix=$PWD/tmp
sudo cp tmp/bin/arm-linux-gdb  /usr/bin
# arm-linux-gdb -v
GNU gdb (GDB) 7.6

一些注意点

  • 编译出的运行在ubuntu下的GDB需要调试arm下的程序,这里需要指定target=arm-linux

  • --disable-werror 是百度到的,不然接下去make失败

  • 使用arm-linux-gcc version 3.4.5编译

  • 先安装到当前目录的tmp下,可以通过生成的Makefie看到默认的路径是usr/local

    build_alias=x86_64-unknown-linux-gnu
    build_vendor=unknown
    build_os=linux-gnu
    build=x86_64-unknown-linux-gnu
    host_alias=x86_64-unknown-linux-gnu
    host_vendor=unknown
    host_os=linux-gnu
    host=x86_64-unknown-linux-gnu
    target_alias=arm-linux
    target_vendor=unknown
    target_os=linux-gnu
    target=arm-unknown-linux-gnu

编译单板GDBserver

命令如下

cd gdb/gdbserver/
./configure  --host=arm-linux
make
# 这里需要修改 vi linux-arm-low.c
make #因为这gdbserver是要放在开发板上的,就不用make install了 #查看下生成的文件
book@100ask:~/stu/gdb/gdb-7.6/gdb/gdbserver$ file gdbserver
gdbserver: ELF 32-bit LSB executable, ARM, version 1, dynamically linked, interpreter /lib/ld-linux.so.2, for GNU/Linux 2.4.3, not stripped
# 复制到单板
cp gdbserver ~/stu/code
  • gdbserver调试的程序是运行在arm下的,也就是指定host,这里应该不需要指定host,我测试了一下,生成的执行文件crc一致

  • 修该代码,这里make会有如下错误提示找不到宏定义,我们需要去交叉编译工具链的目录去搜索

    linux-arm-low.c: In function arm_stopped_by_watchpoint':
    linux-arm-low.c:643: error:
    PTRACE_GETSIGINFO' undeclared (first use in this function) # 查看环境变量
    echo 
    不能识别此Latex公式:
    PATH
    /home/book/bin:/home/book/.local/bin:/opt/slickedit-pro2017/bin:/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin:/usr/games:/usr/local/games:/opt/gcc-3.4.5-glibc-2.3.6/bin:/snap/bin
    cd /opt/gcc-3.4.5-glibc-2.3.6/ #搜索宏  PTRACE_GETSIGINFO
    book@100ask:/opt/gcc-3.4.5-glibc-2.3.6
     grep "PTRACE_GETSIGINFO" * -nR
    arm-linux/include/linux/ptrace.h:27:#define PTRACE_GETSIGINFO   0x4202
    arm-linux/sys-include/linux/ptrace.h:27:#define PTRACE_GETSIGINFO       0x4202
    distributed/arm-linux/include/linux/ptrace.h:27:#define PTRACE_GETSIGINFO       0x4202
    distributed/arm-linux/sys-include/linux/ptrace.h:27:#define PTRACE_GETSIGINFO   0x4202 # 修改代码 加入 #include<linux/ptrace.h>
    vi linux-arm-low.c

调试程序

  1. 编译程序,加入-g选项,形如arm-linux-gcc -g -o test test.c

  2. 单板开启gdbserver,命令形如gdbserver 192.168.95.2345 ./test,这IP是单板的IP,端口随意

  3. 主机远程连接到单板

    arm-linux-gdb ./test
        target remote 192.168.95.222:2345
  4. 接下去就可以调试了,注意这里打断点到哪一行,该行并未执行,这和我们ide是一样的,并且使用step等命令显示的下一句c代码,实际上也是没有执行的

    应用调试(二)GDBmark

  5. 我们这个程序故意引入空指针,所以最后会有错误提示

    Program received signal SIGSEGV, Segmentation fault.
    0x000084ac in C (p=0x0) at test.c:6
    6               *p = 0x12;

CoreDump

这个类似一个日志文件,首先需要在单板上开启这个功能

ulimit -c unlimited
# ./test
a = 0x12
Segmentation fault (core dumped) # 复制这个 core 文件去主机
# 我们这里本身就在主机nfs上

直接使用这个文件在主机上运行,这里可能需要加权限,直接chmod 777 core

book@100ask:~/stu/code$ sudo chmod 777  core
book@100ask:~/stu/code$ arm-linux-gdb ./test core
GNU gdb (GDB) 7.6
Copyright (C) 2013 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-linux".
For bug reporting instructions, please see:
<http://www.gnu.org/software/gdb/bugs/>...
Reading symbols from /home/book/stu/code/test...done.
[New LWP 812] warning: `/lib/ld-linux.so.2': Shared library architecture unknown is not compatible with target architecture arm. warning: Could not load shared library symbols for /lib/libc.so.6.
Do you need "set solib-search-path" or "set sysroot"?
Core was generated by `./test'.
Program terminated with signal 11, Segmentation fault.
#0  0x000084ac in C (p=0x0) at test.c:6
6               *p = 0x12;

可以看到出错的地方

直接使用命令bt查看调用树

(gdb) bt
#0  0x000084ac in C (p=0x0) at test.c:6
#1  0x000084d0 in B (p=0x0) at test.c:12
#2  0x000084f0 in A (p=0x0) at test.c:17
#3  0x0000856c in main (argc=1, argv=0xbedb6ed4) at test.c:38

GDB命令

http://www.cnblogs.com/veryStrong/p/6240775.html

http://blog.sciencenet.cn/blog-619295-813770.html

名称 命令 描述
断点 break main  
  break test.c:20 在test.c的20行 中间不能有空格
继续 c continue
单步进入 step 进入函数
单步跳过 next 不会进入函数
打印变量 print a 打印a变量

杂项

FAQ

这里我试了在编译gdb的时候,最后安装的时候出现gdb.info出错,但实际上arm-linux-gdb已经生成了,好像是可以用的了

一些命令

#查看文件属性
file gdbserver #链接文件
sudo mv ./gdb ./gdb.old
cd /usr/local/gdb8/bin
sudo ln -s $(pwd)/gdb /usr/bin/gdb

附录源码

#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);   printf("step1");
  printf("step2");
  printf("step3");     A(p);    // A > B > C     return 0;
}