web-dev-qa-db-ja.com

_GLIBCXX_USE_CXX11_ABI、GCC 4.8およびABIの互換性

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以上に切り替えることができれば、これを機能させることができますか?

ありがとう!

14
Tanasis

Gcc 4.8.2でC++ 11 ABIを使用することは可能ですが、危険なハックです。ベンダーにC++ 03 ABI(-D_GLIBCXX_USE_CXX11_ABI=0)でコンパイルされたライブラリを出荷するか、GCC 5以上にアップグレードするようにベンダーに依頼できる場合は、はるかに良い方法です。

Libstdc ++ヘッダーとライブラリを使用できるようにgcc 5をダウンロードしてインストールし、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など)を再定義すると、構造のレイアウトを変更したり、プログラムとベンダーのライブラリ間のバイナリ非互換性を引き起こす可能性があるため、危険です。

結果の実行可能ファイルを実行するには、/usr/local/gcc-5.4.0/lib64/etc/ld.so.confに追加するか、-Wl,-rpath /usr/local/gcc-5.4.0/lib64を使用するなどして、動的リンカーが互換性のあるlibstdc ++を見つけるようにする必要もあります。

8
ecatmur