web-dev-qa-db-ja.com

Apple LLVM 4.1のSTDリンカーエラー

最初にiOS(armv7)用に構築されたObjective-Cのビットを含むC++の大規模な静的ライブラリーがあります。

OS X(64ビットIntel x86_64)バージョンをビルドしましたが、OS Xアプリプロジェクト(Lion 10.7を対象としています)で使用しようとすると、数十のリンカーエラーが表示されましたが、そのほとんどは標準ライブラリに関するものですシンボル。

「私の」リンカの問題を解決する方法を知っていますが、以下にコピーされたSTDの問題は私を悩ませています。

"std::basic_filebuf<char, std::char_traits<char> >::is_open() const"
"std::basic_stringbuf<char, std::char_traits<char>, std::allocator<char> >::str() const"
"std::basic_ios<char, std::char_traits<char> >::widen(char) const"
"std::istream& std::istream::_M_extract<double>(double&)"
"std::ostream::put(char)"
"std::ostream::flush()"
"std::ostream& std::ostream::_M_insert<void const*>(void const*)"
"std::ostream& std::ostream::_M_insert<bool>(bool)"
"std::ostream& std::ostream::_M_insert<double>(double)"
"std::ostream& std::ostream::_M_insert<unsigned long>(unsigned long)"
"std::ostream::operator<<(int)"
"std::ostream::operator<<(short)"
"std::string::_Rep::_M_destroy(std::allocator<char> const&)"
"std::string::_Rep::_S_terminal"
"std::string::_Rep::_S_empty_rep_storage"
"std::string::_Rep::_S_create(unsigned long, unsigned long, std::allocator<char> const&)"
"std::string::append(char const*, unsigned long)"
"std::string::append(std::string const&)"
"std::string::assign(std::string const&)"
"std::string::reserve(unsigned long)"
"std::basic_string<char, std::char_traits<char>, std::allocator<char> >::basic_string(char const*, std::allocator<char> const&)"
"std::basic_string<char, std::char_traits<char>, std::allocator<char> >::basic_string(std::string const&)"
"std::basic_string<char, std::char_traits<char>, std::allocator<char> >::~basic_string()"
"std::basic_ofstream<char, std::char_traits<char> >::open(char const*, std::_Ios_Openmode)"
"std::basic_ofstream<char, std::char_traits<char> >::close()"
"std::basic_ofstream<char, std::char_traits<char> >::basic_ofstream()"
"std::basic_ofstream<char, std::char_traits<char> >::~basic_ofstream()"
"std::_List_node_base::hook(std::_List_node_base*)"
"std::_List_node_base::unhook()"
"std::basic_stringstream<char, std::char_traits<char>, std::allocator<char> >::basic_stringstream(std::string const&, std::_Ios_Openmode)"
"std::basic_stringstream<char, std::char_traits<char>, std::allocator<char> >::basic_stringstream(std::_Ios_Openmode)"
"std::basic_stringstream<char, std::char_traits<char>, std::allocator<char> >::~basic_stringstream()"
"std::basic_ostringstream<char, std::char_traits<char>, std::allocator<char> >::basic_ostringstream(std::_Ios_Openmode)"
"std::basic_ostringstream<char, std::char_traits<char>, std::allocator<char> >::~basic_ostringstream()"
"std::ios_base::Init::Init()"
"std::ios_base::Init::~Init()"
"std::basic_ios<char, std::char_traits<char> >::clear(std::_Ios_Iostate)"
"std::basic_ostream<char, std::char_traits<char> >& std::__ostream_insert<char, std::char_traits<char> >(std::basic_ostream<char, std::char_traits<char> >&, char const*, long)"
"std::_Rb_tree_decrement(std::_Rb_tree_node_base*)"
"std::_Rb_tree_increment(std::_Rb_tree_node_base const*)"
"std::_Rb_tree_increment(std::_Rb_tree_node_base*)"
"std::__throw_logic_error(char const*)"
"std::__throw_length_error(char const*)"
"std::__throw_out_of_range(char const*)"
"std::_Rb_tree_rebalance_for_erase(std::_Rb_tree_node_base*, std::_Rb_tree_node_base&)"
"std::_Rb_tree_insert_and_rebalance(bool, std::_Rb_tree_node_base*, std::_Rb_tree_node_base*, std::_Rb_tree_node_base&)"
"std::cerr"
"std::cout"

