在VIM中的嵌入式软件调试

发布者:老实巴交的大叔最新更新时间:2013-02-25 来源: dzsc关键字:VIM  软件调试  S3C2440 手机看文章 扫描二维码
随时随地手机看文章

  引言

  GNU免费提供了一整套工具链,为嵌入式Linux程序的开发和调试提供了完整的支持。其强大的gdb调试工具可以方便地对嵌入式平台上的程序进行跟踪调试;而Linux下强悍的VIM编辑器,不仅可以方便地调用make文件对代码进行编译,而且通过脚本的配置还可轻松地成为高效的代码编辑环境。流传着这样一种说法,“世界上的程序员分三种,一种使用Emacs,一种使用VIM,剩余的是其他。”不去辩论这句话的对与错,单纯从字面意义上来理解,也足见VIM的魅力了。因此,在VIM中实现对嵌入式软件的调试,我们便得到了一个高效、稳定的嵌入式Linux的开发环境。

  1  gdb对嵌入式软件的调试模式

  许多非Linux的嵌入式系统已经在使用gdb与gdb stub对目标板进行远程“交叉调试”;然而,因为Linux内核实现了ptrace()系统调用,所以在对嵌入式应用程序进行调试的时候并不需要gdb stub,而采用gdb套件提供的gdb服务器来对目标板上的嵌入式应用程序进行调试。

  目标板上的gdb服务端gdbserver与主机上的gdb调试器的通信方式主要有两种:使用串口通信的“交叉串行连接”和使用网口的“TCP/IP”联机。鉴于PC端的方便性以及串口资源有限,尤其是现在的笔记本电脑甚至已经不存在串口,所以,大多采用TCP/IP方式,即PC主机与目标板通过网线直连或者PC机与目标板通过路由或者hub等组成局域网通信。这种调试模式如图1所示。

  2  在VIM中实现对嵌入式软件的调试

  我们知道,gdb的功能虽然强大,但由于其基于命令行的操作,所以调试过程不直观,而且Windows下的调试环境集调试与代码编辑为一体,当出现bug的时候,可以方便地对源代码进行修改,相比而言,gdb在这方面又有些失色。既然VIM和gdb的功能如此强大,又完全免费,而且完全适合嵌入式这种特殊的开发模式,那么有没有将二者强强联合的方法呢?有,那就是vimgdb。

  vimgdb是给VIM提供一个可选特性的补丁。它可以在VIM编辑器里提供完整的gdb调试器支持,比如设置断点、查看变量值、gdb命令补全等等,并且这些操作可以在VIM中直观地显示出来。下面阐述在VIM中实现对嵌入式软件调试的具体过程。

  2.1  系统环境及所用软件包版本

  PC操作系统:Ubuntu8.10。

  PC编译器:GNU gcc4.3.1。


  图1  TCP/IP联机的嵌入式软件调试模式

  PC调试器:GNU gdb6.8。

  目标板Linux内核:2.6.13。

  目标板CPU:S3C2440(ARM9架构)。

  交叉编译器:armlinuxgcc3.4.1。

  交叉调试器:自编译GNU gdb6.8。

  目标板gdb服务端:自编译 GNU gdbserver6.8。

  跨平台开发工具路径:~/buildtools/armlinux,且已经设置好系统路径变量。

  测试代码及程序路径:~/test,包含程序代码test.c及Makefile。

  所用软件包存放路径:~/down。

  所用软件包:VIM编辑器源码vim7.1.tar.bz2、vimgdb711.13.tar.gz、GNU gdb源码gdb6.8.tar.bz2。

  2.2  对VIM源码打vimgdb补丁并编译安装

  ① 运行下面的命令,解压VIM源码及vimgdb补丁文件,并对VIM源码打补丁:

  cd ~/down

  tar jxvf vim7.1.tar.bz2

  tar zxvf vimgdb711.13.tar.gz

  patch d vim71 backup p0 < vimgdb/vim71.diff

  ② 运行下面命令,对VIM编译器进行编译和安装:

  cd ~/down/vim71/src

  make

  make install

  执行完上述操作后,VIM将会被安装在/usr/local路径下。如果想修改安装路径,可在上述的编译安装前,打开~/down/vim71/src/Makefile文件的862行安装路径选项并修改。如将VIM安装在/usr路径下,则将 862 #prefix = $(HOME)修改为862 prefix = /usr。[page]

  ③ 安装vimgdb的runtime文件,运行下面的命令:

  cd ~/down/vimgdb

  tar zxfv vimgdb_runtime.tgz C /usr/share/vim/vimfiles

  2.3  建立交叉调试嵌入式软件的gdb组件

  ① 编译嵌入式gdb调试器服务端gdbserver,运行如下的命令:

  cd ~/down/gdb6.8/gdb/gdbserver

  ./configurehost=armlinux target=armlinux

  CC=armlinuxgcc make

  将当前目录下的gdbserver拷贝到目标板文件系统的/bin目录下,以备交叉调试用。

  ② 编译安装交叉调试器gdb,运行如下命令:

  cd ~/down/gdb6.8

  ./configure target=armlinux prefix=/home/popeye/buildtools/armlinux/

  注意,这里的prefix的值必须填写绝对路径,而不能用“ ~”来替代用户路径/home/popeye,否则会提示prefix路径赋值错误。然后运行:

  make

  这个过程中,可能会出现图2所示的错误。

  出现这种情况的原因是,编译规则中选择了警告选项“Werror”。它会将所有的警告转变为错误,而且出现的有关“getwd”函数的提示信息表明,这里编译器检测到的应该是一个“警告”,而不是真正的语法错误。所以,需改正编译选项:

  cd~/down/gdb6.8/gdb

  gedit Makefile

  注意,此处的Makefile是在执行完上述的make命令后才产生的,在最初的代码包里不含有这个文件。对文件的145行进行修改,去掉WERROR_CFLAGS的赋值,即将“145 WERROR_CFLAGS = Werror”修改成“145 WERROR_CFLAGS =”。然后:

  cd ~/down/gdb6.8

  make

  make install


  图2  make过程中的错误提示

  最后进入~/buildtools/armlinux/bin中,发现交叉调试器armlinuxgdb已经存在了。

  2.4  在VIM中实现对嵌入式软件调试前的准备

  在嵌入式软件开发过程中的习惯做法是: 首先,在PC机上编译调试程序,如果在PC机上运行正常,再进行交叉编译。然后,将软件移植到目标板上,如果在目标板上出现bug,再用交叉调试器armlinuxgdb进行调试。

  简而言之,对嵌入式软件的调试过程包含两个部分:PC机上调试部分和嵌入式平台上的调试部分。在这个过程中,可能既用到PC机上的调试器gdb,又用到交叉调试器armlinuxgdb,而对应的是同一个源代码程序和运行在不同平台上的两个可执行程序。同时涉及两个调试器转换的问题,但vimgdb只能对字符串为“gdb”的系统命令进行调用。

  下面,将这个比较困难的问题简单化:

  ① 编辑适用的Makefile,控制生成对应不同平台的可执行程序:

  cd ~/test

  其中,test.c为实验代码,Makefile为编译规则,我们简单编写Makefile的内容为:

  testpc: test.c

  gcc g Wall o testpc test.c

  testem: test.c

  armlinuxgcc g Wall o testem test.C

  当执行“make testpc”命令时,就会生成可运行在PC机上的可执行程序;执行“make testem”则生成可运行在嵌入式目标板上的可执行程序。

  ② 修改vimgdb的快捷键映射脚本,在VIM中实现PC调试器与交叉调试器的轻松切换。

  首先,针对vimgdb只能对字符串为“gdb”的系统命令进行调用,做如下的工作:

  mv /usr/bin/gdb /usr/bin/gdbpc

  cd ~/buildtools/armlinux/bin

  ln s /usr/bin/gdbpc gdb

  由于已经将~/buildtools/armlinux/bin添加到了系统路径里面,所以执行完上述操作后,在任何时候,运行“gdb”命令时,真正运行的调试器取决于这里gdb所连接的调试器。[page]

  其次,编辑文件/etc/vim/macros/gdb_mappings.vim。主要修改和添加的部分为:

  a. 添加调试器转换函数,并设置转换开关为大写“E”键(Shift+E实现):

  let s:emOS_k = 1

  nmap E :call emOS()

  function! s:emOS()

  if s:emOS_k

  let s:emOS_k = 0

  exec ":!ln sf ~/buildtools/armlinux/bin/armlinuxgdb ~/buildtools/armlinux/bin/gdb"

  echohl ErrorMsg

  echo "NOW! Gdb is ready for Embedded System !!!"

  echohl None

  else

  let s:emOS_k = 1

  exec ":!ln sf /usr/bin/gdbpc ~/buildtools/armlinux/bin/gdb"

  echohl ErrorMsg

  echo "Gdb is ready for PC,, Now"

  echohl None

  endif

  Endfunction

  b. 在语句if s:gdb_k行下添加代码:

  nmap :bel 25vsplit gdbvariables

  nunmap E

  即在进入调试状态后,屏蔽掉调试器转换快捷键E,并设置快捷键F8来显示变量值监测窗口。

  c. 在let s:gdb_k = 1行下添加代码:

  exec ":!ln sf /usr/bin/gdbpc ~/buildtools/armlinux/bin/gdb"

  nmap E :call emOS()

  即在退出调试状态后,还原gdb命令为gdbpc的调用,并还原“E”的调试器转换开关作用。

  d. 在/etc/vim/vimrc中添写语句:

  run macros/gdb_mappings.vim

  使得启动vim后,便会在vim中启动对gdb进行调用的快捷键映射。

  至于在gdb_mappings.vim中具体设定的其他快捷键,由读者自己分析或设定即可。

  2.5  在VIM中对嵌入式软件进行调试

  下面设定目标板上的嵌入式软件调试时所用的快捷键: E为调试器转换开关;F9为进入调试模式;F8为开启变量监视窗口;空格键为开启命令行输入窗口;调试模式为PC通过超级终端对嵌入式目标板进行输入输出,Ubuntu8.10通过TCP/IP方式对嵌入式软件进行调试;PC机Linux IP为222.31.51.147;目标板IP为222.31.51.180;调试连接端口为1234。

  ① 用VIM打开~/test/test.c,运行命令“:make testem”,将生成的testem文件拷贝到嵌入式平台的文件系统下,并在嵌入式平台运行命令,指定等待连接的交叉调试器地址、连接端口以及要调试的嵌入式程序:

  gdbserver 222.31.51.147:1234 testem

  嵌入式端会出现如下的类似提示信息,进入等待连接状态:

  Process testem created; pid = 801

  Listening on port 1234

  ② 按下大写“E”键(Shift+按键E),根据VIM窗口下方的提示信息,确定所选调试器。

  提示信息为“NOW! Gdb is ready for Embedded System!!!”或者“Gdb is ready for PC,, Now”。

  ③ 按下F9,在出现的命令窗口输入命令“file testem”后,会在VIM中的另一个窗口出现以下类似的调试信息:

  GNU gdb 6.8

  Copyright (C) 2008 Free Software Foundation, Inc.

  ……

  This GDB was configured as "host=i686pclinuxgnu target=armlinux".

  (gdb) file testem

  Reading symbols from /home/popeye/test/testem…done.

  (gdb)

  可以发现,在VIM中已经成功调用交叉调试器armlinuxgdb了。以后的调试命令,都是通过先按下空格键,调出命令窗口,输入命令,回车后传递给调试器。[page]

  按下空格,在命令窗口输入命令,连接嵌入式目标板端:

  target remote 222.31.51.180:1234

  此时,VIM中的调试信息窗口出现信息:

  (gdb) target remote 222.31.51.180:1234

  Remote debugging using 222.31.51.180:1234

  [New Thread 801]

  0x40000dd0 in ?? () from /lib/ldlinux.so.2

  (gdb)

  而在嵌入式端出现提示信息:

  Process testem created; pid = 801

  Listening on port 1234

  Remote debugging from host 222.31.51.147

  至此,PC交叉调试器与嵌入式软件的连接完成,现在可以在VIM中对程序运行进行例如断点一类的设置动作;而嵌入式软件的输入输出,需要在嵌入式端来完成。这里先通过命令对代码设置断点,然后用命令continue继续程序运行(注意,这里不用run开始,因为当调试器与嵌入式端连接完成时,被调试的嵌入式软件已经开始运行),用命令next对程序实现步进调试。调试过程中的VIM如图3所示。


  图3  正在进行交叉调试的VIM

  图中测试代码要实现的是让用户输入5个数,然后经过排序后输出。对应的输入输出在嵌入式端体现出来。对应图3,此时在嵌入式端需要进行输入动作:

  Listening on port 1234

  Remote debugging from host 222.31.51.147

  Enter 10 numbers:

  a[0]=25

  a[1]=56

  a[2]=……

  从图3中可以直观地看清断点设置在哪里,程序现在步进到哪里。当程序第一次运行到图3中的17行时,按下F8键,开启变量值观测窗口,然后先后执行3个命令“cr i”、“cr j”、“cr a[i]”,这样,就可以在变量观测窗口实时地监测变量的数值了,如图4所示。


  图4  带实时变量值监测窗口的VIM

  这种调试方式提供对gdb所有命令功能的支持,而且当发现bug时,可以通过q命令终止调试,然后按F9跳出调试模式,就可以继续对源代码进行修改。

  至于在这之前的嵌入式软件在本地PC机上的调试,其过程比起调试运行在嵌入式设备上的软件来讲,只少了个远程连接的过程,其余调试过程都一样。至此,实现了在VIM中对嵌入式软件的调试。

  3 结语

  嵌入式Linux系统的广泛应用,对嵌入式软件开发和调试环境的效率提出了更高的要求。GNU所提供的支持交叉编译与调试的工具链是一个很好的选择,尤其是其中的gdb调试工具完全满足嵌入式软件“交叉编译”的这种特殊需要;而且,功能强大的VIM编辑器又可实现对gdb调试器的整合,从而在VIM中实现了对嵌入式软件的调试功能。通过上面的实例可以看到,在VIM中对嵌入式软件进行调试更加直观和高效,从而也促使嵌入式软件的开发效率得到了质的提高。

