CMake编译遇到这种ABI不兼容问题不要慌

Posted 吴尼玛

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了CMake编译遇到这种ABI不兼容问题不要慌相关的知识,希望对你有一定的参考价值。

最近,在Linux下使用CMake编译程序的时候遇到一个问题,特此做一个记录。

事情是这样的,我编译的程序使用了2个第三方库,在写好CMakeLists后,启动编译,然后就报链接错误,一直报一堆找不到定义。

类似这样的一堆:

‘***** std::__cxx11 *******’未定义的引用

我仔细检查代码和CMakeLists,各种修改尝试无果。最后,在同事和搜索引擎的帮助下终于找到问题所在了。

问题出在我使用的GCC和第三方库编译的时候使用的GCC版本差异太大,两者的ABI不兼容。

参考GCC提供的手册<Dual ABI>,事情就清晰了。

在GCC5.1发布的libstdc++中,添加了新的特性,其中包括了std::string和std::list的新实现。新的实现使得两者都符合了C++11的标准。但是,为了与现存代码兼容,libstdc++保留了老的实现,与新实现并存。具体怎么实现的呢?相比于老的实现,GCC5.1添加了__cxx11命名空间,在新的实现中的string和list,实际上是std::__cxx11::string和std::__cxx11::list。新旧ABI不兼容。

那么,问题怎么解决呢?

也很简单,就是统一你的程序和第三方库使用的实现版本。如果你有第三方库源码,那统一重新编译下就行。如果第三方库是已经编译好的那也没关系,GCC为我们提供了一个选择新旧实现的宏定义-D_GLIBCXX_USE_CXX11_ABI。

-D_GLIBCXX_USE_CXX11_ABI=0 表示使用旧的实现
-D_GLIBCXX_USE_CXX11_ABI=1 表示使用新的实现

按需添加到CMakeLists中就可以了。

以上是关于CMake编译遇到这种ABI不兼容问题不要慌的主要内容,如果未能解决你的问题,请参考以下文章

使用 Android Studio 2.2 和 CMake 在 Android 中编译和使用依赖于 ABI 的可执行二进制文件

Eclipse 是不是必须处理 C++ ABI 兼容性问题?

Qt5 for Android:不兼容的 ABI

遇到 400500 错误千万不要慌!

VS编译失败但是错误输出页中没有显示任何错误信息

遇到 400500 错误千万不要慌!