前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >Android NDK R19 交叉编译 thrift C++ 库

Android NDK R19 交叉编译 thrift C++ 库

作者头像
10km
发布2020-12-07 15:10:30
1.6K0
发布2020-12-07 15:10:30
举报
文章被收录于专栏:10km的专栏10km的专栏10km的专栏

在NDK下编译thrift C++库,先要要编译android版的boost,这个不是本文讨论的内容,关于编译android 版本的boost,参见这个开源项目 Boost-for-Android,很好用。

我用Boost-for-Androidk顺利编译了boost 1.69.(Android NDK r19)

有了android版的boost就可以编译thrift C++库了。在这里走了不少弯路。

先说明一下我的编译环境: 操作系统:Windows 7 cmake 3.11.1,编译thrift需要cmake,因为boost 1.69版本较高,所以对cmake版本有要求 Android NDK r19

下面是完整的bat编译脚本: ndk_build_thrift.bat

:: NDK 交叉编译thrift c++ 库
:: NDK version 19 or above required
:: cmake 3.11.1 or ablove required
:: Flags you can use to change this behavior:
::
::   /DEBUG            - debug instead of release
::   /NOBUILD          - skip cmake build - useful if you just
::                       want to generate a solution
::

@ECHO OFF
SETLOCAL EnableDelayedExpansion
SET sh_folder=%~dp0
SET PACKAGE_NAME=thrift

:: 检测是否安装NDK,没有安装NDK则报错退出
IF NOT DEFINED ANDROID_NDK (
	echo "ERROR:environment variable ANDROID_NDK not defined" 
	EXIT /B 255
	)

IF NOT EXIST "%ANDROID_NDK%" (
	echo "ERROR:invalid environment variable ANDROID_NDK=%ANDROID_NDK%" 
	EXIT /B 255
	)

SET BUILDTYPE=Release
SET OPT_BUILD=1
SET OPT_COMPILER=OFF

:: parse command arguments
:loop
IF x%1 == x goto :pare_end
IF /I "%1" == "/DEBUG"        SET BUILDTYPE=Debug
IF /I "%1" == "/NOBUILD"      SET OPT_BUILD=0
SHIFT
goto :loop
:pare_end

:: 目标平台
IF NOT DEFINED ANDROID_ABI SET ANDROID_ABI=armeabi-v7a

IF "%ANDROID_ABI%"=="armeabi-v7a" ( 
	SET ANDROID_TOOLCHAIN_NAME=arm-linux-androideabi
	SET ARCH=ARMEABI-V7A
) ELSE IF "%ANDROID_ABI%"=="arm64-v8a" (
	SET ANDROID_TOOLCHAIN_NAME="aarch64-linux-android"
	SET ARCH=ARM64-V8A
) ELSE IF "%ANDROID_ABI%"=="x86" (
	SET ANDROID_TOOLCHAIN_NAME=i686-linux-android
	SET ARCH=X86
) ELSE IF "%ANDROID_ABI%"=="x86_64" (
	SET ANDROID_TOOLCHAIN_NAME=x86_64-linux-android
	SET ARCH=x64
) ELSE (
  	ECHO "Invalid Android ABI: %ANDROID_ABI%." 
  	EXIT /B 255
)

:: boost 安装位置
IF NOT DEFINED BOOST_ROOT SET BOOST_ROOT="%sh_folder%dist\boost_1_69_0_android\%ANDROID_ABI%"
:: thrift 源码位置
IF NOT DEFINED THRIFT_FOLDER SET THRIFT_FOLDER="%sh_folder%thrift"
:: 编译输出路径
SET BUILDDIR=%sh_folder%build\%PACKAGE_NAME%-%ANDROID_TOOLCHAIN_NAME%
:: 安装路径
SET OUTDIR=%sh_folder%dist\%PACKAGE_NAME%-%ANDROID_TOOLCHAIN_NAME%

IF NOT EXIST "%BOOST_ROOT%" (
	ECHO NOT FOUND BOOST FOLDER %BOOST_ROOT%
	EXIT /B 255
)

ECHO =========================================================================
ECHO     Configuration: %PACKAGE_NAME% %ANDROID_TOOLCHAIN_NAME%:%BUILDTYPE%
ECHO   Build Directory: %BUILDDIR%
ECHO Install Directory: %OUTDIR%
ECHO   Boost Directory: %BOOST_ROOT%
ECHO =========================================================================

IF EXIST "%BUILDDIR%" ( RMDIR "%BUILDDIR%" /s/q || EXIT /B )
MKDIR "%BUILDDIR%" || EXIT /B
CD "%BUILDDIR%" || EXIT /B

