#!/usr/bin/env bash #===----------------------------------------------------------------------===## # # Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. # See https://llvm.org/LICENSE.txt for license information. # SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception # #===----------------------------------------------------------------------===## set -e PROGNAME="$(basename "${0}")" function error() { printf "error: %s\n" "$*"; exit 1; } function usage() { cat < Path to the root of the LLVM monorepo. Only the libcxx and libcxxabi directories are required. --build-dir Path to the directory to use for building. This will contain intermediate build products. --install-dir Path to the directory to install the library to. --symbols-dir Path to the directory to install the .dSYM bundle to. --sdk SDK used for building the library. This represents the target platform that the library will run on. You can get a list of SDKs with \`xcodebuild -showsdks\`. --architectures "..." A whitespace separated list of architectures to build for. The library will be built for each architecture independently, and a universal binary containing all architectures will be created from that. --version X[.Y[.Z]] The version of the library to encode in the dylib. --cache The CMake cache to use to control how the library gets built. EOF } while [[ $# -gt 0 ]]; do case ${1} in -h|--help) usage exit 0 ;; --llvm-root) llvm_root="${2}" shift; shift ;; --build-dir) build_dir="${2}" shift; shift ;; --symbols-dir) symbols_dir="${2}" shift; shift ;; --install-dir) install_dir="${2}" shift; shift ;; --sdk) sdk="${2}" shift; shift ;; --architectures) architectures="${2}" shift; shift ;; --version) version="${2}" shift; shift ;; --cache) cache="${2}" shift; shift ;; *) error "Unknown argument '${1}'" ;; esac done for arg in llvm_root build_dir symbols_dir install_dir sdk architectures version cache; do if [ -z ${!arg+x} ]; then error "Missing required argument '--${arg//_/-}'" elif [ "${!arg}" == "" ]; then error "Argument to --${arg//_/-} must not be empty" fi done # Allow using relative paths function realpath() { if [[ $1 = /* ]]; then echo "$1"; else echo "$(pwd)/${1#./}"; fi } for arg in llvm_root build_dir symbols_dir install_dir cache; do path="$(realpath "${!arg}")" eval "${arg}=\"${path}\"" done function step() { separator="$(printf "%0.s-" $(seq 1 ${#1}))" echo echo "${separator}" echo "${1}" echo "${separator}" } install_name_dir="/usr/lib" dylib_name="libc++.1.dylib" make_symlink="yes" headers_prefix="${install_dir}" for arch in ${architectures}; do step "Building libc++.dylib and libc++abi.dylib for architecture ${arch}" mkdir -p "${build_dir}/${arch}" (cd "${build_dir}/${arch}" && xcrun --sdk "${sdk}" cmake "${llvm_root}/libcxx/utils/ci/runtimes" \ -GNinja \ -DCMAKE_MAKE_PROGRAM="$(xcrun --sdk "${sdk}" --find ninja)" \ -DLLVM_ENABLE_PROJECTS="libcxx;libcxxabi" \ -C "${cache}" \ -DCMAKE_INSTALL_PREFIX="${build_dir}/${arch}-install" \ -DCMAKE_INSTALL_NAME_DIR="${install_name_dir}" \ -DCMAKE_OSX_ARCHITECTURES="${arch}" \ -DLIBCXXABI_LIBRARY_VERSION="${version}" \ -DLIBCXX_INCLUDE_BENCHMARKS=OFF \ -DLIBCXX_INCLUDE_TESTS=OFF ) xcrun --sdk "${sdk}" cmake --build "${build_dir}/${arch}" --target install-cxx install-cxxabi -- -v done function universal_dylib() { dylib=${1} inputs=$(for arch in ${architectures}; do echo "${build_dir}/${arch}-install/lib/${dylib}"; done) step "Creating a universal dylib ${dylib} from the dylibs for all architectures" xcrun --sdk "${sdk}" lipo -create ${inputs} -output "${build_dir}/${dylib}" step "Installing the (stripped) universal dylib to ${install_dir}/usr/lib" mkdir -p "${install_dir}/usr/lib" cp "${build_dir}/${dylib}" "${install_dir}/usr/lib/${dylib}" xcrun --sdk "${sdk}" strip -S "${install_dir}/usr/lib/${dylib}" step "Installing the unstripped dylib and the dSYM bundle to ${symbols_dir}" xcrun --sdk "${sdk}" dsymutil "${build_dir}/${dylib}" -o "${symbols_dir}/${dylib}.dSYM" cp "${build_dir}/${dylib}" "${symbols_dir}/${dylib}" } universal_dylib ${dylib_name} universal_dylib libc++abi.dylib if [[ "${make_symlink}" == "yes" ]]; then (cd "${install_dir}/usr/lib" && ln -s "${dylib_name}" libc++.dylib) fi # Install the headers by copying the headers from one of the built architectures # into the install directory. Headers from all architectures should be the same. step "Installing the libc++ and libc++abi headers to ${headers_prefix}/usr/include" any_arch=$(echo ${architectures} | cut -d ' ' -f 1) mkdir -p "${headers_prefix}/usr/include" ditto "${build_dir}/${any_arch}-install/include" "${headers_prefix}/usr/include" ditto "${llvm_root}/libcxxabi/include" "${headers_prefix}/usr/include" # TODO: libcxxabi should install its headers in CMake if [[ $EUID -eq 0 ]]; then # Only chown if we're running as root chown -R root:wheel "${headers_prefix}/usr/include" fi step "Installing the libc++ and libc++abi licenses" mkdir -p "${headers_prefix}/usr/local/OpenSourceLicenses" cp "${llvm_root}/libcxx/LICENSE.TXT" "${headers_prefix}/usr/local/OpenSourceLicenses/libcxx.txt" cp "${llvm_root}/libcxxabi/LICENSE.TXT" "${headers_prefix}/usr/local/OpenSourceLicenses/libcxxabi.txt" # Also install a static archive for libc++abi libcxxabi_archives=$(for arch in ${architectures}; do echo "${build_dir}/${arch}-install/lib/libc++abi.a"; done) step "Creating a universal libc++abi static archive from the static archives for each architecture" mkdir -p "${install_dir}/usr/local/lib/libcxx" xcrun --sdk "${sdk}" libtool -static ${libcxxabi_archives} -o "${install_dir}/usr/local/lib/libcxx/libc++abi-static.a"