使用GDB调试RB-tree的几个问题

http://www.oschina.net/question/234345_42190

Content

 

0. 引子

1. 第1个例子

(1) at提示前半部分代表什么?

(2) at提示后半部分代表什么?

(3) 如果要阅读gcc的源代码,那么(2)中的文件在哪里?

2. 第2个例子

(1) gcc源代码中该函数在哪里?

(2) 为什么没有单步进入(step in)_Rb_tree_insert_and_rebalance函数?

(3) 该函数的实现在什么地方?即被编译进了哪个库?能否看到其信息?

(4) 如何单步调试关于红黑树的操作,例如左旋、右旋、平衡等(tree.cc中的函数)

(4.1) 利用disassemble命令找到_Rb_tree_insert_and_rebalance的符号。

(4.2) 利用disassemble命令查看该函数的起始地址

(4.3) 在该函数中设置断点

3. 小结

(1) 本文使用的命令

(2) STL源代码位置

 

0. 引子

 

Linux平台上开发C/C++程序,就免不了要使用GDB,当然你也可以使用DDD(Data Display Debugger),关于DDD可以参考Appendix和官方网站,非本文重点,不做讨论。本文就使用GDB调试的过程中碰到的几个问题进行简单介绍。

 

1. 第1个例子

 

我们先看一个使用GDB调试的例子。

 

(gdb)

operator new (__p=0x8246018)

    at /usr/lib/gcc/i386-redhat-linux/4.1.2/../../../../include/c++/4.1.2/new:94

94      inline void* operator new(std::size_t, void* __p) throw() { return __p; }

(gdb)

0x08049059 in __gnu_cxx::new_allocator ::construct (this=0xbfca1b9f, __p=0x8246018,

    __val=@0xbfca1cec)

    at /usr/lib/gcc/i386-redhat-linux/4.1.2/../../../../include/c++/4.1.2/ext/new_allocator.h:104

104           { ::new(__p) _Tp(__val); }

(gdb)

~allocator (this=0xbfca1b9f)

    at /usr/lib/gcc/i386-redhat-linux/4.1.2/../../../../include/c++/4.1.2/bits/allocator.h:105

105           ~allocator() throw() { }

(gdb)

 

本文讨论的问题如下。

 

(1) at提示前半部分代表什么?

表示当前调用的代码所在的库的路径,即某个symbol存在于该路径下的某个库中。该例子指出当前调用的函数均存在于/usr/lib/gcc/i386-redhat-linux/4.1.2目录下的库文件中。例如,

 

# cd /usr/lib/gcc/i386-redhat-linux/4.1.2

# ls

crtbegin.o     finclude      libgcj.so           libgfortran.so  libstdc++.so

crtbeginS.o    include       libgcj.spec         libgij.so       libsupc++.a

crtbeginT.o    libgcc.a      libgcj-tools.so     libgomp.a       SYSCALLS.c.X

crtend.o       libgcc_eh.a   libgcov.a           libgomp.so

crtendS.o      libgcc_s.so   libgfortran.a       libgomp.spec

crtfastmath.o  libgcj_bc.so  libgfortranbegin.a  libstdc++.a

 

# objdump -x libsupc++.a | grep new

new_handler.o:     file format elf32-i386

rw-r--r-- 102/102   3564 Oct 21 23:42 2007 new_handler.o

  8 .text._ZSt15set_new_handlerPFvvE 0000001f  00000000  00000000  00000060  2**4

 13 .bss.__new_handler 00000004  00000000  00000000  00000120  2**2

00000000 l    d  .text._ZSt15set_new_handlerPFvvE       00000000 .text._ZSt15set_new_handlerPFvvE

00000000 l    d  .bss.__new_handler     00000000 .bss.__new_handler

00000000 g     F .text._ZSt15set_new_handlerPFvvE       0000001f _ZSt15set_new_handlerPFvvE

00000000 g     O .bss.__new_handler     00000004 __new_handler

RELOCATION RECORDS FOR [.text._ZSt15set_new_handlerPFvvE]:

00000014 R_386_GOT32       __new_handler

00000024 R_386_PC32        .text._ZSt15set_new_handlerPFvvE

new_op.o:     file format elf32-i386

rw-r--r-- 102/102   2432 Oct 21 23:42 2007 new_op.o

00000000         *UND*  00000000 __new_handler

0000002d R_386_GOT32       __new_handler

new_opnt.o:     file format elf32-i386

rw-r--r-- 102/102   2360 Oct 21 23:42 2007 new_opnt.o

00000000         *UND*  00000000 __new_handler

00000030 R_386_GOT32       __new_handler

new_opv.o:     file format elf32-i386

rw-r--r-- 102/102   2100 Oct 21 23:42 2007 new_opv.o

new_opvnt.o:     file format elf32-i386