CMAKE.EXE %THRIFT_FOLDER% ^
    -G "Unix Makefiles" ^
    -DCMAKE_INSTALL_PREFIX=%OUTDIR% ^
    -DCMAKE_BUILD_TYPE=%BUILDTYPE% ^
    -DBOOST_ROOT=%BOOST_ROOT% ^
 	-DBoost_ADDITIONAL_VERSIONS="1.69.0;1.69" ^
  	-DBoost_NO_SYSTEM_PATHS=ON ^
	-DBoost_COMPILER=-clang ^
   	-DWITH_BOOSTTHREADS=ON ^
   	-DWITH_C_GLIB=OFF ^
	-DWITH_LIBEVENT=OFF ^
    -DWITH_OPENSSL=OFF ^
    -DWITH_SHARED_LIB=OFF ^
	-DWITH_STATIC_LIB=ON ^
    -DWITH_JAVA=OFF ^
    -DWITH_PYTHON=OFF ^
	-DWITH_HASKELL=OFF ^
    -DWITH_LIBEVENT=OFF ^
    -DWITH_ZLIB=OFF ^
    -DBUILD_COMPILER=OFF ^
    -DBUILD_TESTING=OFF ^
    -DBUILD_EXAMPLES=OFF ^
    -DBUILD_TUTORIALS=OFF ^
   	-DCMAKE_POSITION_INDEPENDENT_CODE=ON ^
    -DCMAKE_CXX_COMPILER_ARCHITECTURE_ID=%ARCH% ^
	-DCMAKE_FIND_ROOT_PATH_MODE_INCLUDE=BOTH ^
	-DCMAKE_FIND_ROOT_PATH_MODE_LIBRARY=BOTH ^
    -DCMAKE_TOOLCHAIN_FILE=%ANDROID_NDK%\build\cmake\android.toolchain.cmake ^
     || EXIT /B


IF %OPT_BUILD% == 1 (
  CD %BUILDDIR%
  CMAKE.EXE --build . --target install -- -j8 || EXIT /B
)

ENDLOCAL

说明:

-DBoost_ADDITIONAL_VERSIONS=“1.69.0;1.69”

因为 cmake 3.11.1中的 FindBoost.cmake 不能识别boost 1.69 版本,所以这里要用定义 Boost_ADDITIONAL_VERSIONS 让 FindBoost.cmake 能正确识别 1.69 版本. cmake 3.17.1 以后的版本不需要指定该参数.

-DBoost_COMPILER=-clang

因为 cmake 3.11.1 中的 FindBoost.cmake 还不支持 clang 编译器自动识别,所以这里通过定义 Boost_COMPILER 指定搜索库(library)时的库名后缀. 使用 camek 3.17.1 以后的版本可以不指定此参数

-DCMAKE_CXX_COMPILER_ARCHITECTURE_ID=%ARCH%

boost 1.69 版本以后生成的库名中包含CPU体系的识别码,比如 libboost_atomic-clang-mt-a32-1_69.a,这里的a32代表arm32. 在FindBoost.cmake中这个值由 CMAKE_CXX_COMPILER_ARCHITECTURE_ID 计算得出,如果不指定 CMAKE_CXX_COMPILER_ARCHITECTURE_ID,FindBoost.cmake 就无法正确计算出搜索库的文件名. cmake 3.17.1版本以后的 FindBoost.cmake 增加了 Boost_ARCHITECTURE 参数,可以直接指定这个识别码,比如 -x32,代码X86,-a32代表arm32

-DCMAKE_FIND_ROOT_PATH_MODE_INCLUDE=BOTH -DCMAKE_FIND_ROOT_PATH_MODE_LIBRARY=BOTH

工具链文件 android.toolchain.cmake 中上面的参数都定义为ONLY,这里指为BOTH,否则 FindBoost.cmake 不能正确找到 include 文件夹和library

本文参与 腾讯云自媒体分享计划,分享自作者个人站点/博客。
原始发表:2020-12-03 ,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 说明:
    • -DBoost_ADDITIONAL_VERSIONS=“1.69.0;1.69”
      • -DBoost_COMPILER=-clang
        • -DCMAKE_CXX_COMPILER_ARCHITECTURE_ID=%ARCH%
          • -DCMAKE_FIND_ROOT_PATH_MODE_INCLUDE=BOTH -DCMAKE_FIND_ROOT_PATH_MODE_LIBRARY=BOTH
          领券
          问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档