首页 > 解决方案 > 在 GDB 中,在非调试二进制文件中的命名空间或类中调用 C++ 函数的正确方法是什么?

问题描述

call只要符号存在,GDB 的命令通常非常适合调用函数。但是,如果函数在命名空间或类中,突然它就无法工作,除非它使用调试信息进行编译。

例如,假设我有这个程序:

#include <iostream>

namespace ns {
    void test()
    {
        std::cout << "ns::test" << std::endl;
    }
}

struct cl {
    static void test()
    {
        std::cout << "cl::test" << std::endl;
    }
};

void func()
{
    std::cout << "func" << std::endl;
}

int main()
{
    ns::test();
    cl::test();
    func();
    return 0;
}

我将test.cpp它保存为 ,用 编译g++ test.cpp -o test,然后在 GDB 中启动它:

$ gdb test
GNU gdb (GDB) 11.1
[...]
Reading symbols from test...
(No debugging symbols found in test)
(gdb) start

从 GDB调用func按预期工作:

(gdb) call (void)func()
func

但是,其他人不会:

(gdb) call (void)ns::test()
No symbol "ns" in current context.
(gdb) call (void)cl::test()
No symbol "cl" in current context.

如果它是用 编译的,它工作得很好-ggdb,但如果源代码不可用,这通常不是一个选项。

值得指出的是,GDB 知道这些函数及其地址:

(gdb) info functions
...
0x0000000000001169  ns::test()
0x000000000000119b  func()
0x000000000000124e  cl::test()
...
(gdb) info symbol 0x1169
ns::test() in section .text
(gdb) break cl::test()
Breakpoint 1 at 0x1252

如果我按 Tab 键,即使在命令中,这些名称也会自动完成call,这意味着在这种情况下,它会自动完成一些不起作用的东西。

此外,如果我使用它们的原始名称,调用这些函数可以正常工作:

(gdb) call (void)_ZN2ns4testEv()
ns::test
(gdb) call (void)_ZN2cl4testEv()
cl::test

那么这里有什么问题呢?这是 GDB 中的错误,还是有某种我不知道的特殊语法?

标签: c++debuggingnamespacesgdb

解决方案


您的 C++ 编译器请放入ns::test符号表中。我们需要做的就是阻止 GDB 的表达式求值器尝试查找不存在的符号ns。为此,请将整个函数名放在单引号中。

(gdb) call (void)'ns::test'()
ns::test

推荐阅读