rw-r--r-- 102/102   1608 Oct 21 23:42 2007 new_opvnt.o

 14 .text.__cxa_vec_new2 000000c5  00000000  00000000  00000630  2**4

 15 .text.__cxa_vec_new 00000053  00000000  00000000  00000700  2**4

 16 .text.__cxa_vec_new3 000000cc  00000000  00000000  00000760  2**4

00000000 l    d  .text.__cxa_vec_new2   00000000 .text.__cxa_vec_new2

00000000 l    d  .text.__cxa_vec_new    00000000 .text.__cxa_vec_new

00000000 l    d  .text.__cxa_vec_new3   00000000 .text.__cxa_vec_new3

00000000 g     F .text.__cxa_vec_new2   000000c5 __cxa_vec_new2

00000000 g     F .text.__cxa_vec_new    00000053 __cxa_vec_new

00000000 g     F .text.__cxa_vec_new3   000000cc __cxa_vec_new3

RELOCATION RECORDS FOR [.text.__cxa_vec_new2]:

RELOCATION RECORDS FOR [.text.__cxa_vec_new]:

00000049 R_386_PLT32       __cxa_vec_new2

RELOCATION RECORDS FOR [.text.__cxa_vec_new3]:

0000014c R_386_PC32        .text.__cxa_vec_new2

00000174 R_386_PC32        .text.__cxa_vec_new

00000194 R_386_PC32        .text.__cxa_vec_new3

 

# nm libsupc++.a | grep new

nm: eh_arm.o: no symbols

new_handler.o:

00000000 T _ZSt15set_new_handlerPFvvE

00000000 B __new_handler

new_op.o:

         U __new_handler

new_opnt.o:

         U __new_handler

new_opv.o:

new_opvnt.o:

00000000 T __cxa_vec_new

00000000 T __cxa_vec_new2

00000000 T __cxa_vec_new3

 

解析new_handler.o中的一个符号,

# c++filt _ZSt15set_new_handlerPFvvE

std::set_new_handler(void (*)())

 

该函数在new_handler.cc文件中的定义如下:

 

using std::new_handler;

new_handler __new_handler;

 

new_handler

std::set_new_handler (new_handler handler) throw()

{

  new_handler prev_handler = __new_handler;

  __new_handler = handler;

  return prev_handler;

}

 

其中,new_handlernew文件中被定义,如下。

typedef void (*new_handler)();

 

另外,我们从makefile文件中也可以看到的确是以下5个关于"new".cc文件被编译。

new_handler.cc /

new_op.cc /

new_opnt.cc /

new_opv.cc /

new_opvnt.cc /

 

(2) at提示后半部分代表什么?

后半部分的提示表示当前代码执行到该文件,文件后面的数字表示行号。

../include只是表示在INCLUDE路径下,并没有指明是哪个include目录,可能是/usr/include,可能是/usr/local/include,也可能是第三方的某个include

那么../include/c++/4.1.2/new文件具体在哪里?

 

笔者实验的Linux平台上,文件如下。

/usr/include/c++/4.1.2/new

/usr/include/c++/4.1.2/ext/new_allocator.h

/usr/include/c++/4.1.2/bits/stl_tree.h

 

(3) 如果要阅读gcc的源代码,那么(2)中的文件在哪里?

 

以上3个文件对应gcc的源代码,如下。

./libstdc++-v3/libsupc++/new

./libstdc++-v3/include/ext/new_allocator.h

./libstdc++-v3/include/bits/stl_tree.h (红黑树的实现就在这个文件中,该目录下以stl_开头的文件就是GCC使用的STL)

 

.表示gcc源代码目录,本文为E:/opensource/gcc-4.1.2

2. 2个例子

 

再看一个使用GDB单步运行的例子。

 

(gdb) step

std::_Rb_tree <int, std::_identity, std::less , std::allocator > ::_M_insert (this=0xbfd33574, __x=0x0, __p=0xbfd33578, __v=@0xbfd3358c)

    at /usr/lib/gcc/i386-redhat-linux/4.1.2/../../../../include/c++/4.1.2/bits/stl_tree.h:817

817                                                           _S_key(__p)));

(gdb) n

819           _Link_type __z = _M_create_node(__v);

(gdb)

821           _Rb_tree_insert_and_rebalance(__insert_left, __z, __p, 

(gdb) step

823           ++_M_impl._M_node_count;

(gdb) l

818

819           _Link_type __z = _M_create_node(__v);

820

821           _Rb_tree_insert_and_rebalance(__insert_left, __z, __p, 

822                                         this->_M_impl._M_header);

823           ++_M_impl._M_node_count;

824           return iterator(__z);

825         }

826

(gdb)

 

该例子要讨论的问题如下。

stl_tree.h821行的_Rb_tree_insert_and_rebalance调用为什么没有step in(进入函数内部)?在Linux平台,该函数的实现在哪里?在gcc源代码中该函数在哪里?

 

