termux-packages/packages/emscripten/build.sh

404 lines
14 KiB
Bash

TERMUX_PKG_HOMEPAGE=https://emscripten.org
TERMUX_PKG_DESCRIPTION="Emscripten: An LLVM-to-WebAssembly Compiler"
TERMUX_PKG_LICENSE="MIT"
TERMUX_PKG_MAINTAINER="@termux"
TERMUX_PKG_VERSION="3.1.57"
TERMUX_PKG_SRCURL=git+https://github.com/emscripten-core/emscripten
TERMUX_PKG_GIT_BRANCH=${TERMUX_PKG_VERSION}
TERMUX_PKG_DEPENDS="nodejs-lts | nodejs, python"
TERMUX_PKG_ANTI_BUILD_DEPENDS="nodejs, nodejs-lts, python"
TERMUX_PKG_HOSTBUILD=true
TERMUX_PKG_NO_STATICSPLIT=true
TERMUX_PKG_PLATFORM_INDEPENDENT=true
TERMUX_PKG_AUTO_UPDATE=true
# remove files according to emsdk/upstream directory
# git clone https://github.com/emscripten-core/emsdk --depth=1
# cd emsdk
# ./emsdk install latest
# ls ./upstream/bin/*
TERMUX_PKG_RM_AFTER_INSTALL="
opt/emscripten-llvm/bin/amdgpu-arch
opt/emscripten-llvm/bin/clang-check
opt/emscripten-llvm/bin/clang-cl
opt/emscripten-llvm/bin/clang-cpp
opt/emscripten-llvm/bin/clang-extdef-mapping
opt/emscripten-llvm/bin/clang-format
opt/emscripten-llvm/bin/clang-func-mapping
opt/emscripten-llvm/bin/clang-import-test
opt/emscripten-llvm/bin/clang-installapi
opt/emscripten-llvm/bin/clang-linker-wrapper
opt/emscripten-llvm/bin/clang-nvlink-wrapper
opt/emscripten-llvm/bin/clang-offload-bundler
opt/emscripten-llvm/bin/clang-offload-packager
opt/emscripten-llvm/bin/clang-offload-wrapper
opt/emscripten-llvm/bin/clang-pseudo
opt/emscripten-llvm/bin/clang-refactor
opt/emscripten-llvm/bin/clang-rename
opt/emscripten-llvm/bin/clang-repl
opt/emscripten-llvm/bin/clang-scan-deps
opt/emscripten-llvm/bin/diagtool
opt/emscripten-llvm/bin/git-clang-format
opt/emscripten-llvm/bin/hmaptool
opt/emscripten-llvm/bin/ld.lld
opt/emscripten-llvm/bin/ld64.lld
opt/emscripten-llvm/bin/ld64.lld.darwin*
opt/emscripten-llvm/bin/lld-link
opt/emscripten-llvm/bin/llvm-cov
opt/emscripten-llvm/bin/llvm-dlltool
opt/emscripten-llvm/bin/llvm-lib
opt/emscripten-llvm/bin/llvm-link
opt/emscripten-llvm/bin/llvm-mca
opt/emscripten-llvm/bin/llvm-ml
opt/emscripten-llvm/bin/llvm-pdbutil
opt/emscripten-llvm/bin/llvm-profdata
opt/emscripten-llvm/bin/llvm-rc
opt/emscripten-llvm/bin/nvptx-arch
opt/emscripten-llvm/lib/libclang.so*
opt/emscripten-llvm/share
opt/emscripten/LICENSE
"
# https://github.com/emscripten-core/emscripten/issues/11362
# can switch to stable LLVM to save space once above is fixed
_LLVM_COMMIT=ccdebbae4d77d3efc236af92c22941de5d437e01
_LLVM_TGZ_SHA256=32a4416ae815be3622de7501011d3f67a5ffe86e15072aa22e613aed5340ce98
# https://github.com/emscripten-core/emscripten/issues/12252
# upstream says better bundle the right binaryen revision for now
_BINARYEN_COMMIT=f0dd9941de2df62e0a29f2faeadf007e37a425a9
_BINARYEN_TGZ_SHA256=32b12fcb4bc96e3d43e329b90f617993c90150c5414f950fe4f65b1b9cd0251e
# https://github.com/emscripten-core/emsdk/blob/main/emsdk.py
# https://chromium.googlesource.com/emscripten-releases/+/refs/heads/main/src/build.py
# https://github.com/llvm/llvm-project
_LLVM_BUILD_ARGS="
-DCMAKE_BUILD_TYPE=MinSizeRel
-DCMAKE_BUILD_WITH_INSTALL_RPATH=ON
-DCMAKE_CROSSCOMPILING=ON
-DCMAKE_INSTALL_PREFIX=${TERMUX_PREFIX}/opt/emscripten-llvm
-DDEFAULT_SYSROOT=$(dirname ${TERMUX_PREFIX})
-DGENERATOR_IS_MULTI_CONFIG=ON
-DLLVM_ENABLE_ASSERTIONS=ON
-DLLVM_ENABLE_BINDINGS=OFF
-DLLVM_ENABLE_LIBEDIT=OFF
-DLLVM_ENABLE_LIBPFM=OFF
-DLLVM_ENABLE_LIBXML2=OFF
-DLLVM_ENABLE_LTO=Thin
-DLLVM_ENABLE_PROJECTS=clang;compiler-rt;lld
-DLLVM_ENABLE_TERMINFO=OFF
-DLLVM_INCLUDE_BENCHMARKS=OFF
-DLLVM_INCLUDE_EXAMPLES=OFF
-DLLVM_INCLUDE_TESTS=OFF
-DLLVM_INCLUDE_UTILS=OFF
-DLLVM_INSTALL_TOOLCHAIN_ONLY=ON
-DLLVM_LINK_LLVM_DYLIB=ON
-DLLVM_NATIVE_TOOL_DIR=${TERMUX_PKG_HOSTBUILD_DIR}/bin
-DCLANG_DEFAULT_LINKER=lld
-DCLANG_ENABLE_ARCMT=OFF
-DCLANG_ENABLE_STATIC_ANALYZER=OFF
-DCOMPILER_RT_BUILD_CRT=OFF
-DCOMPILER_RT_BUILD_LIBFUZZER=OFF
-DCOMPILER_RT_BUILD_MEMPROF=OFF
-DCOMPILER_RT_BUILD_PROFILE=OFF
-DCOMPILER_RT_BUILD_SANITIZERS=OFF
-DCOMPILER_RT_BUILD_XRAY=OFF
-DCOMPILER_RT_INCLUDE_TESTS=OFF
"
# https://github.com/WebAssembly/binaryen/blob/main/CMakeLists.txt
_BINARYEN_BUILD_ARGS="
-DCMAKE_INSTALL_PREFIX=${TERMUX_PREFIX}/opt/emscripten-binaryen
-DBUILD_TESTS=OFF
-DBYN_ENABLE_LTO=ON
"
# based on scripts/updates/internal/termux_github_auto_update.sh
termux_pkg_auto_update() {
local latest_tag
latest_tag=$(termux_github_api_get_tag "${TERMUX_PKG_SRCURL}" "${TERMUX_PKG_UPDATE_TAG_TYPE}")
if [[ -z "${latest_tag}" ]]; then
termux_error_exit "ERROR: Unable to get tag from ${TERMUX_PKG_SRCURL}"
fi
if [[ "${latest_tag}" == "${TERMUX_PKG_VERSION}" ]]; then
echo "INFO: No update needed. Already at version '${TERMUX_PKG_VERSION}'."
return
fi
# https://github.com/emscripten-core/emscripten/blob/main/docs/packaging.md
# https://github.com/archlinux/svntogit-community/tree/packages/emscripten/trunk
# below generates commit hash for the deps according to emscripten releases
local tmpdir=$(mktemp -d)
local releases_tags release_tag deps_revision deps_json llvm_commit binaryen_commit llvm_tgz_sha256 binaryen_tgz_sha256
releases_tags=$(curl -s https://raw.githubusercontent.com/emscripten-core/emsdk/main/emscripten-releases-tags.json)
release_tag=$(echo "${releases_tags}" | python3 -c "import json,sys;print(json.load(sys.stdin)[\"releases\"][\"${latest_tag}\"])")
deps_revision=$(curl -s "https://chromium.googlesource.com/emscripten-releases/+/${release_tag}/DEPS?format=text" | base64 -d | grep "_revision':" | sed -e "s|'|\"|g")
deps_json=$(echo -e "{\n${deps_revision}EOL" | sed -e "s|,EOL|\n}|")
llvm_commit=$(echo "${deps_json}" | python3 -c "import json,sys;print(json.load(sys.stdin)[\"llvm_project_revision\"])")
binaryen_commit=$(echo "${deps_json}" | python3 -c "import json,sys;print(json.load(sys.stdin)[\"binaryen_revision\"])")
curl -LC - "https://github.com/llvm/llvm-project/archive/${llvm_commit}.tar.gz" -o "${tmpdir}/${llvm_commit}.tar.gz"
curl -LC - "https://github.com/WebAssembly/binaryen/archive/${binaryen_commit}.tar.gz" -o "${tmpdir}/${binaryen_commit}.tar.gz"
llvm_tgz_sha256=$(sha256sum "${tmpdir}/${llvm_commit}.tar.gz" | sed -e "s| .*$||")
binaryen_tgz_sha256=$(sha256sum "${tmpdir}/${binaryen_commit}.tar.gz" | sed -e "s| .*$||")
cat <<- EOL
INFO: Generated *.tar.gz checksum for:
_LLVM_COMMIT ${llvm_commit} = ${llvm_tgz_sha256}
_BINARYEN_COMMIT ${binaryen_commit} = ${binaryen_tgz_sha256}
EOL
sed \
-e "s|^_LLVM_COMMIT=.*|_LLVM_COMMIT=${llvm_commit}|" \
-e "s|^_LLVM_TGZ_SHA256=.*|_LLVM_TGZ_SHA256=${llvm_tgz_sha256}|" \
-e "s|^_BINARYEN_COMMIT=.*|_BINARYEN_COMMIT=${binaryen_commit}|" \
-e "s|^_BINARYEN_TGZ_SHA256=.*|_BINARYEN_TGZ_SHA256=${binaryen_tgz_sha256}|" \
-i "${TERMUX_PKG_BUILDER_DIR}/build.sh"
rm -fr "${tmpdir}"
termux_pkg_upgrade_version "$latest_tag"
}
termux_step_post_get_source() {
termux_download \
"https://github.com/llvm/llvm-project/archive/${_LLVM_COMMIT}.tar.gz" \
"${TERMUX_PKG_CACHEDIR}/llvm.tar.gz" \
"${_LLVM_TGZ_SHA256}"
termux_download \
"https://github.com/WebAssembly/binaryen/archive/${_BINARYEN_COMMIT}.tar.gz" \
"${TERMUX_PKG_CACHEDIR}/binaryen.tar.gz" \
"${_BINARYEN_TGZ_SHA256}"
tar -xf "${TERMUX_PKG_CACHEDIR}/llvm.tar.gz" -C "${TERMUX_PKG_CACHEDIR}"
tar -xf "${TERMUX_PKG_CACHEDIR}/binaryen.tar.gz" -C "${TERMUX_PKG_CACHEDIR}"
local llvm_patches=$(find "${TERMUX_PKG_BUILDER_DIR}" -mindepth 1 -maxdepth 1 -type f -name 'llvm-project-*.diff')
if [[ -n "${llvm_patches}" ]]; then
pushd "${TERMUX_PKG_CACHEDIR}/llvm-project-${_LLVM_COMMIT}"
for patch in ${llvm_patches}; do
patch -p1 -i "${patch}" || :
done
local llvm_patches_rej=$(find . -type f -name '*.rej')
if [[ -n "${llvm_patches_rej}" ]]; then
echo "INFO: Patch failed! Printing *.rej files ..."
for rej in ${llvm_patches_rej}; do
echo -e "\n\n${rej}"
cat "${rej}"
done
termux_error_exit "Patch failed! Please check patch errors above!"
fi
popd
fi
local binaryen_patches=$(find "${TERMUX_PKG_BUILDER_DIR}" -mindepth 1 -maxdepth 1 -type f -name 'binaryen-*.diff')
if [[ -n "${binaryen_patches}" ]]; then
pushd "${TERMUX_PKG_CACHEDIR}/binaryen-${_BINARYEN_COMMIT}"
for patch in ${binaryen_patches}; do
patch -p1 -i "${patch}" || :
done
local binaryen_patches_rej=$(find . -type f -name '*.rej')
if [[ -n "${binaryen_patches_rej}" ]]; then
echo "INFO: Patch failed! Printing *.rej files ..."
for rej in ${binaryen_patches_rej}; do
echo -e "\n\n${rej}"
cat "${rej}"
done
termux_error_exit "Patch failed! Please check patch errors above!"
fi
popd
fi
}
termux_step_host_build() {
termux_setup_cmake
termux_setup_ninja
cmake \
-G Ninja \
-S "${TERMUX_PKG_CACHEDIR}/llvm-project-${_LLVM_COMMIT}/llvm" \
-DCMAKE_BUILD_TYPE=Release \
-DLLVM_ENABLE_PROJECTS=clang \
-DLLVM_INCLUDE_BENCHMARKS=OFF \
-DLLVM_INCLUDE_EXAMPLES=OFF \
-DLLVM_INCLUDE_TESTS=OFF \
-DLLVM_INCLUDE_UTILS=OFF
ninja \
-C "${TERMUX_PKG_HOSTBUILD_DIR}" \
-j "${TERMUX_MAKE_PROCESSES}" \
llvm-tblgen clang-tblgen
}
termux_step_pre_configure() {
# this is a workaround for build-all.sh issue
TERMUX_PKG_DEPENDS+=", emscripten-binaryen, emscripten-llvm"
# https://github.com/termux/termux-packages/issues/16358
# TODO libclang-cpp.so* is not affected
if [[ "${TERMUX_ON_DEVICE_BUILD}" == "true" ]]; then
echo "WARN: ld.lld wrapper is not working for on-device builds. Skipping."
return
fi
local _WRAPPER_BIN=${TERMUX_PKG_BUILDDIR}/_wrapper/bin
mkdir -p "${_WRAPPER_BIN}"
ln -fs "${TERMUX_STANDALONE_TOOLCHAIN}/bin/lld" "${_WRAPPER_BIN}/ld.lld"
cat <<- EOF > "${_WRAPPER_BIN}/ld.lld.sh"
#!/bin/bash
tmpfile=\$(mktemp)
python ${TERMUX_PKG_BUILDER_DIR}/fix-rpath.py -rpath=${TERMUX_PREFIX}/lib \$@ > \${tmpfile}
args=\$(cat \${tmpfile})
rm -f \${tmpfile}
${_WRAPPER_BIN}/ld.lld \${args}
EOF
chmod +x "${_WRAPPER_BIN}/ld.lld.sh"
rm -f "${TERMUX_STANDALONE_TOOLCHAIN}/bin/ld.lld"
ln -fs "${_WRAPPER_BIN}/ld.lld.sh" "${TERMUX_STANDALONE_TOOLCHAIN}/bin/ld.lld"
}
termux_step_make() {
termux_setup_cmake
termux_setup_ninja
# from packages/libllvm/build.sh
local _LLVM_TARGET_TRIPLE=${TERMUX_HOST_PLATFORM/-/-unknown-}${TERMUX_PKG_API_LEVEL}
local _LLVM_TARGET_ARCH
case "${TERMUX_ARCH}" in
aarch64) _LLVM_TARGET_ARCH=AArch64 ;;
arm) _LLVM_TARGET_ARCH=ARM ;;
i686|x86_64) _LLVM_TARGET_ARCH=X86 ;;
*) termux_error_exit "Invalid arch: ${TERMUX_ARCH}" ;;
esac
_LLVM_BUILD_ARGS+="
-DLLVM_HOST_TRIPLE=${_LLVM_TARGET_TRIPLE}
-DLLVM_TARGET_ARCH=${_LLVM_TARGET_ARCH}
-DLLVM_TARGETS_TO_BUILD=WebAssembly;${_LLVM_TARGET_ARCH}
"
cmake \
-G Ninja \
-S "${TERMUX_PKG_CACHEDIR}/llvm-project-${_LLVM_COMMIT}/llvm" \
-B "${TERMUX_PKG_BUILDDIR}/build-llvm" \
${_LLVM_BUILD_ARGS}
ninja \
-C "${TERMUX_PKG_BUILDDIR}/build-llvm" \
-j "${TERMUX_MAKE_PROCESSES}" \
install
cmake \
-G Ninja \
-S "${TERMUX_PKG_CACHEDIR}/binaryen-${_BINARYEN_COMMIT}" \
-B "${TERMUX_PKG_BUILDDIR}/build-binaryen" \
${_BINARYEN_BUILD_ARGS}
ninja \
-C "${TERMUX_PKG_BUILDDIR}/build-binaryen" \
-j "${TERMUX_MAKE_PROCESSES}" \
install
}
termux_step_make_install() {
pushd "${TERMUX_PKG_SRCDIR}"
# https://github.com/emscripten-core/emscripten/pull/15840
sed -e "s|-git||" -i "${TERMUX_PKG_SRCDIR}/emscripten-version.txt"
# skip using Makefile which does host npm install
rm -fr "${TERMUX_PREFIX}/opt/emscripten"
./tools/install.py "${TERMUX_PREFIX}/opt/emscripten"
# subpackage optional third party test suite files
cp -fr "${TERMUX_PKG_SRCDIR}/test/third_party" "${TERMUX_PREFIX}/opt/emscripten/test/third_party"
# first run generates .emscripten and exits immediately
rm -f "${TERMUX_PKG_SRCDIR}/.emscripten"
./emcc --generate-config
sed \
-e "s|^EMSCRIPTEN_ROOT.*|EMSCRIPTEN_ROOT = '${TERMUX_PREFIX}/opt/emscripten' # directory|" \
-e "s|^LLVM_ROOT.*|LLVM_ROOT = '${TERMUX_PREFIX}/opt/emscripten-llvm/bin' # directory|" \
-e "s|^BINARYEN_ROOT.*|BINARYEN_ROOT = '${TERMUX_PREFIX}/opt/emscripten-binaryen' # directory|" \
-e "s|^NODE_JS.*|NODE_JS = '${TERMUX_PREFIX}/bin/node' # executable|" \
-i .emscripten
grep "${TERMUX_PREFIX}" "${TERMUX_PKG_SRCDIR}/.emscripten"
install -Dm644 "${TERMUX_PKG_SRCDIR}/.emscripten" "${TERMUX_PREFIX}/opt/emscripten/.emscripten"
# add emscripten directory to PATH var
cat <<- EOF > "${TERMUX_PKG_TMPDIR}/emscripten.sh"
#!${TERMUX_PREFIX}/bin/sh
export PATH=\${PATH}:${TERMUX_PREFIX}/opt/emscripten
EOF
install -Dm644 "${TERMUX_PKG_TMPDIR}/emscripten.sh" "${TERMUX_PREFIX}/etc/profile.d/emscripten.sh"
# add useful tools not installed by LLVM_INSTALL_TOOLCHAIN_ONLY=ON
for tool in llvm-{addr2line,dwarfdump,dwp,link,nm,objdump,ranlib,readobj,size,strings}; do
install -Dm755 "${TERMUX_PKG_BUILDDIR}/build-llvm/bin/${tool}" "${TERMUX_PREFIX}/opt/emscripten-llvm/bin/${tool}"
done
# wasm32 triplets
rm -fr "${TERMUX_PREFIX}"/opt/emscripten-llvm/bin/wasm32-{clang,clang++,wasi-clang,wasi-clang++}
rm -fr "${TERMUX_PREFIX}/opt/emscripten-llvm/bin/wasm-ld"
ln -fs "clang" "${TERMUX_PREFIX}/opt/emscripten-llvm/bin/wasm32-clang"
ln -fs "clang++" "${TERMUX_PREFIX}/opt/emscripten-llvm/bin/wasm32-clang++"
ln -fs "clang" "${TERMUX_PREFIX}/opt/emscripten-llvm/bin/wasm32-wasi-clang"
ln -fs "clang++" "${TERMUX_PREFIX}/opt/emscripten-llvm/bin/wasm32-wasi-clang++"
ln -fs "lld" "${TERMUX_PREFIX}/opt/emscripten-llvm/bin/wasm-ld"
# termux_step_massage strip does not cover opt dir
find "${TERMUX_PREFIX}/opt" \( \
-path "*/emscripten-llvm/bin/*" -o \
-path "*/emscripten-llvm/lib/*" -o \
-path "*/emscripten-binaryen/bin/*" -o \
-path "*/emscripten-binaryen/lib/*" \
\) -type f -print0 | \
xargs -0 -r file | grep -E "ELF .+ (executable|shared object)" | \
cut -d":" -f1 | xargs -r "${STRIP}" --strip-unneeded --preserve-dates
popd
}
termux_step_create_debscripts() {
# emscripten's package-lock.json is generated with nodejs v12.13.0
# which comes with npm v6 which used lockfile version 1
# which isn't compatible with lockfile version 2 used in npm v7 and v8
cat <<- EOF > postinst
#!${TERMUX_PREFIX}/bin/sh
DIR="${TERMUX_PREFIX}/opt/emscripten"
cd "\${DIR}"
if [ -n "\$(command -v npm)" ]; then
if [ -n "\$(npm --version | grep "^6.")" ]; then
CMD="ci --production --no-optional"
else
CMD="install --omit=dev --omit=optional"
rm package-lock.json
fi
echo "Running 'npm \${CMD}' in \${DIR} ..."
npm \${CMD}
else
echo '
WARNING: npm is not installed! Emscripten may not work properly without installing node modules!
' >&2
fi
echo '
===== Post-install notice =====
Please start a new session to use Emscripten.
You may want to clear the cache by running
the command below to fix issues.
emcc --clear-cache
===== Post-install notice =====
'
EOF
cat <<- EOF > postrm
#!${TERMUX_PREFIX}/bin/sh
case "\$1" in
purge|remove)
rm -fr "${TERMUX_PREFIX}/opt/emscripten"
esac
EOF
}