forked from OSchip/llvm-project
199 lines
9.4 KiB
CMake
199 lines
9.4 KiB
CMake
# Toolchain config for Android standalone NDK.
|
|
#
|
|
# Usage:
|
|
# build host llvm and clang first
|
|
# cmake -DCMAKE_TOOLCHAIN_FILE=../lldb/cmake/platforms/Android.cmake \
|
|
# -DANDROID_TOOLCHAIN_DIR=<toolchain_dir> \
|
|
# -DANDROID_ABI=<target_abi> \
|
|
# -DCMAKE_CXX_COMPILER_VERSION=<gcc_version> \
|
|
# -DLLVM_TARGET_ARCH=<llvm_target_arch> \
|
|
# -DLLVM_TARGETS_TO_BUILD=<llvm_targets_to_build> \
|
|
# -DLLVM_TABLEGEN=<path_to_llvm-tblgen> \
|
|
# -DCLANG_TABLEGEN=<path_to_clang-tblgen>
|
|
#
|
|
# Current Support:
|
|
# ANDROID_ABI = x86, x86_64
|
|
# CMAKE_CXX_COMPILER_VERSION = 4.9
|
|
# LLVM_TARGET_ARCH = X86
|
|
# LLVM_TARGETS_TO_BUILD = X86
|
|
# LLVM_TABLEGEN = path to host llvm-tblgen
|
|
# CLANG_TABLEGEN = path to host clang-tblgen
|
|
|
|
if( DEFINED CMAKE_CROSSCOMPILING )
|
|
return()
|
|
endif()
|
|
|
|
get_property( IS_IN_TRY_COMPILE GLOBAL PROPERTY IN_TRY_COMPILE )
|
|
if( IS_IN_TRY_COMPILE )
|
|
# this seems necessary and works fine but I'm unsure if it breaks anything
|
|
return()
|
|
endif()
|
|
|
|
set( CMAKE_SYSTEM_NAME Linux )
|
|
include( CMakeForceCompiler )
|
|
|
|
# flags and definitions
|
|
remove_definitions( -DANDROID -D__ANDROID__ )
|
|
add_definitions( -DANDROID -D__ANDROID_NDK__ -DLLDB_DISABLE_LIBEDIT )
|
|
set( ANDROID True )
|
|
set( __ANDROID_NDK__ True )
|
|
set( LLDB_DEFAULT_DISABLE_LIBEDIT True )
|
|
|
|
# linking lldb-server statically for Android avoids the need to ship two
|
|
# binaries (pie for API 21+ and non-pie for API 16-). It's possible to use
|
|
# a non-pie shim on API 16-, but that requires lldb-server to dynamically export
|
|
# its symbols, which significantly increases the binary size. Static linking, on
|
|
# the other hand, has little to no effect on the binary size.
|
|
if( NOT DEFINED LLVM_BUILD_STATIC )
|
|
set( LLVM_BUILD_STATIC True CACHE INTERNAL "" FORCE )
|
|
set( LLVM_ENABLE_PIC FALSE CACHE INTERNAL "" FORCE )
|
|
set( BUILD_SHARED_LIBS FALSE CACHE INTERNAL "" FORCE )
|
|
endif()
|
|
|
|
set( ANDROID_ABI "${ANDROID_ABI}" CACHE INTERNAL "Android Abi" FORCE )
|
|
if( ANDROID_ABI STREQUAL "x86" )
|
|
set( CMAKE_SYSTEM_PROCESSOR "i686" )
|
|
set( ANDROID_TOOLCHAIN_NAME "i686-linux-android" )
|
|
elseif( ANDROID_ABI STREQUAL "x86_64" )
|
|
set( CMAKE_SYSTEM_PROCESSOR "x86_64" )
|
|
set( ANDROID_TOOLCHAIN_NAME "x86_64-linux-android" )
|
|
elseif( ANDROID_ABI STREQUAL "armeabi" )
|
|
set( CMAKE_SYSTEM_PROCESSOR "armv5te" )
|
|
set( ANDROID_TOOLCHAIN_NAME "arm-linux-androideabi" )
|
|
elseif( ANDROID_ABI STREQUAL "aarch64" )
|
|
set( CMAKE_SYSTEM_PROCESSOR "aarch64" )
|
|
set( ANDROID_TOOLCHAIN_NAME "aarch64-linux-android" )
|
|
elseif( ANDROID_ABI STREQUAL "mips" )
|
|
set( CMAKE_SYSTEM_PROCESSOR "mips" )
|
|
set( ANDROID_TOOLCHAIN_NAME "mipsel-linux-android" )
|
|
elseif( ANDROID_ABI STREQUAL "mips64" )
|
|
set( CMAKE_SYSTEM_PROCESSOR "mips64" )
|
|
set( ANDROID_TOOLCHAIN_NAME "mips64el-linux-android" )
|
|
else()
|
|
message( SEND_ERROR "Unknown ANDROID_ABI = \"${ANDROID_ABI}\"." )
|
|
endif()
|
|
|
|
set( ANDROID_TOOLCHAIN_DIR "${ANDROID_TOOLCHAIN_DIR}" CACHE PATH "Android standalone toolchain directory" )
|
|
set( ANDROID_SYSROOT "${ANDROID_TOOLCHAIN_DIR}/sysroot" CACHE PATH "Android Sysroot" )
|
|
|
|
# CMAKE_EXECUTABLE_SUFFIX is undefined in CMAKE_TOOLCHAIN_FILE
|
|
if( WIN32 )
|
|
set( EXECUTABLE_SUFFIX ".exe" )
|
|
endif()
|
|
|
|
set( PYTHON_EXECUTABLE "${ANDROID_TOOLCHAIN_DIR}/bin/python${EXECUTABLE_SUFFIX}" CACHE PATH "Python exec path" )
|
|
|
|
if( NOT CMAKE_C_COMPILER )
|
|
set( CMAKE_C_COMPILER "${ANDROID_TOOLCHAIN_DIR}/bin/${ANDROID_TOOLCHAIN_NAME}-gcc${EXECUTABLE_SUFFIX}" CACHE PATH "C compiler" )
|
|
set( CMAKE_CXX_COMPILER "${ANDROID_TOOLCHAIN_DIR}/bin/${ANDROID_TOOLCHAIN_NAME}-g++${EXECUTABLE_SUFFIX}" CACHE PATH "C++ compiler" )
|
|
set( CMAKE_ASM_COMPILER "${ANDROID_TOOLCHAIN_DIR}/bin/${ANDROID_TOOLCHAIN_NAME}-gcc${EXECUTABLE_SUFFIX}" CACHE PATH "assembler" )
|
|
set( CMAKE_STRIP "${ANDROID_TOOLCHAIN_DIR}/bin/${ANDROID_TOOLCHAIN_NAME}-strip${EXECUTABLE_SUFFIX}" CACHE PATH "strip" )
|
|
set( CMAKE_AR "${ANDROID_TOOLCHAIN_DIR}/bin/${ANDROID_TOOLCHAIN_NAME}-ar${EXECUTABLE_SUFFIX}" CACHE PATH "archive" )
|
|
set( CMAKE_LINKER "${ANDROID_TOOLCHAIN_DIR}/bin/${ANDROID_TOOLCHAIN_NAME}-ld${EXECUTABLE_SUFFIX}" CACHE PATH "linker" )
|
|
set( CMAKE_NM "${ANDROID_TOOLCHAIN_DIR}/bin/${ANDROID_TOOLCHAIN_NAME}-nm${EXECUTABLE_SUFFIX}" CACHE PATH "nm" )
|
|
set( CMAKE_OBJCOPY "${ANDROID_TOOLCHAIN_DIR}/bin/${ANDROID_TOOLCHAIN_NAME}-objcopy${EXECUTABLE_SUFFIX}" CACHE PATH "objcopy" )
|
|
set( CMAKE_OBJDUMP "${ANDROID_TOOLCHAIN_DIR}/bin/${ANDROID_TOOLCHAIN_NAME}-objdump${EXECUTABLE_SUFFIX}" CACHE PATH "objdump" )
|
|
set( CMAKE_RANLIB "${ANDROID_TOOLCHAIN_DIR}/bin/${ANDROID_TOOLCHAIN_NAME}-ranlib${EXECUTABLE_SUFFIX}" CACHE PATH "ranlib" )
|
|
endif()
|
|
|
|
set( ANDROID_CXX_FLAGS "--sysroot=${ANDROID_SYSROOT} -funwind-tables -fsigned-char -no-canonical-prefixes" )
|
|
# TODO: different ARM abi have different flags such as neon, vfpv etc
|
|
if( X86 )
|
|
set( ANDROID_CXX_FLAGS "${ANDROID_CXX_FLAGS} -funswitch-loops -finline-limit=300" )
|
|
elseif( ANDROID_ABI STREQUAL "armeabi" )
|
|
# 64 bit atomic operations used in c++ libraries require armv7-a instructions
|
|
# armv5te and armv6 were tried but do not work.
|
|
set( ANDROID_CXX_FLAGS "${ANDROID_CXX_FLAGS} -march=armv7-a -mthumb" )
|
|
if( LLVM_BUILD_STATIC )
|
|
# Temporary workaround for static linking with the latest API.
|
|
set( ANDROID_CXX_FLAGS "${ANDROID_CXX_FLAGS} -DANDROID_ARM_BUILD_STATIC" )
|
|
endif()
|
|
elseif( ANDROID_ABI STREQUAL "mips" )
|
|
# http://b.android.com/182094
|
|
list( FIND LLDB_SYSTEM_LIBS atomic index )
|
|
if( index EQUAL -1 )
|
|
list( APPEND LLDB_SYSTEM_LIBS atomic )
|
|
set( LLDB_SYSTEM_LIBS ${LLDB_SYSTEM_LIBS} CACHE INTERNAL "" FORCE )
|
|
endif()
|
|
if( LLVM_BUILD_STATIC )
|
|
# Temporary workaround for static linking with the latest API.
|
|
set( ANDROID_CXX_FLAGS "${ANDROID_CXX_FLAGS} -DANDROID_MIPS_BUILD_STATIC" )
|
|
endif()
|
|
endif()
|
|
|
|
# Use gold linker and enable safe ICF in case of x86, x86_64 and arm
|
|
if ( ANDROID_ABI STREQUAL "x86" OR
|
|
ANDROID_ABI STREQUAL "x86_64" OR
|
|
ANDROID_ABI STREQUAL "armeabi")
|
|
set( ANDROID_LINKER_FLAGS "${ANDROID_LINKER_FLAGS} -fuse-ld=gold -Wl,--icf=safe" )
|
|
endif()
|
|
|
|
if( NOT LLVM_BUILD_STATIC )
|
|
# PIE is required for API 21+ so we enable it if we're not statically linking
|
|
# unfortunately, it is not supported before API 16 so we need to do something
|
|
# else there see http://llvm.org/pr23457
|
|
set( ANDROID_CXX_FLAGS "${ANDROID_CXX_FLAGS} -pie -fPIE" )
|
|
endif()
|
|
|
|
# linker flags
|
|
set( ANDROID_CXX_FLAGS "${ANDROID_CXX_FLAGS} -fdata-sections -ffunction-sections" )
|
|
set( ANDROID_LINKER_FLAGS "${ANDROID_LINKER_FLAGS} -Wl,--gc-sections" )
|
|
|
|
# cache flags
|
|
set( CMAKE_CXX_FLAGS "" CACHE STRING "c++ flags" )
|
|
set( CMAKE_C_FLAGS "" CACHE STRING "c flags" )
|
|
set( CMAKE_EXE_LINKER_FLAGS "-Wl,-z,nocopyreloc" CACHE STRING "executable linker flags" )
|
|
set( ANDROID_CXX_FLAGS "${ANDROID_CXX_FLAGS}" CACHE INTERNAL "Android c/c++ flags" )
|
|
set( ANDROID_LINKER_FLAGS "${ANDROID_LINKER_FLAGS}" CACHE INTERNAL "Android c/c++ linker flags" )
|
|
|
|
# final flags
|
|
set( CMAKE_CXX_FLAGS "${ANDROID_CXX_FLAGS} ${CMAKE_CXX_FLAGS}" )
|
|
set( CMAKE_C_FLAGS "${ANDROID_CXX_FLAGS} ${CMAKE_C_FLAGS}" )
|
|
set( CMAKE_EXE_LINKER_FLAGS "${ANDROID_LINKER_FLAGS} ${CMAKE_EXE_LINKER_FLAGS}" )
|
|
|
|
# global includes and link directories
|
|
set( ANDROID_INCLUDE_DIRS "${ANDROID_TOOLCHAIN_DIR}/include/c++/${ANDROID_COMPILER_VERSION}" )
|
|
list( APPEND ANDROID_INCLUDE_DIRS "${ANDROID_TOOLCHAIN_DIR}/include/python2.7" )
|
|
include_directories( SYSTEM "${ANDROID_SYSROOT}/usr/include" ${ANDROID_INCLUDE_DIRS} )
|
|
|
|
# target environment
|
|
set( CMAKE_FIND_ROOT_PATH "${ANDROID_TOOLCHAIN_DIR}/bin" "${ANDROID_TOOLCHAIN_DIR}/${ANDROID_TOOLCHAIN_NAME}" "${ANDROID_SYSROOT}" )
|
|
|
|
# only search for libraries and includes in the ndk toolchain
|
|
set( CMAKE_FIND_ROOT_PATH_MODE_PROGRAM ONLY )
|
|
set( CMAKE_FIND_ROOT_PATH_MODE_LIBRARY ONLY )
|
|
set( CMAKE_FIND_ROOT_PATH_MODE_INCLUDE ONLY )
|
|
|
|
################# BEGIN EVIL HACK ##################
|
|
# In the android-arm NDK unwind.h and link.h contains 2 conflicting
|
|
# typedef for _Unwind_Ptr. Force HAVE_UNWIND_BACKTRACE to 0 to prevent
|
|
# LLVM from finding unwind.h what would break the build.
|
|
if ( ANDROID_ABI STREQUAL "armeabi" )
|
|
set( HAVE_UNWIND_BACKTRACE 0 CACHE INTERNAL "Hack to disable the finding of unwind.h on Android arm" )
|
|
endif()
|
|
################# END EVIL HACK ####################
|
|
|
|
################# BEGIN EVIL HACK ##################
|
|
# lldb-server links against libdl even though it's not being used and
|
|
# libdl.a is currently missing from the toolchain (b.android.com/178517).
|
|
# Therefore, in order to statically link lldb-server, we need a temporary
|
|
# workaround. This creates a dummy libdl.a stub until the actual
|
|
# libdl.a can be implemented in the toolchain.
|
|
if( LLVM_BUILD_STATIC )
|
|
set( libdl "${CMAKE_BINARY_DIR}/libdl_stub" )
|
|
file( MAKE_DIRECTORY ${libdl} )
|
|
file( WRITE "${libdl}/libdl.c" "
|
|
#include <dlfcn.h>
|
|
void * dlopen (const char *filename, int flag) { return 0; }
|
|
const char * dlerror (void) { return 0; }
|
|
void * dlsym (void *handle, const char *symbol) { return 0; }
|
|
int dlclose (void *handle) { return 0; }
|
|
int dladdr (const void *addr, Dl_info *info) { return 0; }")
|
|
set( flags "${CMAKE_C_FLAGS}" )
|
|
separate_arguments( flags )
|
|
execute_process( COMMAND ${CMAKE_C_COMPILER} ${flags} -c ${libdl}/libdl.c -o ${libdl}/libdl.o )
|
|
execute_process( COMMAND ${CMAKE_AR} rcs ${libdl}/libdl.a ${libdl}/libdl.o )
|
|
set( CMAKE_EXE_LINKER_FLAGS "${CMAKE_EXE_LINKER_FLAGS} -L${libdl}" )
|
|
endif()
|
|
################# END EVIL HACK ##################
|