ビルド設定を確認しました。プロジェクトは標準ライブラリ(-stdlib=libc++)そして、main.cppでstd :: coutを問題なく使用できます。

ビルド設定のコンパイラをApple LLVM 4.1からLLVM GCC 4.2に変更したところ、問題が解決しました。引き続きApple LLVM 4.1を使用します。私はそれを修正しますか?

ありがとう!

23
antho

リンクされている標準ライブラリを変更して、libstdc++の代わりにlibc++を使用します。問題は、他のライブラリがg++ライブラリを使用するlibstdc++モードを使用してコンパイルされたことです。

次のサンプルコードを考えてみます。

dhcp-191:~/Development/testy/fred% cat fred.cpp
#include <iostream>
#include <string>
#include "fred.h"

using namespace std;

bool dofred(string &x)
{
    cout << x << endl;
    return true;
}
dhcp-191:~/Development/testy/fred% cat fred.h

#include <iostream>
#include <string>

bool dofred(std::string &x);

dhcp-191:~/Development/testy/fred% clang++ -stdlib=libc++ -shared -o fred.dylib fred.cpp
dhcp-191:~/Development/testy/fred% nm fred.dylib | c++filt | grep dofred
0000000000000fa0 T dofred(std::__1::basic_string<char, std::__1::char_traits<char>, std::__1::allocator<char> >&)
dhcp-191:~/Development/testy/fred% clang++ -stdlib=libstdc++ -shared -o fred.dylib fred.cpp
dhcp-191:~/Development/testy/fred% nm fred.dylib | c++filt | grep dofred                     
0000000000000e30 T dofred(std::string&)

完全に異なる2つのエクスポートされたシンボルを取得します。シンボルを使用しようとすると、同じ-stdlibフラグを使用するアプリはリンクできますが、そうでないアプリはリンクエラーを表示します。

45
Petesh

IOS 7では、チャートのライブラリを使用していますが、同じ問題があります。この場合、lib stdc ++は問題を解決しません。

Stdc ++。6.dylibをビルドフェーズに追加すると、シンボルが見つかります。

47
gzfrancisco

すべてのC++ファイルを別のライブラリに置いた後、この問題が発生しました。すべてのプロジェクトの設定をlibc ++を使用するように設定しましたが、リンカーはlibc ++とリンクしません。 C++ファイルをメインプロジェクトに追加すると、問題は解消します。これを修正するには、メインプロジェクトの「その他のリンカーフラグ」セクションに「-lc ++」を追加します。これにより、XCodeがlibc ++にリンクされます。

編集:他のポスターが言ったように、XCodeは正しく動作しているかもしれません。 C++ libソースコードが同じワークスペースにあるため、C++リンケージを追加することを知っていると期待していました。

11
jlukanta

同様の問題が発生し、「ビルド設定」、「Apple LLVM 5.1-言語-C++」の順に進み、「C++標準ライブラリ」をlibstdc ++に変更する必要がありました。

3
djacobs7

空の.cppファイルをプロジェクトに追加することもできます。これはxcodeをだましてC++ stdライブラリをロードします

2
Lee Irvine

Jlukantaへの応答:私は同じ問題を抱えていました。私は正しいSTDを選択するように注意してきましたが、それでもこれらのエラーが発生します。しかし、それはバグではありません。実際には理にかなっています。プロジェクトにC++コードがない場合、Xcodeがc ++ stdlibとリンクする必要があるのはなぜですか。

もちろん、これはプロジェクトにC++コードがなくてもC++ライブラリがある場合の問題です。

0
user2073196