我正在尝试使用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
并打印功能名称。但是我的问题是调试器无法打印虚拟函数的名称,我很困惑它应该在那里。我在那里做错了什么?
答案 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不包含名称,因为它们仅由偏移量标识(即使导出的函数具有名称)。