使用带有 clang++ -stdlib=libc++ 的 libstdc++ 编译库

Posted

技术标签:

【中文标题】使用带有 clang++ -stdlib=libc++ 的 libstdc++ 编译库【英文标题】:Using libstdc++ compiled libraries with clang++ -stdlib=libc++ 【发布时间】:2012-09-14 14:39:20 【问题描述】:

我在 Mac OS X (10.8.2) 下使用 C++ 工作,最近我想出了使用 C++11 功能的需求,这些功能可通过使用 libc++ stdlib 的 clang++ 编译器获得。 但是,我还需要使用一些针对 libstdc++(来自 MacPorts)编译和链接的遗留库。

这样做时,我遇到了链接错误,因为使用 std::string 等遗留库的标头需要针对 std::__1::basic_string(即 std::string 的 libc++ 实现)而不是std::basic_string 实现。

有没有办法在开发中混合使用这两个库(例如,通过使用一些预处理器标志?)

【问题讨论】:

【参考方案1】:

您看到的是使用内联命名空间来实现 ABI 版本控制。

这意味着什么:

libstdc++ std::string 是与 libc++ std::string 不同的数据结构。前者是参考计数设计,而后者不是。尽管它们与 API 兼容,但它们与 ABI 不兼容。这意味着,如果您使用 libstdc++ 构造 std::string,然后将其传递给与 libc++ 链接的其他代码,则接收代码会认为它具有 libc++ std::string。 IE。接收者不会知道它应该增加或减少引用计数。

如果没有内联命名空间,结果将是运行时错误。你能期望的最好的结果就是崩溃。使用内联命名空间时,此运行时错误会转换为链接时错误。

对于程序员来说,libstdc++ std::string 和 libc++ std::string 看起来像是同一类型。但是对于链接器来说,它们看起来像是完全不同的类型(线索是 std::__1 命名空间)。并且链接者的观点是正确的。它们完全不同的类型。

所以是的,您可以操纵一些预处理器标志来链接事物。但是,您将不得不花时间调试由此产生的运行时错误。

做你想做的唯一方法是使这些dylib之间的接口不涉及std::类型,例如string。例如,您可以改为传递 char 的数组。您甚至可以将内存所有权从 libstdc++ 链接的代码转移到 libc++ 链接的代码,反之亦然(它们都将掉入同一个 malloc 池)。

【讨论】:

因此,总而言之,没有希望将遗留库(即与 libstdc++ 链接的库)与 libc++ 一起使用。这确实是一个严重的限制 有一个最大的希望:只要您不跨 dylib 边界传递版本化符号,您就可以在一个进程中混合这两个库。如果你不小心这样做了,错误就会在链接时被发现。您可以跨 dylib 边界抛出std::exception 派生的异常,并跨 dylib 边界转移内存所有权。我怀疑这可能比 gcc 提供自己的 4.2 版本更好的 ABI 兼容性。请注意,根据标准规范,即使 std::string 在 C++98/03 和 C++11 之间也不兼容 ABI。前者被广泛引用,而后者被禁止。 另一个例子:C++11 之前的 gcc std::liststd::list 的 C++11 规范不兼容 ABI。换句话说:您对 libc++ 的评论同样适用于(如果不是更多)libstdc++ 的 C++11 之前的版本。 libc++ 与 C++11 之前的 libstdc++ 不兼容的 ABI 并不比符合 C++11 的 libstdc++ 更不兼容。 @user1690715 更准确地说,当它们在其 API 上公开 STL 对象时,针对这两个运行时构建的互通库是没有希望的——这一直是一个非常冒险的提议。跨度> 难道不能为 std::string 创建一个赋值重载到 std::_1::string ,反之亦然吗?

以上是关于使用带有 clang++ -stdlib=libc++ 的 libstdc++ 编译库的主要内容,如果未能解决你的问题,请参考以下文章

clang 命令

使用C ++ 11时CUDA nvcc编译器失败(Linux; clang 3.8)

使用带有 Clang 的 Boost 编译 c++ 项目时出现问题

使用带有 NMake 样式 Makefile 的 clang-cl 无法回显

带有clang的C大数组中的奇怪分段错误

带有“#”的 Clang 诊断标志会导致构建错误