当前位置: 首页 > 面试题库 >

_GLIBCXX_USE_CXX11_ABI,GCC 4.8和ABI兼容性

宋昊然
2023-03-14
问题内容

我们收到了一些为linux编译的库(.a)(可能是用GCC 6.x编译的)。

我们正在使用GCC 4.8,并且undefined reference to std::__cxx11::basic_string在尝试链接时遇到类型错误:

通常,可以通过确保所有单元都使用相同的_GLIBCXX_USE_CXX11_ABI标志进行编译来解决此问题。但是,如果我理解正确,它是由GCC
5.1及更高版本引入的。

  1. 是否有办法在GCC 4.8上进行这项工作,还是我们需要让人们用不同的方式重新编译库_GLIBCXX_USE_CXX11_ABI
  2. 我想如果我们能够切换到GCC> = 5.1,那么我们可以做到这一点吗?

谢谢!


问题答案:

可以在gcc 4.8.2中使用C 11 ABI,但这是一个危险的技巧。如果完全可以要求您的供应商提供使用C 03
ABI(-D_GLIBCXX_USE_CXX11_ABI=0)编译的库或升级到GCC 5或更高版本,那么情况会好得多。

您需要下载并安装gcc 5,以便可以使用其libstdc 头文件和库,然后指示gcc 4.8优先使用它们。另外,由于gcc 4.8缺少gcc
5随附的libstdc
所需的某些内在函数,因此您需要弄清楚它们的用法。

例如,要编译一个简单的单文件应用程序,其中包括<string>

/usr/local/gcc-4.8.2/bin/g++ \
   -std=c++11 \
   -D_GLIBCXX_USE_CXX11_ABI=1 \
   -D'__is_trivially_copyable(...)=0' \
   -D'__is_trivially_constructible(...)=0' \
   -D'__is_trivially_assignable(...)=0' \
   -nostdinc++ \
   -isystem /usr/local/gcc-5.4.0/include/c++/5.4.0/ \
   -isystem /usr/local/gcc-5.4.0/include/c++/5.4.0/x86_64-unknown-linux-gnu \
   -L /usr/local/gcc-5.4.0/lib64
   a.cpp

这很危险,因为gcc 5.4 libstdc ++不适用于gcc
4.8,并且重新定义所使用的内在函数(__is_trivially_copyable等)可能会更改结构的布局,或导致程序与供应商的库之间的二进制不兼容。

为了运行生成的可执行文件,您还需要确保动态链接器找到兼容的libstdc
++,例如通过添加/usr/local/gcc-5.4.0/lib64/etc/ld.so.conf或使用-Wl,-rpath /usr/local/gcc-5.4.0/lib64



 类似资料:
  • 我的项目计划使用SSO(可能使用Gluu或Auth0,…)并且正在考虑将GraphQL应用到我们的API中。 乍一看,这两个应该很容易兼容,因为它们在不同的层上工作。但是我仍然想听听有这两个方面经验的人在将它们应用于项目时是否有任何问题、考虑因素或指导方针?

  • 本章提供了有关 版本控制 章节中提供的破坏性和非破坏性修改列表的详细说明。 什么算是一个破坏性(不兼容)的变化并没有明确的定义。本指南应该被视为指示性的,而不是每一种可能变化的全面清单。 这里列出的规则只涉及客户端兼容性。预期API生产者明白在部署方面的要求,包括实现细节的变化。 一般目的是,服务端更新到一个新的minor版本或patch版本不该破坏客户端。可预期的破坏类型有: 源代码兼容性:针对

  • 语言补丁 Array.isArray Object.assign JSON.stringify console-polyfill Object.keys Object.is Array.prototype.forEach Function.prototype.bind 或者直接使用https://polyfill.io/ 提供的动态补丁方案 <script src="https://cdn.pol

  • 最近,我开始研究activiti框架,将其集成到我当前的项目中。在我们的项目中,我们使用teradata数据库。因此,我添加了activiti依赖项,并创建了简单的bpmn流程以进行测试。我用h2 inmemory数据库测试了这个过程,效果很好。但当将项目配置为使用teradata时,我在spring boot应用程序启动时遇到了异常。 我在google上只找到了这个主题: https://hub

  • 如果我想支持Chromium WebView,我将最低要求的SDK设置为API19:Android4.4(KitKat)。工作很好。 但是如果我将SDK设置为API 16 Android4.1,我就可以通过旧的WebKit WebView支持更多的设备。

  • 我想在优雅的关闭上做一些工作。 我尝试了如下所示的方法,但它不起作用。 我找到了一个解决方法(在ContextClosedEvent的@EventListener标记方法上放置方面注释),但我想了解它失败的原因(方法没有任何异常根本没有调用)。 就我对Spring 5的研究而言,我发现@PreDestroy由CommonAnnotationBeanPostProcessor处理,而@Aspect类