我们先看该函数在gcc源代码中的位置。

 

(1) gcc源代码中该函数在哪里?

 

仔细查看gcc的源代码,我们发现,_Rb_tree_insert_and_rebalance函数的定义并不在str_tree.h中,而是在tree.cc文件中,位于./libstdc++-v3/src目录下,该目录下还有其他的pool_allocator.ccmt_allocator.cclist.cc等,其对应的头文件分别为:

./libstdc++-v3/include/ext/pool_allocator.h

./libstdc++-v3/include/ext/mt_allocator.h

./libstdc++-v3/include/std/std_list.h

 

对于list.cc需要解释一下。在list.cc文件中,直接#include ,但实际上,list文件在gcc的源代码包里是不存在的。那么包含list实际上是包含谁?我们在Linux系统中查看,list文件在/usr/include/c++/4.1.2目录中,如下。

 

# pwd

/usr/include/c++/4.1.2

# ls

algorithm  clocale  ctime                functional         java     ostream    typeinfo

backward   cmath    cwchar               gcj                javax    queue      utility

bits       complex  cwctype              gnu                limits   set        valarray

bitset     csetjmp  cxxabi.h             i386-redhat-linux  list     sstream    vector

...

 

打开list文件,其内容如下,这个标准的头文件实际上就是包含stl_list的定义文件stl_list.h及其需要的其他头文件,如allocator.h等。

 

 60 #ifndef _GLIBCXX_LIST

 61 #define _GLIBCXX_LIST 1

 62

 63 #pragma GCC system_header

 64

 65 #include

 66 #include

 67 #include

 68 #include

 69 #include

 70 #include    //stl list的定义文件

 71

 72 #ifndef _GLIBCXX_EXPORT_TEMPLATE

 73 # include     //stl list的某些模板成员函数的实现文件

 74 #endif

 75

 76 #ifdef _GLIBCXX_DEBUG

 77 # include

 78 #endif

 

对应于gcc源代码,安装到Linux系统中的list文件实际上就是./libstdc++-v3/include/std/std_list.h文件,只是在安装到时候将其改名并拷贝到/usr/include/c++/4.1.2目录。同样的道理也适用于STL其他的container,如vectorsetdequemapstackqueue等。所以我们使用STL编写代码时,包含的文件是,,等,而非 , ,

 

以下是readme文件中对./libstdc++-v3/include/std./libstdc++-v3/src目录下的文件的解释。

    include/std

      Files meant to be found by #include directives in

      standard-conforming user programs. 

 

  src

    Files that are used in constructing the library, but are not

    installed.

 

问题扯远了,我们回到要讨论的问题。

 

(2) 为什么没有单步进入(step in)_Rb_tree_insert_and_rebalance函数?

 

要想step in某个函数,一定要有相应的源代码的文件,很显然,该函数所在的文件tree.ccLinux系统里并不存在。那如果我们希望单步进入该函数,可能需要重新编译gcc源代码并将所有.h.c/.cc/.cpp的文件都安装到相应的目录下。例如在笔者的虚拟机里,调试ACE的程序,都可以进入ACE源代码内部进行。但从上述对./libstdc++-v3/src目录下的文件的解释可以看出,在Linux系统上并不安装这些.cc文件,因此,要单步调试这些程序,貌似很困难。但也不是没有方法,我们稍后讨论。

 

那么,

 

(3) 该函数的实现在什么地方?即被编译进了哪个库?能否看到其信息?

 

摘_Rb_tree_increment函数的代码,如下。

 

namespace std

{

  _Rb_tree_node_base*

  _Rb_tree_increment(_Rb_tree_node_base* __x)

  {

    if (__x->_M_right != 0)

      {

        __x = __x->_M_right;

        while (__x->_M_left != 0)

          __x = __x->_M_left;

      }

    else

      {

        _Rb_tree_node_base* __y = __x->_M_parent;

        while (__x == __y->_M_right)

          {

            __x = __y;

            __y = __y->_M_parent;

          }

        if (__x->_M_right != __y)

          __x = __y;

      }

    return __x;

  }

...

}

 

其参数_Rb_tree_node_base定义如下,

  struct _Rb_tree_node_base

  {

    typedef _Rb_tree_node_base* _Base_ptr;

    typedef const _Rb_tree_node_base* _Const_Base_ptr;

 

    _Rb_tree_color        _M_color;

    _Base_ptr                _M_parent;

    _Base_ptr                _M_left;

    _Base_ptr                _M_right;

    ...

  };

 

很显然,该函数是属于std空间的一个全局函数,不是类的成员函数,也不是模板函数,其参数_Rb_tree_node_base也不是模板类。tree.cc中的9个函数均是这样,且在每个函数内部,只是对指针的操作,不涉及定义一个对象(编译时需要分配逻辑地址,占虚拟空间),或者使用new/delete操作符,因此所有这些函数(tree.cc文件)在编译期间不需要确定一些数据类型,gcc在编译时便将其编译到了静态库libstdc++.a(tree.cc的目标文件为tree.o),使用时再链接到目标可执行文件。

 

