10

我正在尝试为使用 Boost 库的代码创建一个 CMake 项目。我想为 Android 构建它,我使用这个项目作为参考:https ://code.google.com/p/android-cmake/ 。

我想使用我之前为 android 构建的 Boost 静态库,而不是让它们成为要编译的项目的一部分。

这是我使用 Boost 的项目部分的 CMakeLists.txt

#BOOST

set(BOOST_ROOT /home/neb/workspace/SDLActivityCMAKE/jni/boost)

set(Boost_INCLUDE_DIR /home/neb/workspace/SDLActivityCMAKE/jni/boost/include/boost-1_53)
set(BOOST_INCLUDEDIR /home/neb/workspace/SDLActivityCMAKE/jni/boost/include/boost-1_53)
set(Boost_LIBRARY_DIR /home/neb/workspace/SDLActivityCMAKE/jni/boost/lib)
set(BOOST_LIBRARYDIR /home/neb/workspace/SDLActivityCMAKE/jni/boost/lib)

set(Boost_USE_STATIC_LIBS ON)
set(USE_STATIC_BOOST ON)
set(Boost_USE_STATIC_RUNTIME ON)
set(BOOST_COMPILER gcc)

find_package(Boost COMPONENTS "thread-gcc-mt-1_53" "date_time-gcc-mt-1_53" REQUIRED)

include_directories(${Boost_INCLUDE_DIR})
link_directories(${Boost_LIBRARY_DIR})
#BOOSTEND

add_executable(main main.cpp)
target_link_libraries( main ${Boost_LIBRARIES} )

当我运行常规cmake命令(cmake ..来自android项目中jni文件夹的子目录)没有问题时,cmake会找到库。当我让 cmake 使用上述项目提供的 android.toolchain.cmake 时,我得到了这个(我正在使用 -DBoost_DEBUG 选项)。

