Chris Bieneman | 18929dd | 2016-05-09 21:45:52 +0000 | [diff] [blame] | 1 | # The CompilerRT build system requires CMake version 2.8.8 or higher in order |
| 2 | # to use its support for building convenience "libraries" as a collection of |
| 3 | # .o files. This is particularly useful in producing larger, more complex |
| 4 | # runtime libraries. |
| 5 | |
| 6 | include(CheckIncludeFile) |
| 7 | check_include_file(unwind.h HAVE_UNWIND_H) |
| 8 | |
| 9 | # Top level target used to build all compiler-rt libraries. |
| 10 | add_custom_target(compiler-rt ALL) |
| 11 | |
Chris Bieneman | b6fb383 | 2016-06-03 23:15:04 +0000 | [diff] [blame] | 12 | # Setting these variables from an LLVM build is sufficient that compiler-rt can |
| 13 | # construct the output paths, so it can behave as if it were in-tree here. |
| 14 | if (LLVM_LIBRARY_OUTPUT_INTDIR AND LLVM_RUNTIME_OUTPUT_INTDIR AND PACKAGE_VERSION) |
| 15 | set(LLVM_TREE_AVAILABLE On) |
| 16 | endif() |
| 17 | |
| 18 | if (LLVM_TREE_AVAILABLE) |
Chris Bieneman | 18929dd | 2016-05-09 21:45:52 +0000 | [diff] [blame] | 19 | # Compute the Clang version from the LLVM version. |
| 20 | # FIXME: We should be able to reuse CLANG_VERSION variable calculated |
| 21 | # in Clang cmake files, instead of copying the rules here. |
| 22 | string(REGEX MATCH "[0-9]+\\.[0-9]+(\\.[0-9]+)?" CLANG_VERSION |
| 23 | ${PACKAGE_VERSION}) |
| 24 | # Setup the paths where compiler-rt runtimes and headers should be stored. |
| 25 | set(COMPILER_RT_OUTPUT_DIR ${LLVM_LIBRARY_OUTPUT_INTDIR}/clang/${CLANG_VERSION}) |
| 26 | set(COMPILER_RT_EXEC_OUTPUT_DIR ${LLVM_RUNTIME_OUTPUT_INTDIR}) |
| 27 | set(COMPILER_RT_INSTALL_PATH lib${LLVM_LIBDIR_SUFFIX}/clang/${CLANG_VERSION}) |
| 28 | option(COMPILER_RT_INCLUDE_TESTS "Generate and build compiler-rt unit tests." |
| 29 | ${LLVM_INCLUDE_TESTS}) |
| 30 | option(COMPILER_RT_ENABLE_WERROR "Fail and stop if warning is triggered" |
| 31 | ${LLVM_ENABLE_WERROR}) |
| 32 | # Use just-built Clang to compile/link tests on all platforms, except for |
| 33 | # Windows where we need to use clang-cl instead. |
| 34 | if(NOT MSVC) |
| 35 | set(COMPILER_RT_TEST_COMPILER ${LLVM_RUNTIME_OUTPUT_INTDIR}/clang) |
| 36 | set(COMPILER_RT_TEST_CXX_COMPILER ${LLVM_RUNTIME_OUTPUT_INTDIR}/clang++) |
| 37 | else() |
| 38 | set(COMPILER_RT_TEST_COMPILER ${LLVM_RUNTIME_OUTPUT_INTDIR}/clang.exe) |
| 39 | set(COMPILER_RT_TEST_CXX_COMPILER ${LLVM_RUNTIME_OUTPUT_INTDIR}/clang++.exe) |
| 40 | endif() |
| 41 | else() |
| 42 | # Take output dir and install path from the user. |
| 43 | set(COMPILER_RT_OUTPUT_DIR ${CMAKE_CURRENT_BINARY_DIR} CACHE PATH |
| 44 | "Path where built compiler-rt libraries should be stored.") |
| 45 | set(COMPILER_RT_EXEC_OUTPUT_DIR ${CMAKE_CURRENT_BINARY_DIR}/bin CACHE PATH |
| 46 | "Path where built compiler-rt executables should be stored.") |
| 47 | set(COMPILER_RT_INSTALL_PATH ${CMAKE_INSTALL_PREFIX} CACHE PATH |
| 48 | "Path where built compiler-rt libraries should be installed.") |
| 49 | option(COMPILER_RT_INCLUDE_TESTS "Generate and build compiler-rt unit tests." OFF) |
| 50 | option(COMPILER_RT_ENABLE_WERROR "Fail and stop if warning is triggered" OFF) |
| 51 | # Use a host compiler to compile/link tests. |
| 52 | set(COMPILER_RT_TEST_COMPILER ${CMAKE_C_COMPILER} CACHE PATH "Compiler to use for testing") |
| 53 | set(COMPILER_RT_TEST_CXX_COMPILER ${CMAKE_CXX_COMPILER} CACHE PATH "C++ Compiler to use for testing") |
| 54 | endif() |
| 55 | |
| 56 | if("${COMPILER_RT_TEST_COMPILER}" MATCHES "clang[+]*$") |
| 57 | set(COMPILER_RT_TEST_COMPILER_ID Clang) |
| 58 | elseif("${COMPILER_RT_TEST_COMPILER}" MATCHES "clang.*.exe$") |
| 59 | set(COMPILER_RT_TEST_COMPILER_ID Clang) |
| 60 | else() |
| 61 | set(COMPILER_RT_TEST_COMPILER_ID GNU) |
| 62 | endif() |
| 63 | |
| 64 | if ("${COMPILER_RT_DEFAULT_TARGET_ABI}" STREQUAL "androideabi") |
| 65 | set(ANDROID 1) |
| 66 | endif() |
| 67 | |
| 68 | string(TOLOWER ${CMAKE_SYSTEM_NAME} COMPILER_RT_OS_DIR) |
| 69 | set(COMPILER_RT_LIBRARY_OUTPUT_DIR |
| 70 | ${COMPILER_RT_OUTPUT_DIR}/lib/${COMPILER_RT_OS_DIR}) |
| 71 | set(COMPILER_RT_LIBRARY_INSTALL_DIR |
| 72 | ${COMPILER_RT_INSTALL_PATH}/lib/${COMPILER_RT_OS_DIR}) |
| 73 | |
| 74 | if(APPLE) |
| 75 | # On Darwin if /usr/include doesn't exist, the user probably has Xcode but not |
| 76 | # the command line tools. If this is the case, we need to find the OS X |
| 77 | # sysroot to pass to clang. |
| 78 | if(NOT EXISTS /usr/include) |
| 79 | execute_process(COMMAND xcodebuild -version -sdk macosx Path |
| 80 | OUTPUT_VARIABLE OSX_SYSROOT |
| 81 | ERROR_QUIET |
| 82 | OUTPUT_STRIP_TRAILING_WHITESPACE) |
| 83 | set(OSX_SYSROOT_FLAG "-isysroot${OSX_SYSROOT}") |
| 84 | endif() |
| 85 | |
| 86 | option(COMPILER_RT_ENABLE_IOS "Enable building for iOS" Off) |
| 87 | option(COMPILER_RT_ENABLE_WATCHOS "Enable building for watchOS - Experimental" Off) |
| 88 | option(COMPILER_RT_ENABLE_TVOS "Enable building for tvOS - Experimental" Off) |
| 89 | endif() |
| 90 | |
| 91 | macro(test_targets) |
| 92 | # Find and run MSVC (not clang-cl) and get its version. This will tell clang-cl |
| 93 | # what version of MSVC to pretend to be so that the STL works. |
| 94 | set(MSVC_VERSION_FLAG "") |
| 95 | if (MSVC) |
| 96 | # Find and run MSVC (not clang-cl) and get its version. This will tell |
| 97 | # clang-cl what version of MSVC to pretend to be so that the STL works. |
| 98 | execute_process(COMMAND "$ENV{VSINSTALLDIR}/VC/bin/cl.exe" |
| 99 | OUTPUT_QUIET |
| 100 | ERROR_VARIABLE MSVC_COMPAT_VERSION |
| 101 | ) |
| 102 | string(REGEX REPLACE "^.*Compiler Version ([0-9.]+) for .*$" "\\1" |
| 103 | MSVC_COMPAT_VERSION "${MSVC_COMPAT_VERSION}") |
| 104 | if (MSVC_COMPAT_VERSION MATCHES "^[0-9].+$") |
| 105 | set(MSVC_VERSION_FLAG "-fms-compatibility-version=${MSVC_COMPAT_VERSION}") |
| 106 | # Add this flag into the host build if this is clang-cl. |
| 107 | if (CMAKE_CXX_COMPILER_ID MATCHES "Clang") |
| 108 | append("${MSVC_VERSION_FLAG}" CMAKE_C_FLAGS CMAKE_CXX_FLAGS) |
Reid Kleckner | d80ed42 | 2016-06-17 17:48:52 +0000 | [diff] [blame^] | 109 | elseif (COMPILER_RT_TEST_COMPILER_ID MATCHES "Clang") |
| 110 | # Add this flag to test compiles to suppress clang's auto-detection |
| 111 | # logic. |
| 112 | append("${MSVC_VERSION_FLAG}" COMPILER_RT_TEST_COMPILER_CFLAGS) |
Chris Bieneman | 18929dd | 2016-05-09 21:45:52 +0000 | [diff] [blame] | 113 | endif() |
| 114 | endif() |
| 115 | endif() |
| 116 | |
| 117 | # Generate the COMPILER_RT_SUPPORTED_ARCH list. |
| 118 | if(ANDROID) |
| 119 | # Examine compiler output to determine target architecture. |
| 120 | detect_target_arch() |
| 121 | set(COMPILER_RT_OS_SUFFIX "-android") |
| 122 | elseif(NOT APPLE) # Supported archs for Apple platforms are generated later |
| 123 | if("${COMPILER_RT_DEFAULT_TARGET_ARCH}" MATCHES "i[2-6]86|x86|amd64") |
| 124 | if(NOT MSVC) |
| 125 | test_target_arch(x86_64 "" "-m64") |
| 126 | # FIXME: We build runtimes for both i686 and i386, as "clang -m32" may |
| 127 | # target different variant than "$CMAKE_C_COMPILER -m32". This part should |
| 128 | # be gone after we resolve PR14109. |
| 129 | test_target_arch(i686 __i686__ "-m32") |
| 130 | test_target_arch(i386 __i386__ "-m32") |
| 131 | else() |
| 132 | if (CMAKE_SIZEOF_VOID_P EQUAL 4) |
Reid Kleckner | d80ed42 | 2016-06-17 17:48:52 +0000 | [diff] [blame^] | 133 | test_target_arch(i386 "" "") |
Chris Bieneman | 18929dd | 2016-05-09 21:45:52 +0000 | [diff] [blame] | 134 | else() |
Reid Kleckner | d80ed42 | 2016-06-17 17:48:52 +0000 | [diff] [blame^] | 135 | test_target_arch(x86_64 "" "") |
Chris Bieneman | 18929dd | 2016-05-09 21:45:52 +0000 | [diff] [blame] | 136 | endif() |
| 137 | endif() |
| 138 | elseif("${COMPILER_RT_DEFAULT_TARGET_ARCH}" MATCHES "powerpc") |
| 139 | TEST_BIG_ENDIAN(HOST_IS_BIG_ENDIAN) |
| 140 | if(HOST_IS_BIG_ENDIAN) |
| 141 | test_target_arch(powerpc64 "" "-m64") |
| 142 | else() |
| 143 | test_target_arch(powerpc64le "" "-m64") |
| 144 | endif() |
| 145 | elseif("${COMPILER_RT_DEFAULT_TARGET_ARCH}" MATCHES "s390x") |
| 146 | test_target_arch(s390x "" "") |
| 147 | elseif("${COMPILER_RT_DEFAULT_TARGET_ARCH}" MATCHES "mipsel|mips64el") |
| 148 | # Gcc doesn't accept -m32/-m64 so we do the next best thing and use |
| 149 | # -mips32r2/-mips64r2. We don't use -mips1/-mips3 because we want to match |
| 150 | # clang's default CPU's. In the 64-bit case, we must also specify the ABI |
| 151 | # since the default ABI differs between gcc and clang. |
| 152 | # FIXME: Ideally, we would build the N32 library too. |
| 153 | test_target_arch(mipsel "" "-mips32r2" "--target=mipsel-linux-gnu") |
| 154 | test_target_arch(mips64el "" "-mips64r2" "--target=mips64el-linux-gnu" "-mabi=n64") |
| 155 | elseif("${COMPILER_RT_DEFAULT_TARGET_ARCH}" MATCHES "mips") |
| 156 | test_target_arch(mips "" "-mips32r2" "--target=mips-linux-gnu") |
| 157 | test_target_arch(mips64 "" "-mips64r2" "--target=mips64-linux-gnu" "-mabi=n64") |
| 158 | elseif("${COMPILER_RT_DEFAULT_TARGET_ARCH}" MATCHES "arm") |
| 159 | test_target_arch(arm "" "-march=armv7-a" "-mfloat-abi=soft") |
| 160 | test_target_arch(armhf "" "-march=armv7-a" "-mfloat-abi=hard") |
| 161 | elseif("${COMPILER_RT_DEFAULT_TARGET_ARCH}" MATCHES "aarch32") |
| 162 | test_target_arch(aarch32 "" "-march=armv8-a") |
| 163 | elseif("${COMPILER_RT_DEFAULT_TARGET_ARCH}" MATCHES "aarch64") |
| 164 | test_target_arch(aarch64 "" "-march=armv8-a") |
| 165 | elseif("${COMPILER_RT_DEFAULT_TARGET_ARCH}" MATCHES "wasm32") |
| 166 | test_target_arch(wasm32 "" "--target=wasm32-unknown-unknown") |
| 167 | elseif("${COMPILER_RT_DEFAULT_TARGET_ARCH}" MATCHES "wasm64") |
| 168 | test_target_arch(wasm64 "" "--target=wasm64-unknown-unknown") |
| 169 | endif() |
| 170 | set(COMPILER_RT_OS_SUFFIX "") |
| 171 | endif() |
| 172 | endmacro() |