使用自己编译的动态库时,一般会出现如下类型的错误提示undefined symbol: _ZN6triton7backend2op8OpResizeD1Ev ;之前看这个错误摸不着头脑,或者直接忽略了,但是这个字符串准确地反馈了错误的地方,下面解释一下。

具体解释

这个符号的命名方式是C++的名称修饰规则(Name Mangling)生成的,因此看起来比较奇怪。具体来说,"_ZN6triton7backend2op8OpResizeD1Ev"可以被拆分为以下几个部分:

  • _Z:这是C++的名称修饰前缀。
  • N6triton7backend2op:这是符号所在的命名空间,即triton::backend::op。当然这个是我自己的程序的命名空间。
  • 8OpResize:这是符号的名称,即OpResize。
  • D1Ev:这是符号的后缀,表示该符号是一个析构函数(D)并且没有参数(1E表示一个空的参数列表)。

因此,这个错误提示表明在程序运行时,使用自己编译的动态库时,链接器无法找到名为triton::backend::op::OpResize::~OpResize()的析构函数的实现。可能是因为动态库编译时缺少了相关的源代码或库文件,或者编译选项不正确,导致符号没有被正确链接。

举一反三

因此,如果有提示类似的错误,应该怎么去参考c++的名称修饰规则,具体的可以参考C++ ABI规范,这里给一个链接:link

但是文档肯定是看不下去的,不过在GCC编译器中,可以使用__cxa_demangle函数来将C++名称修饰符还原为原始的函数名或变量名。__cxa_demangle函数定义在cxxabi.h头文件中,调用该函数需要传入三个参数:符号名称(mangled_name)、输出缓冲区(output_buffer)以及输出缓冲区大小(length和buffer_length)。

#include <cxxabi.h>
#include <iostream>

int main()
{
    const char* mangled_name = "_ZN6triton7backend2op8OpResizeD1Ev";
    int status;
    char* demangled_name = abi::__cxa_demangle(mangled_name, nullptr, nullptr, &status);

    if (status == 0) {
        std::cout << "Original name: " << demangled_name << std::endl;
    } else {
        std::cerr << "Failed to demangle name: " << mangled_name << std::endl;
    }

    free(demangled_name);

    return 0;
}

所以如果你不知道究竟是啥错误,用这个程序翻译一下。

Logo

欢迎来到由智源人工智能研究院发起的Triton中文社区,这里是一个汇聚了AI开发者、数据科学家、机器学习爱好者以及业界专家的活力平台。我们致力于成为业内领先的Triton技术交流与应用分享的殿堂,为推动人工智能技术的普及与深化应用贡献力量。

更多推荐