2012-06-18 18 views
8

Tengo problemas para superar la fase de enlace en mi programa C++ debido a problemas con Boost 1.49. He cambiado a C++ (-std=c++11 -libc=libc++) que funciona bien para otra pieza de código (que también usa boost). Boost se instaló mediante homebrew con:Problemas de vinculación con boost :: program_options en OSX usando LLVM

brew install boost --universal --with-mpi --with-icu 

El problema comienza con boost::program_options. Consigo los errores de enlace de esta manera:

"boost::program_options::validate(boost::any&, std::__1::vector<std::__1::basic_string<char, std::__1::char_traits<char>, std::__1::allocator<char> >, std::__1::allocator<std::__1::basic_string<char, std::__1::char_traits<char>, std::__1::allocator<char> > > > const&, std::__1::basic_string<char, std::__1::char_traits<char>, std::__1::allocator<char> >*, int)", referenced from: 

... etc. ... 

ld: symbol(s) not found for architecture x86_64 
clang: error: linker command failed with exit code 1 (use -v to see invocation) 

Esto es un poco extraño, porque hacer una nm en la biblioteca utilizada revela, que el símbolo aparece estar allí:

nm -U /usr/local/lib/libboost_program_options-mt.dylib | grep validate 
0000000000019880 - 01 0000 FUN __ZN5boost15program_options8validateERNS_3anyERKSt6vectorISbIwSt11char_traitsIwESaIwEESaIS7_EEPSsi 
0000000000019880 T __ZN5boost15program_options8validateERNS_3anyERKSt6vectorISbIwSt11char_traitsIwESaIwEESaIS7_EEPSsi 
00000000000199e0 - 01 0000 FUN __ZN5boost15program_options8validateERNS_3anyERKSt6vectorISbIwSt11char_traitsIwESaIwEESaIS7_EEPbi 
00000000000199e0 T __ZN5boost15program_options8validateERNS_3anyERKSt6vectorISbIwSt11char_traitsIwESaIwEESaIS7_EEPbi 
0000000000019930 T __ZN5boost15program_options8validateERNS_3anyERKSt6vectorISsSaISsEEPSsi 
0000000000019930 - 01 0000 FUN __ZN5boost15program_options8validateERNS_3anyERKSt6vectorISsSaISsEEPSsi 
0000000000019c70 - 01 0000 FUN __ZN5boost15program_options8validateERNS_3anyERKSt6vectorISsSaISsEEPbi 
0000000000019c70 T __ZN5boost15program_options8validateERNS_3anyERKSt6vectorISsSaISsEEPbi 

Ya he probado persuadir a homebrew para que compile boost con clang en lugar de gcc configurando CXX y CXX_FLAGS en consecuencia antes de la instalación. Sin embargo, no estoy seguro de haber tenido éxito.

Punteros muy apreciados.

+0

La misma pregunta que: http://stackoverflow.com/questions/8454329/why-cant-clang-with-libc-in-c0x-mode-link-this-boostprogram-options-examp –

Respuesta

8

Necesitará recompilar boost con clang y std11 flags, la librería libC++ no es compatible con binario con libstdC++ instalado en OSX (versión muy antigua de gcc antes de cambiar a gpl3). Si su versión de clang es 3.1 o superior, entonces puede usarla (de lo contrario, cambie C++ 11 a C++ 0x para versiones anteriores).

./bootstrap.sh 
mkdir build 
sudo ./bjam toolset=clang cxxflags="-std=c++0x -stdlib=libc++" variant=release link=static threading=multi runtime-link=shared --build-dir=Build --layout=system --without-mpi --without-python install --prefix=/usr/local 

Por supuesto, puede alterar cualquiera de éstos como desee, excepto

CXXFLAGS conjunto de herramientas = clang = "- std = C++ 0x = -stdlib libC++"

Este debería funcionar para usted.

+0

Hola, gracias por la puntero. Pensé tanto. Sin embargo, parece que no puedo hacer que Boost compile bibliotecas dinámicas ahora. Intenté esto: 'sudo ./bjam toolset = clang cxxflags = "- std = C++ 11 -stdlib = libC++" variant = release link = shared threading = multi runtime-link = shared --build-dir = Build --layout = sistema --without-mpi --without-python install --prefix =/usr/local ' pero solo recibimos los estáticos. –

+1

Finalmente obtuve la respuesta a través del Boost.Build-Forum: http://boost.2283326.n4.nabble.com/Problems-building-Boost-with-clang-toolchain-and-C-11-td4631556.html En resumen: me faltaban 'linkflags' al llamar a bjam: ./b2 toolset = clang cxxflags = "- std = C++ 11 -stdlib = libC++" linkflags = "- stdlib = libC++" –

3

Me gustaría compartir mi experiencia (moderadamente dolorosa) de compilar Boost 1.54 en Mac OS X 10.8.5 con clang 5.0.0 proporcionado por Xcode 5.0. Si desea las funciones de C++ 11, es muy importante compilar y vincular con clang++, no con clang.

Ilustración: tomar el siguiente programa simple:

#include <iostream> 
#include <string> 

int main(int argc, char *argv[]) { 
    std::string str = "OK"; 
    std::cout << str << std::endl; 
    return 0; 
} 

se puede construir con el siguiente comando: sin embargo

clang++ -std=c++11 -stdlib=libc++ clangstr.cc -o clangstr

, si se intenta esto en su lugar:

clang -std=c++11 -stdlib=libc++ clangstr.cc -o clangstr

entonces obtienes errores del enlazador. Tenga en cuenta que la página de manual de clang dice que el idioma se selecciona mediante la opción -std=, pero esto claramente no es suficiente.

La lección es que tenemos que decir bjam para usar explícitamente clang++ al compilar Boost con compatibilidad con C++ 11.

Siguiendo this very useful post, pongo el siguiente en mi tools/build/v2/user-config.jam:

using clang : 11 
    : "/usr/bin/clang++" 
    : <cxxflags>"-std=c++11 -stdlib=libc++ -ftemplate-depth=512" <linkflags>"-stdlib=libc++" 
    ; 

Entonces me encontré con ./b2 clean, entonces construyeron Boost con el siguiente comando:

mkdir -p build/clangstage/ 
./b2 -j8 --build-dir=build --stagedir=build/clangstage toolset=clang-11 define=BOOST_SYSTEM_NO_DEPRECATED variant=release threading=multi address-model=64 stage 

Esto construye la de 64 bits estática y bibliotecas dinámicas con soporte multihilo. Si necesita un conjunto diferente, cambie el comando de arriba según corresponda.

Cuestiones relacionadas