-- [ /usr/share/cmake-2.8/Modules/FindBoost.cmake:566 ] _boost_TEST_VERSIONS = 1.56.0;1.56;1.55.0;1.55;1.54.0;1.54;1.53.0;1.53;1.52.0;1.52;1.51.0;1.51;1.50.0;1.50;1.49.0;1.49;1.48.0;1.48;1.47.0;1.47;1.46.1;1.46.0;1.46;1.45.0;1.45;1.44.0;1.44;1.43.0;1.43;1.42.0;1.42;1.41.0;1.41;1.40.0;1.40;1.39.0;1.39;1.38.0;1.38;1.37.0;1.37;1.36.1;1.36.0;1.36;1.35.1;1.35.0;1.35;1.34.1;1.34.0;1.34;1.33.1;1.33.0;1.33
-- [ /usr/share/cmake-2.8/Modules/FindBoost.cmake:568 ] Boost_USE_MULTITHREADED = TRUE
-- [ /usr/share/cmake-2.8/Modules/FindBoost.cmake:570 ] Boost_USE_STATIC_LIBS = ON
-- [ /usr/share/cmake-2.8/Modules/FindBoost.cmake:572 ] Boost_USE_STATIC_RUNTIME = ON
-- [ /usr/share/cmake-2.8/Modules/FindBoost.cmake:574 ] Boost_ADDITIONAL_VERSIONS = 
-- [ /usr/share/cmake-2.8/Modules/FindBoost.cmake:576 ] Boost_NO_SYSTEM_PATHS = 
-- [ /usr/share/cmake-2.8/Modules/FindBoost.cmake:644 ] Declared as CMake or Environmental Variables:
-- [ /usr/share/cmake-2.8/Modules/FindBoost.cmake:646 ]   BOOST_ROOT = /home/neb/workspace/SDLActivityCMAKE/jni/boost
-- [ /usr/share/cmake-2.8/Modules/FindBoost.cmake:648 ]   BOOST_INCLUDEDIR = /home/neb/workspace/SDLActivityCMAKE/jni/boost/include/boost-1_53
-- [ /usr/share/cmake-2.8/Modules/FindBoost.cmake:650 ]   BOOST_LIBRARYDIR = /home/neb/workspace/SDLActivityCMAKE/jni/boost/lib
-- [ /usr/share/cmake-2.8/Modules/FindBoost.cmake:652 ] _boost_TEST_VERSIONS = 1.56.0;1.56;1.55.0;1.55;1.54.0;1.54;1.53.0;1.53;1.52.0;1.52;1.51.0;1.51;1.50.0;1.50;1.49.0;1.49;1.48.0;1.48;1.47.0;1.47;1.46.1;1.46.0;1.46;1.45.0;1.45;1.44.0;1.44;1.43.0;1.43;1.42.0;1.42;1.41.0;1.41;1.40.0;1.40;1.39.0;1.39;1.38.0;1.38;1.37.0;1.37;1.36.1;1.36.0;1.36;1.35.1;1.35.0;1.35;1.34.1;1.34.0;1.34;1.33.1;1.33.0;1.33
-- [ /usr/share/cmake-2.8/Modules/FindBoost.cmake:734 ] location of version.hpp: /home/neb/workspace/SDLActivityCMAKE/jni/boost/include/boost-1_53/boost/version.hpp
-- [ /usr/share/cmake-2.8/Modules/FindBoost.cmake:754 ] version.hpp reveals boost 1.53.0
-- [ /usr/share/cmake-2.8/Modules/FindBoost.cmake:786 ] guessed _boost_COMPILER = -gcc46
-- [ /usr/share/cmake-2.8/Modules/FindBoost.cmake:796 ] _boost_MULTITHREADED = -mt
-- [ /usr/share/cmake-2.8/Modules/FindBoost.cmake:839 ] _boost_RELEASE_ABI_TAG = -s
-- [ /usr/share/cmake-2.8/Modules/FindBoost.cmake:841 ] _boost_DEBUG_ABI_TAG = -sd
-- [ /usr/share/cmake-2.8/Modules/FindBoost.cmake:884 ] _boost_LIBRARY_SEARCH_DIRS = /home/neb/workspace/SDLActivityCMAKE/jni/boost/lib;/home/neb/workspace/SDLActivityCMAKE/jni/boost/lib;/home/neb/workspace/SDLActivityCMAKE/jni/boost/stage/lib;/home/neb/workspace/SDLActivityCMAKE/jni/boost/include/boost-1_53/lib;/home/neb/workspace/SDLActivityCMAKE/jni/boost/include/boost-1_53/../lib;/home/neb/workspace/SDLActivityCMAKE/jni/boost/include/boost-1_53/stage/lib;C:/boost/lib;C:/boost;/boost/boost_1_53_0/lib;/boost/boost_1_53/lib;/boost/lib;/boost;/sw/local/lib
-- [ /usr/share/cmake-2.8/Modules/FindBoost.cmake:962 ] Searching for THREAD-GCC-MT-1_53_LIBRARY_RELEASE: boost_thread-gcc-mt-1_53-gcc46-mt-s-1_53;boost_thread-gcc-mt-1_53-gcc46-mt-s;boost_thread-gcc-mt-1_53-mt-s-1_53;boost_thread-gcc-mt-1_53-mt-s;boost_thread-gcc-mt-1_53
-- [ /usr/share/cmake-2.8/Modules/FindBoost.cmake:994 ] Searching for THREAD-GCC-MT-1_53_LIBRARY_DEBUG: boost_thread-gcc-mt-1_53-gcc46-mt-sd-1_53;boost_thread-gcc-mt-1_53-gcc46-mt-sd;boost_thread-gcc-mt-1_53-mt-sd-1_53;boost_thread-gcc-mt-1_53-mt-sd;boost_thread-gcc-mt-1_53-mt;boost_thread-gcc-mt-1_53
-- [ /usr/share/cmake-2.8/Modules/FindBoost.cmake:962 ] Searching for DATE_TIME-GCC-MT-1_53_LIBRARY_RELEASE: boost_date_time-gcc-mt-1_53-gcc46-mt-s-1_53;boost_date_time-gcc-mt-1_53-gcc46-mt-s;boost_date_time-gcc-mt-1_53-mt-s-1_53;boost_date_time-gcc-mt-1_53-mt-s;boost_date_time-gcc-mt-1_53
-- [ /usr/share/cmake-2.8/Modules/FindBoost.cmake:994 ] Searching for DATE_TIME-GCC-MT-1_53_LIBRARY_DEBUG: boost_date_time-gcc-mt-1_53-gcc46-mt-sd-1_53;boost_date_time-gcc-mt-1_53-gcc46-mt-sd;boost_date_time-gcc-mt-1_53-mt-sd-1_53;boost_date_time-gcc-mt-1_53-mt-sd;boost_date_time-gcc-mt-1_53-mt;boost_date_time-gcc-mt-1_53
-- [ /usr/share/cmake-2.8/Modules/FindBoost.cmake:1108 ] Boost_FOUND = FALSE
CMake Error at /usr/share/cmake-2.8/Modules/FindBoost.cmake:1195 (message):
  Unable to find the requested Boost libraries.

  Boost version: 1.53.0

  Boost include path:
  /home/neb/workspace/SDLActivityCMAKE/jni/boost/include/boost-1_53

  The following Boost libraries could not be found:

          boost_thread-gcc-mt-1_53
          boost_date_time-gcc-mt-1_53

  No Boost libraries were found.  You may need to set BOOST_LIBRARYDIR to the
  directory containing Boost libraries or BOOST_ROOT to the location of
  Boost.
