什么时候/为什么libstdc ++和libc ++符号不兼容?

时间:2018-11-21 15:37:10

标签: c++ libstdc++ abi libc++ object-files

设置:

test.cpp

#include <set>
#include <string>

void common_config_file_iterator(const std::set<std::string>& allowed_options) {}

include.cpp

#include <set>
#include <string>

void common_config_file_iterator(const std::set<std::string>&) noexcept;

int main() {
    std::set<std::string> set;
    common_config_file_iterator(set);
    return 0;
}

test.sh

clang++-7 test.cpp -c -O3 -fno-rtti -fno-exceptions -o test.o
g++-8 test.o include.cpp -O3 -fno-rtti -fno-exceptions -o test

输出:

Undefined symbols for architecture x86_64:
  "common_config_file_iterator(std::set<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::less<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > >, std::allocator<std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > > > const&)", referenced from:
      _main in ccWoGgrX.o
ld: symbol(s) not found for architecture x86_64
collect2: error: ld returned 1 exit status

所以我做了nm -g test.o

0000000000000000 T __Z27common_config_file_iteratorRKNSt3__13setINS_12basic_stringIcNS_11char_traitsIcEENS_9allocatorIcEEEENS_4lessIS6_EENS4_IS6_EEEE

根据demangler.com的意思是:

common_config_file_iterator(std::__1::set<std::__1::basic_string<char, std::__1::char_traits<char>, std::__1::allocator<char> >, std::__1::less<std::__1::basic_string<char, std::__1::char_traits<char>, std::__1::allocator<char> > >, std::__1::allocator<std::__1::basic_string<char, std::__1::char_traits<char>, std::__1::allocator<char> > > > const&)

Libc++ says

  

[功能和目标:]与gcc的libstdc ++的ABI兼容性,可用于某些低级功能,例如异常对象,rtti和内存分配。

那么,问题std::allocator<char>是吗?

请注意,我使用的是macOS汇编程序。

this issueboost/program-options引起的好奇。

1 个答案:

答案 0 :(得分:2)

  

那么,问题std::allocator<char>是吗?

什么?否。在您的示例中,是一切

您明确引用的文档说,目标是兼容“低级功能,例如异常对象,rtti和内存分配”。

std::setstd::string不是“低级功能,例如异常对象,rtti和内存分配”。它们绝对是libc ++和libstdc ++之间不兼容的,它们是完全不同的库,具有完全不同的实现。

兼容的部分是std::type_infostd::exception(以及<stdexcept>中派生的异常类型)之类的东西,因为它们是基本语言运行时的一部分。除此以外的任何内容,例如容器,字符串,算法,I / O,语言环境等,都不兼容。