关键字:VIM  软件调试  S3C2440 引用地址:在VIM中的嵌入式软件调试

上一篇:简论Wi-Fi在EPON中的应用
下一篇:基于双机通信的研究

推荐阅读最新更新时间:2024-03-16 13:19

02 Linux入门命令
1 shell解释器 shell解释器接收输入字符并马上显示,在输入回车后,根据字符串去查找命令。去哪找?去环境变量指定的路径去找。 # 显示环境变量 echo $PATH # 结果如下 /usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin:/usr/games:/usr/local/games:/snap/bin:/usr/local/arm/bin 什么是命令?命令就是可执行程序。 shell解释器根据第一个字符串去找应用程序,把后面的字符串作为参数传给应用程序,然后执行应用程序。为了测试shell,写了如下的c测试代码。 #include std
[单片机]
02 Linux入门命令
基于ELF的嵌入式软件源码级交叉调试技术
1 引言   在计算机软件技术的发展过程中,各种编程调试技术与工具取得了重大的发展,其中调试工具一直是工具软件中研究应用的一大热点。 开发任何一个软件都不可避免地存在各种错误,要修正错误必须找出其错误原因。通常程序员利用调试器来跟踪程序执行情况,快速有效地定位错误产生的位置从而找到引起错误的原因,并改正错误。 调试器为用户提供的主要功能包括:在目标程序中设置、删除断点;以单步执行或连续执行等方式控制目标程序运行;浏览程序中的变量或表达式的值;查看、修改目标机寄存器的内容;查看、修改目标机内存的内容。源码级调试器是面向高级语言的符号调试工具,它基于源代码的语句和符号跟踪观察目标程序,同时提供基于汇编级的程序跟踪功能以满足
[嵌入式]
uboot-2011.12移植到S3C2440(序四)—— uboot.lds分析与解析
OUTPUT_FORMAT( elf32-littlearm , elf32-littlearm , elf32-littlearm ) ;指定输出可执行文件是elf格式,32位ARM指令,小端 OUTPUT_ARCH(arm) ;指定输出可执行文件的平台为ARM ENTRY(_start) ;指定输出可执行文件的起始代码段为_start. SECTIONS { . = 0x00000000 ; 指明目标代码的起始地址从0x0位置开始, . 代表的是当前位置 . = ALIGN(4) ; 代码以4字节对齐 .text : ;指定代码段 { cpu/arm920t/start
[单片机]
s3c2440硬件篇之三:MMU
MMU:内存管理单元。功能: (1)虚拟地址---- 物理地址的映射,使得各个进程拥有看起来一样的地址空间。 (2)检查内存访问权限(硬件自己实现)。保护各个进程所用的内存不被其它进程破坏。 在32位的CPU中,虚拟内存地址为0~0xFFFF_FFFF. Cache:介于主存和CPU之间的高速缓冲存储器。 代码详解:(参考韦东山大哥代码) (1)head.S @************************************************************************* @ File:head.S @ 功能:设置SDRAM,将第二部分代码复制到SDRAM,设置页表,启动MMU,
[单片机]
S3C2440时钟体系及设置方法
这几天一直忙着研究移植U-boot,移植U-boot的过程中有一步很重要,就是要设置s3c2440的时钟,什么 Fin,Fclk,Hclk,Pclk,Mpll,Upll等时钟信号,让初学者看得一头雾水,各种信号混淆不清,本人特意花了一点时间,把这些东东理了理,现整理如下: 首先得弄清一个大的框架,s3c2440 cpu的默认工作主频有两种12MHz和16.9344MHz,也就是我们的晶振的频率,但一般12MHz的晶振用的比较多,Fin就是指我们接的晶振频率。大家都知道s3c2440上电正常工作后频率是远远大于12MHz和16.9344MHz的,我们的s3c2440的cpu正常工作时的频率就是405MHz,因此这就需要一个电路来
[单片机]
<font color='red'>S3C2440</font>时钟体系及设置方法
s3c2440裸机-电阻触摸屏编程(7.触摸屏校准测试 及优化)
1,防止点击一个点,显示出5个点 Isr_Adc中同理也许要上报数据。 这里在touchscreen.c中还补充了上报压力值,当isr_adc上报data时,同时上报了压力值, 这样在ts_read_raw时能够读到压力值讯息。 分析下面这个函数: 当用户点击校准点A时,进入isr_Tc(), 检测到按下,启动adc,adc转换结束产生adc中断,进入isr_adc。然后adc上报坐标和压力值数据。 补充了rs_read_raw时,只有当松开时,也就是read raw读出来的压力值=0的时候才会返回数据。最上面的do while (pressure == 0)是为了过滤掉上一次松开后,下一次还没来得及点击就进入了get_
[单片机]
<font color='red'>s3c2440</font>裸机-电阻触摸屏编程(7.触摸屏校准测试 及优化)
基于WINCE S3C2440 的LED驱动程序的编写
在SMDK2440的BSP包里经常看见这两句语句: volatile IOPreg *s2440IOP = (IOPreg *)IOP_BASE; volatile INTreg *s2440INT = (INTreg *)INT_BASE; 这两条语句简单的理解就是把s2440IOP和IOP_BASE等价,把s2440INT和INT_BASE等价。 IOP_BASE和INT_BASE在s244.h头文件中 #define IOP_BASE 0xB1600000 // 0x56000000 typedef struct { unsigned int rGPACON; // 00 unsigned
[单片机]
MCU软件优化之能源调试
  现在人们更加重视可配置和可编程的解决方案,主要是因为在微控制器中,能源消耗很大程度上直接来自于处理内核和许多外围设备的活动,这也是为什么现在半导体产业对'超低功耗'微控制器解决方案的需求量急剧增加的原因。   传统上,在大多数能源敏感的应用中人们一直采用8位或16位的设备,因为这些设备的内核很小,门相对较少, 泄漏电流的水平较低。然而今天的应用要求的处理能力要比8位或16位内核所能集合的更强大。   过去人们普遍假定32位内核产生的电流在节能模式下就会有益于能量敏感的应用。今天看来这是一种误解。利用现有的全套低功耗设计技术, 32位内核才可以实现低功耗模式,节能效果和8位内核一样好,甚至更好。   Energy Micr
[单片机]
MCU<font color='red'>软件</font>优化之能源<font color='red'>调试</font>
小广播
添点儿料...
无论热点新闻、行业分析、技术干货……
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件
电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved