使用GDB调试时如何在特定地址打印VTable的虚拟功能

时间:2019-01-07 18:35:58

标签: c++ gcc gdb vtable

我正在尝试使用GDB调试器打印VTable中特定内存地址处存在的虚拟函数的名称。

这是我的简单代码:

#include <iostream>
class Parent {
 public:
  virtual void Foo() {}
  virtual void FooNotOverridden() {}
};

class Derived : public Parent {
 public:
  void Foo() override {}
};

int main() {
  Parent p1, p2;
  Derived d1, d2;

  std::cout << "done" << std::endl;
}

以下是gdb的后续信息

0x8201d18 vtable for Derived:   0x00    0x00    0x00    0x00    0x00    0x00    0x00    0x00
0x8201d20 vtable for Derived+8:     0x58    0x1d    0x20    0x08    0x00    0x00    0x00    0x00
0x8201d28 vtable for Derived+16:    0x8e    0x0c    0x00    0x08    0x00    0x00    0x00    0x00
0x8201d30 vtable for Derived+24:    0x82    0x0c    0x00    0x08    0x00    0x00    0x00    0x00
0x8201d38 vtable for Parent:    0x00    0x00    0x00    0x00    0x00    0x00    0x00    0x00
0x8201d40 vtable for Parent+8 : 0x70    0x1d    0x20    0x08    0x00    0x00    0x00    0x00
0x8201d48 vtable for Parent+16 :    0x76    0x0c    0x00    0x08    0x00    0x00    0x00    0x00
0x8201d50 vtable for Parent+24 :    0x82    0x0c    0x00    0x08    0x00    0x00    0x00    0x00

,现在使用以下命令

info symbol 0x8201d8e

我收到此消息

_DYNAMIC + 14 in section .dynamic of .../.../Mypath/MyProject 

根据我在偏移量+8上的一点经验,它将是VTABLE的指针,而在偏移量+16上它将是VTABLE内部存在的第一个方法的指针。因此,如果我得到了地址0x8201d28的值我希望调试器输入的是0x8201d8e,这样的输出

_DYNAMIC + 14 in section .dynamic of .../.../Mypath/MyProject 
 Derived::Foo() in section .text of a.out

并打印功能名称。但是我的问题是调试器无法打印虚拟函数的名称,我很困惑它应该在那里。我在那里做错了什么?

1 个答案:

答案 0 :(得分:1)

您需要从vtable加载函数指针。假设vtable位于地址0x555555755d48

(gdb) info symbol 0x0000555555755d48
vtable for Derived in section .data.rel.ro of /tmp/a.out

我可以这样获得虚函数的名称:

(gdb) print ((void **)0x0000555555755d48)[2]
$3 = (void *) 0x555555554bca <Derived::Foo()>

显然,这将需要调试信息。导出的vtable不包含名称,因为它们仅由偏移量标识(即使导出的函数具有名称)。