实际上,可以认为tree.cc文件(9个函数)是一个独立的小模块,而且是一个独立性很强的模块,_Rb_tree_node_base的改变(stl_tree.h的改变)不会影响tree.cc。像这样的模块,应该做成静态库供用户使用,从而减少编译依赖和时间。

 

可以通过如下命令查看静态库libstdc++.a中的关于RBtree的操作信息。

 

# cd /usr/lib/gcc/i386-redhat-linux/4.1.2  //以下要解析的库在该目录

# nm libstdc++.so                         //libstdc++.so是动态库,其中没有symbols

nm: libstdc++.so: no symbols

 

# nm libstdc++.a | grep _Rb_tree_

00000000 T _ZSt18_Rb_tree_decrementPKSt18_Rb_tree_node_base

00000000 T _ZSt18_Rb_tree_decrementPSt18_Rb_tree_node_base

00000000 T _ZSt18_Rb_tree_incrementPKSt18_Rb_tree_node_base

00000000 T _ZSt18_Rb_tree_incrementPSt18_Rb_tree_node_base

00000000 T _ZSt20_Rb_tree_black_countPKSt18_Rb_tree_node_baseS1_

00000000 T _ZSt20_Rb_tree_rotate_leftPSt18_Rb_tree_node_baseRS0_

00000000 T _ZSt21_Rb_tree_rotate_rightPSt18_Rb_tree_node_baseRS0_

00000000 T _ZSt28_Rb_tree_rebalance_for_erasePSt18_Rb_tree_node_baseRS_

00000000 T _ZSt29_Rb_tree_insert_and_rebalancebPSt18_Rb_tree_node_baseS0_RS_

nm: stubs.o: no symbols

nm: eh_arm.o: no symbols

 

# c++filt _ZSt18_Rb_tree_decrementPKSt18_Rb_tree_node_base

std::_Rb_tree_decrement(std::_Rb_tree_node_base const*)

 

# c++filt _ZSt18_Rb_tree_decrementPSt18_Rb_tree_node_base

std::_Rb_tree_decrement(std::_Rb_tree_node_base*)

 

# c++filt _ZSt28_Rb_tree_rebalance_for_erasePSt18_Rb_tree_node_baseRS_

std::_Rb_tree_rebalance_for_erase(std::_Rb_tree_node_base*, std::_Rb_tree_node_base&)

 

以上通过c++filt命令解析的符号便是./libstdc++-v3/src/tree.cc文件中的函数,其他的符号,读者可自行试验,也是tree.cc中的函数。

 

由以上信息可知,该函数被编译进了libstdc++.a这个静态库中。通过如下命令,可以得知tree.cc编译后的目标文件tree.o的确被打包进了静态库libstdc++.a

 

# cd /usr/lib/gcc/i386-redhat-linux/4.1.2

# mkdir test           //建立一个试验目录,试验完后容易删除

# cp libstdc++.a test/libstdc++.a

# cd test

# ls

libstdc++.a

# ar -x libstdc++.a    //extract file(s) from the archive

# ls                   //所有.o文件便是从静态库libstdc++.a中释放出来的

...

atomicity.o         eh_call.o          istream-inst.o      pool_allocator.o

basic_file.o        eh_catch.o         istream.o           pure.o

bitmap_allocator.o  eh_exception.o     libstdc++.a         sstream-inst.o

c++locale.o         eh_globals.o       limits.o            stdexcept.o

codecvt_members.o   eh_personality.o   list.o              streambuf-inst.o

...

ctype.o             functexcept.o      misc-inst.o         tree.o

debug_list.o        globals_io.o       monetary_members.o  valarray-inst.o

debug.o             globals_locale.o   mt_allocator.o      vec.o

del_opnt.o          guard.o            new_handler.o       vterminate.o

...

eh_arm.o            ios.o              numeric_members.o

 

我们甚至可以直接使用从libstdc++.a中释放出来的tree.o,在生成可执行文件时,直接连接之。不像./libstdc++-v3/include/bits/list.tccvector.tcc中的函数,他们不仅是list类和vector类的成员函数,也是模板函数。因此在std_list.h文件的第73行,需要包含list.tcc文件,如上list文件(Linux平台上安装std_list.h后的文件)的内容所示。同样的道理也适用于std_vector.h文件。

 

我们也可以使用nm命令直接查看tree.o的相关信息。

