我如何在 Red Hat 上强制使用 cxx11 ABI?

How do I force cxx11 ABI on Red Hat?

我正在使用 gcc 7.3 在 Ubuntu 16.04 和 Red Hat 7 上构建一个小型 .so 库。当我使用 nm 命令检查导出符号名称时,我发现在 Ubuntu 上编译的库使用 cxx11 ABI,但在 RedHat 上编译的库没有。

例如,在 Ubuntu 上编译的函数的导出符号如下所示。

_Z12customLoad3DPKN8nlohmann10basic_jsonISt3mapSt6vectorNSt7__cxx1112basic_stringIcSt11char_traitsIcESaIcEEEblmdSaNS_14adl_serializerEEEPP11IVolumeDataPSA_

但是在 RedHat 上编译时相同的导出符号看起来像这样。

_Z12customLoad3DPKN8nlohmann10basic_jsonISt3mapSt6vectorSsblmdSaNS_14adl_serializerEEEPP11IVolumeDataPS4_

两个库使用相同的 makefile。该代码使用#define _GLIBCXX_USECXX11_ABI 1 和命令行选项 -std=c++11。我也尝试过使用具有不同值的 -fabi-version 选项,但都没有效果。我无法用 #ifdef __cplusplus extern "C" 方法解决这个问题,因为这些函数使用模板 class 参数。

如何强制 RedHat 上的 gcc 使用 cxx11 ABI?我不能使用双 ABI 链接,因为 .so 库用作应用程序的插件,该应用程序在运行时使用硬编码的损坏名称列表链接到库中的函数。该插件无法在 Red Hat 上运行,因为损坏的名称与程序的预期不符。我该如何解决这个问题?

谢谢!

Red Hat Enterprise 不支持 libstdc++ C++11 ABI Linux 7. 您有两个选择:

  • 升级到 Red Hat Enterprise Linux 8. 它的系统编译器默认为较新的 ABI。

  • 在 Red Hat Enterprise Linux 7 上重建应用程序,使用 Developer Toolset。混合联动模型确保应用程序将 运行 在 Red Hat Enterprise Linux 7. 只有旧的 ABI 可用,但是由于重建,这通常无关紧要,因为结果是内部一致的。