Call Stack (most recent call first):
  src/CMakeLists.txt:15 (find_package)


-- Configuring incomplete, errors occurred!

所以我看到 CMake 正在这个确切的目录中搜索库: /home/neb/workspace/SDLActivityCMAKE/jni/boost/lib 其中有编译的静态库:

neb@neb-VirtualBox:~/workspace/SDLActivityCMAKE/jni/boost/lib$ ls -l
total 24440
-rwxrwxrwx 1 neb neb   557190 sie 27 12:36 libboost_date_time-gcc-mt-1_53.a
-rwxrwxrwx 1 neb neb  1348474 sie 27 12:36 libboost_filesystem-gcc-mt-1_53.a
-rwxrwxrwx 1 neb neb   827216 sie 27 12:36 libboost_iostreams-gcc-mt-1_53.a
-rwxrwxrwx 1 neb neb  6813972 sie 27 12:36 libboost_program_options-gcc-mt-1_53.a
-rwxrwxrwx 1 neb neb 12927086 sie 27 12:37 libboost_regex-gcc-mt-1_53.a
-rwxrwxrwx 1 neb neb  1197696 sie 27 12:37 libboost_signals-gcc-mt-1_53.a
-rwxrwxrwx 1 neb neb   159462 sie 27 12:36 libboost_system-gcc-mt-1_53.a
-rwxrwxrwx 1 neb neb  1177994 sie 27 12:37 libboost_thread-gcc-mt-1_53.a`

我设置了指向我能找到的 BOOST 目录的所有变量,我设置了变量说 Boost 库应该是静态的,但它仍然找不到它们。我还可以做些什么?

4

4 回答 4

7

使用 CMake 寻找 Boost

首先让我说让 CMake 找到 Boost 可能很困难,因为它的FindBoost.cmake自动化会咬你,而 CMake 的制造商 KitWare 拥有我遇到过的任何流行项目中最糟糕的文档。FindBoost.cmake如果它失败,你将不得不通过源来调试它。

交叉编译的 Boost

假设 1.62.0,您安装的交叉编译 boost 的目录如下所示

boost-arm-build/include
boost-arm-build/include/boost-1_62
boost-arm-build/lib

从 Project-Build 目录调用 CMake

cmake \
   -DBoost_NO_SYSTEM_PATHS=TRUE \
   -DBoost_ADDITIONAL_VERSIONS="1.62" \
   -DCMAKE_TOOLCHAIN_FILE=../android.toolchain \
   -DBOOST_ROOT="../boost-arm-build" \
   ../Project/

-DBoost_ADDITIONAL_VERSIONS- CMake 使用硬编码的字符串列表查找 Boost,如果您的 CMake 比您正在构建的 Boost 版本旧,它将不知道在包含目录中查找的位置,因此您需要将其传递。虽然版本实际上是 1.62.0,但 boost 将包含在 boost-1_62 中,所以我改用 1.62。FindBoost.cmake你会发现每个版本都被添加了两次,因为这个原因在“1.62.0”“1.62”中。

您还应该检查您构建的库的文件名,对于我的示例,让我们看一下system

libboost_system-gcc-mt-1_62.a
libboost_system-gcc-mt-1_62.so    # Symbolic link to libboost_system-gcc-mt-1_62.so.1.62.0
libboost_system-gcc-mt-1_62.so.1.62.0
libboost_system-gcc-mt-d-1_62.a
libboost_system-gcc-mt-d-1_62.so  # Symbolic link to libboost_system-gcc-mt-d-1_62.so.1.62.0
libboost_system-gcc-mt-d-1_62.so.1.62.0
  • gcc- 使用的编译器
  • mt- 多线程
  • d- 调试
  • 1_62- 显然是 boost 的版本

如果您看到 libboost_system.a ,则表示您未能使用 NDK 构建 boost--with-layout=versioned并且可能应该查看我的SO Answer on building boost using the NDK。如果你看到它clang不是gcc很好,你已经用 clang 构建了 boost,但你可能需要使用 Boost_COMPILER 标志通知 CMake 存在 clang。

-DBoost_COMPILER="-clang"

android.toolchain 文件

set(CMAKE_SYSTEM_NAME Android)

set(CROSS_COMPILER_DIR /opt/ndk-R13-standalone)
set(CMAKE_C_COMPILER ${CROSS_COMPILER_DIR}/bin/arm-linux-androideabi-gcc)
set(CMAKE_CXX_COMPILER ${CROSS_COMPILER_DIR}/bin/arm-linux-androideabi-g++)

set(ANDROID_NATIVE_API_LEVEL android-19)
set(ANDROID_ABI armeabi-v7a)
set(ANDROID_STL gnustl_static)

set(CMAKE_SYSROOT ${CROSS_COMPILER_DIR}/sysroot)
set(CMAKE_FIND_ROOT_PATH ${CMAKE_SYSROOT})

# search for programs in the build host directories
set(CMAKE_FIND_ROOT_PATH_MODE_PROGRAM NEVER)
于 2016-10-14T10:05:55.510 回答
4

我有同样的问题。使用find_host_package(Boost COMPONENTS ..)而不是find_package(Boost COMPONENTS ..)为我解决了这个问题。

(在这里找到:http ://code.google.com/p/android-cmake/issues/detail?id=16 )

于 2013-12-09T17:06:51.830 回答
2

这里有几个问题。

看起来您的 boost 库是在没有构建的情况下构建的,runtime-link=static即它们没有静态链接到 C++ 标准库。如果是,他们名字的后缀应该包含一个s(参见关于库命名的 boost 文档

另一个问题是 CMakeFindBoost module期望COMPONENTS参数的格式为“date_time”、“thread”,而不是您提供的全名。

另一个更小的一点是您不需要设置Boost_INCLUDE_DIRor Boost_LIBRARY_DIR; 如果成功,这些将由 CMake 模块设置。设置它们不应该引起任何问题,但这只是不必要的混乱。

此外,对于BOOST_ROOTBOOST_INCLUDEDIRBOOST_LIBRARYDIR,这些通常会通过命令行使用该-D选项进行设置,或者设置为环境变量,以避免将特定于您的机器的路径硬编码到 CMakeLists.txt 中。

最后一点是您应该删除link_directories呼叫。它自己的文档不鼓励使用它,并且不需要它,因为您已经在target_link_libraries调用中将完整路径传递给了 boost 库。

所以,最终版本应该更像:

#BOOST
set(Boost_USE_STATIC_LIBS ON)
set(USE_STATIC_BOOST ON)  # Not sure if you really need this later.  If not, delete.
set(Boost_USE_STATIC_RUNTIME OFF)
set(Boost_COMPILER -gcc)

find_package(Boost COMPONENTS thread date_time REQUIRED)

include_directories(${Boost_INCLUDE_DIRS})
#BOOSTEND

add_executable(main main.cpp)
target_link_libraries(main ${Boost_LIBRARIES})

你可以像这样调用CMake:

cmake . -DBOOST_ROOT=<path to root> -DBOOST_INCLUDEDIR=... -DBOOST_LIBRARYDIR=...
于 2013-09-10T12:42:44.033 回答
0

我遇到了同样的问题。我使用此存储库Boost for android构建了 boost,但我只有静态库。

我不认为你真的需要使用 cmake 的 find_package。这就是我在 CMakeList.txt 中简单使用它的方式

add_library(boost_system-gcc-mt-1_55  STATIC IMPORTED  GLOBAL)
set_target_properties( # Specifies the target library.
                       boost_system-gcc-mt-1_55

                       # Specifies the parameter you want to define.
                       PROPERTIES IMPORTED_LOCATION

                       # Provides the path to the library you want to import.
                       ${CMAKE_SOURCE_DIR}/src/main/jni/boost/lib/libboost_system-gcc-mt-1_55.a )
SET(Boost_LIBRARY_DIR ${Boost_LIBRARY_DIR} boost_system-gcc-mt-1_55)
# You can make this include just once
include_directories("src/main/jni/boost/include/boost-1_55")
target_link_libraries(${MyNDKSampleLibs}
                   ${Boost_LIBRARY_DIR}
                   ${log-lib} 

我在这里唯一可以提到的是不要忘记函数中的 cmake ${CMAKE_SOURCE_DIR}var set_target_properties

于 2017-04-05T18:41:13.803 回答