# nm -A tree.o
tree.o:00000000 T _ZSt18_Rb_tree_decrementPKSt18_Rb_tree_node_base
tree.o:00000000 T _ZSt18_Rb_tree_decrementPSt18_Rb_tree_node_base
tree.o:00000000 T _ZSt18_Rb_tree_incrementPKSt18_Rb_tree_node_base
tree.o:00000000 T _ZSt18_Rb_tree_incrementPSt18_Rb_tree_node_base
tree.o:00000000 T _ZSt20_Rb_tree_black_countPKSt18_Rb_tree_node_baseS1_
tree.o:00000000 T _ZSt20_Rb_tree_rotate_leftPSt18_Rb_tree_node_baseRS0_
tree.o:00000000 T _ZSt21_Rb_tree_rotate_rightPSt18_Rb_tree_node_baseRS0_
tree.o:00000000 T _ZSt28_Rb_tree_rebalance_for_erasePSt18_Rb_tree_node_baseRS_
tree.o:00000000 T _ZSt29_Rb_tree_insert_and_rebalancebPSt18_Rb_tree_node_baseS0_RS_
tree.o:         U __gxx_personality_v0
 

以上符号经demangle后分别与以下命令的结果相对应。

 

# nm -C tree.o

00000000 T std::_Rb_tree_decrement(std::_Rb_tree_node_base const*)

00000000 T std::_Rb_tree_decrement(std::_Rb_tree_node_base*)

00000000 T std::_Rb_tree_increment(std::_Rb_tree_node_base const*)

00000000 T std::_Rb_tree_increment(std::_Rb_tree_node_base*)

00000000 T std::_Rb_tree_black_count(std::_Rb_tree_node_base const*, std::_Rb_tree_node_base const*)

00000000 T std::_Rb_tree_rotate_left(std::_Rb_tree_node_base*, std::_Rb_tree_node_base*&)

00000000 T std::_Rb_tree_rotate_right(std::_Rb_tree_node_base*, std::_Rb_tree_node_base*&)

00000000 T std::_Rb_tree_rebalance_for_erase(std::_Rb_tree_node_base*, std::_Rb_tree_node_base&)

00000000 T std::_Rb_tree_insert_and_rebalance(bool, std::_Rb_tree_node_base*, std::_Rb_tree_node_base*, std::_Rb_tree_node_base&)

         U __gxx_personality_v0

 

也可以试试以下命令。对nm命令输出的解释可参考nmmanual页。

# nm -a tree.o

# nm -A tree.o

# nm -B tree.o

# nm -C tree.o (upper case)

# nm -g tree.o

# nm -l tree.o

# nm -n tree.o

# nm -o tree.o

# nm -p tree.o

# nm -P tree.o (upper case)

# nm -s tree.o

# nm -S tree.o (upper case)

# nm -u tree.o

其余均小写。

 

(4) 如何单步调试关于红黑树的操作,例如左旋、右旋、平衡等(tree.cc中的函数)

 

回答该问题,需要调试STL的源代码。摘录如下。

 

(gdb) step

std::_Rb_tree <int, std::_identity, std::less , std::allocator >::_M_insert (this=0xbf849884, __x=0x0,

    __p=0xbf849888, __v=@0xbf84989c)

    at /usr/lib/gcc/i386-redhat-linux/4.1.2/../../../../include/c++/4.1.2/bits/stl_tree.h:817

817                                                           _S_key(__p)));

(gdb) n

819           _Link_type __z = _M_create_node(__v);

(gdb)

