blob: f981f8c8894c4b691bf426cef25a634c5c386021 [file] [log] [blame]
#!/bin/bash
# Copyright 2022 The IREE Authors
#
# Licensed 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
# build_linux_packages.sh
# One stop build of IREE Python packages for Linux. The Linux build is
# complicated because it has to be done via a docker container that has
# an LTS glibc version, all Python packages and other deps.
# This script handles all of those details.
#
# Usage:
# Build everything (all packages, all python versions):
# ./build_tools/python_deploy/build_linux_packages.sh
#
# Build specific Python versions and packages to custom directory:
# override_python_versions="cp39-cp39 cp310-cp310" \
# packages="iree-base-runtime" \
# output_dir="/tmp/wheelhouse" \
# ./build_tools/python_deploy/build_linux_packages.sh
#
# Valid Python versions match a subdirectory under /opt/python in the docker
# image. Typically:
# cp39-cp39 cp310-cp310
#
# Valid packages:
# iree-base-runtime
# iree-base-compiler
#
# Note that this script is meant to be run on CI and it will pollute both the
# output directory and in-tree build/ directories (under runtime/ and
# compiler/) with docker created, root owned builds. Sorry - there is
# no good way around it.
#
# It can be run on a workstation but recommend using a git worktree dedicated
# to packaging to avoid stomping on development artifacts.
set -xeu -o errtrace
# Function to find the directory the ".git" directory is in.
# We do this instead of using git directly because `git` may complain about
# operating in a directory owned by another user.
function find_git_dir_parent() {
curr_dir="${PWD}"
# Loop until we reach the root directory
while [ "${curr_dir}" != "/" ]; do
# Check if there is a ".git" directory in the current directory
if [ -d "${curr_dir}/.git" ]; then
# Return the path to the directory containing the ".git" directory
echo "${curr_dir}"
return
fi
# Move up one directory
curr_dir="$(dirname "${curr_dir}")"
done
# If we reach the root directory and there is no ".git" directory, return an empty string
echo ""
}
this_dir="$(cd $(dirname $0) && pwd)"
script_name="$(basename $0)"
repo_root=$(cd "${this_dir}" && find_git_dir_parent)
manylinux_docker_image="${manylinux_docker_image:-$(uname -m | awk '{print ($1 == "aarch64") ? "quay.io/pypa/manylinux_2_28_aarch64" : "ghcr.io/iree-org/manylinux_x86_64@sha256:2e0246137819cf10ed84240a971f9dd75cc3eb62dc6907dfd2080ee966b3c9f4" }')}"
python_versions="${override_python_versions:-cp39-cp39 cp310-cp310 cp311-cp311 cp312-cp312 cp313-cp313 cp313-cp313t}"
output_dir="${output_dir:-${this_dir}/wheelhouse}"
packages="${packages:-iree-base-runtime iree-base-compiler}"
package_suffix="${package_suffix:-}"
toolchain_suffix="${toolchain_suffix:-release}"
# Return ON if we are on a supported platform for CUDA.
enable_cuda="$(uname -m | awk '{print ($1 == "x86_64" || $1 == "aarch64") ? "ON" : "OFF"}')"
function run_on_host() {
echo "Running on host"
echo "Launching docker image ${manylinux_docker_image}"
# Canonicalize paths.
mkdir -p "${output_dir}"
output_dir="$(cd "${output_dir}" && pwd)"
echo "Outputting to ${output_dir}"
mkdir -p "${output_dir}"
docker run --rm \
-v "${repo_root}:${repo_root}" \
-v "${output_dir}:${output_dir}" \
-e __MANYLINUX_BUILD_WHEELS_IN_DOCKER=1 \
-e "override_python_versions=${python_versions}" \
-e "packages=${packages}" \
-e "package_suffix=${package_suffix}" \
-e "output_dir=${output_dir}" \
-e "toolchain_suffix=${toolchain_suffix}" \
"${manylinux_docker_image}" \
-- ${this_dir}/${script_name}
echo "******************** BUILD COMPLETE ********************"
echo "Generated binaries:"
ls -l "${output_dir}"
}
function run_in_docker() {
echo "Running in docker"
echo "Marking git safe.directory"
git config --global --add safe.directory '*'
echo "Using python versions: ${python_versions}"
local orig_path="${PATH}"
# Configure toolchain.
export CMAKE_TOOLCHAIN_FILE="${repo_root}/build_tools/pkgci/linux_toolchain_${toolchain_suffix}.cmake"
echo "Using CMake toolchain ${CMAKE_TOOLCHAIN_FILE}"
if ! [ -f "$CMAKE_TOOLCHAIN_FILE" ]; then
echo "CMake toolchain not found (wrong toolchain_suffix?)"
exit 1
fi
# Build phase.
for package in ${packages}; do
echo "******************** BUILDING PACKAGE ${package} ********************"
for python_version in ${python_versions}; do
python_dir="/opt/python/${python_version}"
if ! [ -x "${python_dir}/bin/python" ]; then
echo "ERROR: Could not find python: ${python_dir} (skipping)"
continue
fi
export PATH="${python_dir}/bin:${orig_path}"
echo ":::: Python version $(python --version)"
prepare_python
# replace dashes with underscores
package_suffix="${package_suffix//-/_}"
case "${package}" in
iree-base-runtime)
clean_wheels "iree_base_runtime${package_suffix}" "${python_version}"
install_deps "iree_base_runtime${package_suffix}" "${python_version}"
build_iree_runtime
run_audit_wheel "iree_base_runtime${package_suffix}" "${python_version}"
;;
iree-base-compiler)
clean_wheels "iree_base_compiler${package_suffix}" "${python_version}"
install_deps "iree_base_runtime${package_suffix}" "${python_version}"
build_iree_compiler
run_audit_wheel "iree_base_compiler${package_suffix}" "${python_version}"
;;
*)
echo "Unrecognized package '${package}'"
exit 1
;;
esac
done
done
}
function build_wheel() {
python -m pip wheel --disable-pip-version-check -v -w "${output_dir}" "${repo_root}/$@"
}
function build_iree_runtime() {
export IREE_RUNTIME_BUILD_TRACY=ON
# We install the needed build deps below for the tools.
export IREE_RUNTIME_BUILD_TRACY_TOOLS=ON
export IREE_HAL_DRIVER_CUDA="${enable_cuda}"
export IREE_HAL_DRIVER_HIP=ON
build_wheel runtime/
}
function build_iree_compiler() {
export IREE_TARGET_BACKEND_CUDA="${enable_cuda}"
export IREE_TARGET_BACKEND_ROCM=ON
build_wheel compiler/
}
function run_audit_wheel() {
local wheel_basename="$1"
local python_version="$2"
# Force wildcard expansion here
generic_wheel="$(echo "${output_dir}/${wheel_basename}-"*"-${python_version}-linux_$(uname -m).whl")"
ls "${generic_wheel}"
echo ":::: Auditwheel ${generic_wheel}"
auditwheel repair -w "${output_dir}" "${generic_wheel}"
rm -v "${generic_wheel}"
}
function clean_wheels() {
local wheel_basename="$1"
local python_version="$2"
echo ":::: Clean wheels ${wheel_basename} ${python_version}"
rm -f -v "${output_dir}/${wheel_basename}-"*"-${python_version}-"*".whl"
}
function prepare_python() {
# The 0.17 series of patchelf can randomly corrupt executables. Fixes
# have landed but not yet been released. Consider removing this pin
# once 0.19 is released. We just override the system version with
# a pip side load.
pip install patchelf==0.16.1.0
hash -r
echo "patchelf version: $(patchelf --version) (0.17 is bad: https://github.com/NixOS/patchelf/issues/446)"
}
function install_deps() {
local wheel_basename="$1"
local python_version="$2"
echo ":::: Install Deps for ${wheel_basename} ${python_version}"
# Get the output of uname -m
uname_m=$(uname -m)
# Check if the output is aarch64
if [[ "$uname_m" == "aarch64" ]]; then
echo "The architecture is aarch64 and we use manylinux 2_28 so install deps"
yum install -y epel-release
yum update -y
# Required for Tracy
yum install -y capstone-devel tbb-devel libzstd-devel
yum install -y clang lld
elif [[ "$uname_m" == "x86_64" ]]; then
# Check if the output is x86_64
echo "The architecture is x86_64 so assume we are on a managed image with deps"
else
echo "The architecture is unknown. Exiting"
exit 1
fi
}
# Trampoline to the docker container if running on the host.
if [ -z "${__MANYLINUX_BUILD_WHEELS_IN_DOCKER-}" ]; then
run_on_host "$@"
else
run_in_docker "$@"
fi