821           _Rb_tree_insert_and_rebalance(__insert_left, __z, __p, 

 

(4.1) 利用disassemble命令找到_Rb_tree_insert_and_rebalance的符号。


(gdb) disassemble

Dump of assembler code for function _ZNSt8_Rb_treeIiiSt9_IdentityIiESt4lessIiESaIiEE9_M_insertEPSt18_Rb_tree_node_baseS7_RKi:

...

0x08049245 <_ZNSt8_Rb_treeIiiSt9_IdentityIiESt4lessIiESaIiEE9_M_insertEPSt18_Rb_tree_node_baseS7_RKi+149>:      mov    %ecx,(%esp)

0x08049248 <_ZNSt8_Rb_treeIiiSt9_IdentityIiESt4lessIiESaIiEE9_M_insertEPSt18_Rb_tree_node_baseS7_RKi+152>:      call   0x80487b8 <_ZSt29_Rb_tree_insert_and_rebalancebPSt18_Rb_tree_node_baseS0_RS_@plt>

...

End of assembler dump.

 

可以利用c++filt命令验证该符号。

 

# c++filt _ZSt29_Rb_tree_insert_and_rebalancebPSt18_Rb_tree_node_baseS0_RS_

std::_Rb_tree_insert_and_rebalance(bool, std::_Rb_tree_node_base*, std::_Rb_tree_node_base*, std::_Rb_tree_node_base&)

 

(4.2) 利用disassemble命令查看该函数的起始地址

 

(gdb) disassemble _ZSt29_Rb_tree_insert_and_rebalancebPSt18_Rb_tree_node_baseS0_RS_

Dump of assembler code for function _ZSt29_Rb_tree_insert_and_rebalancebPSt18_Rb_tree_node_baseS0_RS_:

0x00c08750 <_ZSt29_Rb_tree_insert_and_rebalancebPSt18_Rb_tree_node_baseS0_RS_+0>:       push   %ebp

0x00c08751 <_ZSt29_Rb_tree_insert_and_rebalancebPSt18_Rb_tree_node_baseS0_RS_+1>:       mov    %esp,%ebp

0x00c08753 <_ZSt29_Rb_tree_insert_and_rebalancebPSt18_Rb_tree_node_baseS0_RS_+3>:       push   %edi

0x00c08754 <_ZSt29_Rb_tree_insert_and_rebalancebPSt18_Rb_tree_node_baseS0_RS_+4>:       push   %esi

0x00c08755 <_ZSt29_Rb_tree_insert_and_rebalancebPSt18_Rb_tree_node_baseS0_RS_+5>:       push   %ebx

...

 

由上可以看出,该函数的起始地址为0x00c08750,很明显,只是一个虚拟地址,而非物理地址。

 

为什么要这么做?——因为函数被编译时均被mangled,成为symbol

从如下命令可以看到在目标可执行文件中已经不存在_Rb_tree_insert_and_rebalance这样的符号了。

 

(gdb) disassemble _Rb_tree_insert_and_rebalance

No symbol "_Rb_tree_insert_and_rebalance" in current context.

(gdb) disassemble std::_Rb_tree_insert_and_rebalance

No symbol "_Rb_tree_insert_and_rebalance" in namespace "std".

 

(4.3) 在该函数中设置断点

 

(gdb) b *0x00c08750

Breakpoint 2 at 0xc08750

(gdb) c

Continuing.

 

Breakpoint 2, 0x00c08750 in std::_Rb_tree_insert_and_rebalance () from /usr/lib/libstdc++.so.6

(gdb) stepi

0x00c08751 in std::_Rb_tree_insert_and_rebalance () from /usr/lib/libstdc++.so.6

(gdb)

0x00c08753 in std::_Rb_tree_insert_and_rebalance () from /usr/lib/libstdc++.so.6

(gdb)

0x00c08754 in std::_Rb_tree_insert_and_rebalance () from /usr/lib/libstdc++.so.6

(gdb)

0x00c08755 in std::_Rb_tree_insert_and_rebalance () from /usr/lib/libstdc++.so.6

(gdb)

0x00c08756 in std::_Rb_tree_insert_and_rebalance () from /usr/lib/libstdc++.so.6

(gdb)

 

由此,使用stepi命令便可调试汇编代码了。

 

从以上调试信息可以看出,在运行过程中,_Rb_tree_insert_and_rebalance函数是从/usr/lib/libstdc++.so.6这个动态库中加载的。可以通过以下方式验证该函数在libstdc++.so.6中的位置。

 

# gdb /usr/lib/libstdc++.so.6.0.8

...

(gdb) disassemble _ZSt29_Rb_tree_insert_and_rebalancebPSt18_Rb_tree_node_baseS0_RS_

Dump of assembler code for function _ZSt29_Rb_tree_insert_and_rebalancebPSt18_Rb_tree_node_baseS0_RS_:

0x00c08750 <_ZSt29_Rb_tree_insert_and_rebalancebPSt18_Rb_tree_node_baseS0_RS_+0>:       push   %ebp

0x00c08751 <_ZSt29_Rb_tree_insert_and_rebalancebPSt18_Rb_tree_node_baseS0_RS_+1>:       mov    %esp,%ebp

0x00c08753 <_ZSt29_Rb_tree_insert_and_rebalancebPSt18_Rb_tree_node_baseS0_RS_+3>:       push   %edi

...

题外话。

 

还可以通过如下方式查看该函数的汇编代码,可以看到,在tree.o中只是一个相对地址,没有虚拟地址。

 

# gdb ./tree.o

...

(gdb) disassemble 0

Dump of assembler code for function _ZSt18_Rb_tree_incrementPSt18_Rb_tree_node_base:

0x00000000 <_ZSt18_Rb_tree_incrementPSt18_Rb_tree_node_base+0>: push   %ebp

0x00000001 <_ZSt18_Rb_tree_incrementPSt18_Rb_tree_node_base+1>: mov    %esp,%ebp

0x00000003 <_ZSt18_Rb_tree_incrementPSt18_Rb_tree_node_base+3>: mov    0x8(%ebp),%ecx

0x00000006 <_ZSt18_Rb_tree_incrementPSt18_Rb_tree_node_base+6>: mov    0xc(%ecx),%edx

0x00000009 <_ZSt18_Rb_tree_incrementPSt18_Rb_tree_node_base+9>: test   %edx,%edx

...


# objdump -d tree.o

...

Disassembly of section .text._ZSt29_Rb_tree_insert_and_rebalancebPSt18_Rb_tree_node_baseS0_RS_:

 

00000000 <_ZSt29_Rb_tree_insert_and_rebalancebPSt18_Rb_tree_node_baseS0_RS_>:

   0:   55                      push   %ebp

   1:   89 e5                   mov    %esp,%ebp

   3:   57                      push   %edi

   4:   56                      push   %esi

   5:   53                      push   %ebx

   6:   83 ec 0c                sub    $0xc,%esp

   9:   8b 45 14                mov    0x14(%ebp),%eax

   c:   8b 7d 0c                mov    0xc(%ebp),%edi

   f:   8b 5d 10                mov    0x10(%ebp),%ebx

...

 

 

也可以试试以下命令。对objdump命令输出的解释可参考其的manual页。

# objdump -x tree.o

# objdump -d tree.o

# objdump -h tree.o

# objdump -s tree.o

# objdump -S tree.o (upper case)

 

 

以下问题,将另文讨论。

静态库和动态库的区别与使用

动态库如何加载?加载时虚拟地址空间如何重定位?

 

3. 小结

 

(1) 本文使用的命令

nm

objdump

c++filt

 

(2) STL源代码位置

GCC源代码中STL位置:./libstdc++-v3/include/bits (.表示gcc源代码目录,本文为E:/opensource/gcc-4.1.2)

Linux系统中STL位置:/usr/include/c++/4.1.2/bits

 

 

Reference

./libstdc++-v3/readme

man nm (nmmanual)

nm -h (nm help)

 

 

Appendix: About DDD

 

GNU DDD is a graphical front-end for command-line debuggers such as GDB, DBX, WDB, Ladebug, JDB, XDB, the Perl debugger, the bash debugger bashdb, the GNU Make debugger remake, or the Python debugger pydb. Besides ``usual'' front-end features such as viewing source texts, DDD has become famous through its interactive graphical data display, where data structures are displayed as graphs.

Linux平台上的一种可视化调试工具,其最大的特点是数据结构以图形方式显示,非常直观。

http://www.gnu.org/software/ddd



Technorati 标签: STL, RB-tree, 红黑树, GDB


原文链接: http://blog.csdn.net/livelylittlefish/article/details/6299402
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
【优质项目推荐】 1、项目代码均经过严格本地测试,运行OK,确保功能稳定后才上传平台。可放心下载并立即投入使用,若遇到任何使用问题,随时欢迎私信反馈与沟通,博主会第一时间回复。 2、项目适用于计算机相关专业(如计科、信息安全、数据科学、人工智能、通信、物联网、自动化、电子信息等)的在校学生、专业教师,或企业员工,小白入门等都适用。 3、该项目不仅具有很高的学习借鉴价值,对于初学者来说,也是入门进阶的绝佳选择;当然也可以直接用于 毕设、课设、期末大作业或项目初期立项演示等。 3、开放创新:如果您有一定基础,且热爱探索钻研,可以在此代码基础上二次开发,进行修改、扩展,创造出属于自己的独特应用。 欢迎下载使用优质资源!欢迎借鉴使用,并欢迎学习交流,共同探索编程的无穷魅力! 基于业务逻辑生成特征变量python实现源码+数据集+超详细注释.zip基于业务逻辑生成特征变量python实现源码+数据集+超详细注释.zip基于业务逻辑生成特征变量python实现源码+数据集+超详细注释.zip基于业务逻辑生成特征变量python实现源码+数据集+超详细注释.zip基于业务逻辑生成特征变量python实现源码+数据集+超详细注释.zip基于业务逻辑生成特征变量python实现源码+数据集+超详细注释.zip基于业务逻辑生成特征变量python实现源码+数据集+超详细注释.zip 基于业务逻辑生成特征变量python实现源码+数据集+超详细注释.zip 基于业务逻辑生成特征变量python实现源码+数据集+超详细注释.zip
提供的源码资源涵盖了安卓应用、小程序、Python应用和Java应用等多个领域,每个领域都包含了丰富的实例和项目。这些源码都是基于各自平台的最新技术和标准编写,确保了在对应环境下能够无缝运行。同时,源码中配备了详细的注释和文档,帮助用户快速理解代码结构和实现逻辑。 适用人群: 这些源码资源特别适合大学生群体。无论你是计算机相关专业的学生,还是对其他领域编程感兴趣的学生,这些资源都能为你提供宝贵的学习和实践机会。通过学习和运行这些源码,你可以掌握各平台开发的基础知识,提升编程能力和项目实战经验。 使用场景及目标: 在学习阶段,你可以利用这些源码资源进行课程实践、课外项目或毕业设计。通过分析和运行源码,你将深入了解各平台开发的技术细节和最佳实践,逐步培养起自己的项目开发和问题解决能力。此外,在求职或创业过程中,具备跨平台开发能力的大学生将更具竞争力。 其他说明: 为了确保源码资源的可运行性和易用性,特别注意了以下几点:首先,每份源码都提供了详细的运行环境和依赖说明,确保用户能够轻松搭建起开发环境;其次,源码中的注释和文档都非常完善,方便用户快速上手和理解代码;最后,我会定期更新这些源码资源,以适应各平台技术的最新发展和市场需求。
### 回答1: 在 RISC-V 平台上使用 GDB 进行调试,需要先安装 RISC-V 版本的 GDB。可以通过以下命令安装: ``` sudo apt-get install gdb-multiarch ``` 安装完成后,可以使用以下命令启动 GDB: ``` riscv64-unknown-elf-gdb <executable> ``` 其中 `<executable>` 是要调试的可执行文件的路径。接下来,可以使用 GDB 的各种命令进行调试。 以下是一些常用的 GDB 命令: - `break <function>`:在指定函数处设置断点。 - `break <line>`:在指定行处设置断点。 - `run <args>`:运行程序并传入参数。 - `step`:单步执行程序。 - `next`:执行下一条语句,不进入函数内部。 - `continue`:继续执行程序,直到遇到下一个断点。 - `print <variable>`:打印变量的值。 - `backtrace`:查看函数调用栈。 此外,还可以使用 `-g` 选项编译程序时生成调试信息,以便在 GDB 中进行调试。例如: ``` riscv64-unknown-elf-gcc -g -o <executable> <source files> ``` ### 回答2: gdb是一个功能强大的调试工具,适用于多种不同的处理器架构,其中包括risc-v。使用gdb调试risc-v程序可以帮助我们定位并解决程序中的bug。 具体步骤如下: 1. 安装gdb:首先需要确保在计算机上安装了gdb。可以通过包管理器或者从gdb的官方网站上下载并安装。 2. 编译程序:在使用gdb之前,需要将C/C++程序编译为可执行文件。例如,我们可以使用risc-v的交叉编译工具链来编译程序。 3. 运行gdb:在命令行中输入"gdb"命令来启动gdb调试器。 4. 加载可执行文件:在gdb中输入"file <可执行文件路径>"命令来加载要调试的可执行文件。 5. 设置断点:可以使用"break <行号或函数名>"命令在程序中设置断点。这将使程序暂停在指定的位置,以便进行调试。 6. 启动程序:在gdb中输入"run"命令来启动程序的执行。当程序到达设置的断点时,它将暂停执行。 7. 检查程序状态:在程序暂停时,可以使用各种gdb命令来检查当前变量的值、堆栈跟踪等信息,以了解程序的当前状态。 8. 单步执行:可以使用"next"或"step"命令逐行执行程序。这将允许我们逐步跟踪程序的执行,以查找错误。 9. 查看变量值:使用"gdb"命令可以查看当前变量的值。可以为每个变量设置监视点来观察其变化。 10. 修复错误:一旦发现了bug,可以在gdb中进行相应的更改或修复。然后可以重新编译程序并再次使用gdb进行调试,以确保问题已经解决。 使用gdb调试risc-v程序可以更加高效地定位问题并进行调试。通过将gdb与其他调试技术,如断言和日志输出,结合使用,可以帮助开发人员更轻松地编写和维护risc-v程序。 ### 回答3: gdb是一款强大的调试器,可用于调试RISC-V程序。以下是使用gdb调试RISC-V程序的步骤: 首先,确保已经安装了RISC-V架构的gdb工具。如果没有安装,可以通过下载源代码并手动编译安装来获取。 1. 将需要调试的RISC-V可执行文件加载到gdb中。可以使用以下命令: ``` gdb <可执行文件名> ``` 2. 设置断点来在程序的特定位置中断执行。使用`break`命令来设置断点。例如,要在第30行设置断点,可以使用以下命令: ``` break main.c:30 ``` 3. 启动调试会话。可以直接使用`run`命令启动程序,或者使用`start`命令以暂停的方式启动程序(这样可以在设置完断点后再启动)。 4. 当程序在断点处中断时,可以使用`next`命令逐过程执行,或者使用`step`命令逐语句执行。还可以使用`continue`命令使程序继续执行直到下一个断点或程序结束。 5. 在程序执行过程中,可以使用`print`命令打印变量或表达式的值。例如,要打印变量`x`的值,可以使用以下命令: ``` print x ``` 6. 若要查看当前执行的源代码行,可以使用`list`命令。 7. 将gdb作为交互式调试使用时,还可以使用一些其他命令来查看寄存器的值、改变变量的值等。 在调试RISC-V程序时,gdb是一个非常有用的工具。它提供了一系列命令和功能,帮助我们理解程序执行的过程,查找错误并进行调试。以上是使用gdb调试RISC-V程序的基本步骤。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值