From a73c749c2eaa523ea2366f859167a11ec9760a9e Mon Sep 17 00:00:00 2001 From: Hyunsu Cho Date: Sun, 30 Jan 2022 05:24:15 -0800 Subject: [PATCH 1/7] Build Python wheels for OSX (x86_64 and arm64) --- .github/workflows/python_tests.yml | 21 ---------- .github/workflows/python_wheels.yml | 57 +++++++++++++++++++++++++++ python-package/setup.py | 3 ++ tests/ci_build/build_python_wheels.sh | 34 ++++++++++++++++ 4 files changed, 94 insertions(+), 21 deletions(-) create mode 100644 .github/workflows/python_wheels.yml create mode 100644 tests/ci_build/build_python_wheels.sh diff --git a/.github/workflows/python_tests.yml b/.github/workflows/python_tests.yml index fb5b85f2bd31..a989c4643ace 100644 --- a/.github/workflows/python_tests.yml +++ b/.github/workflows/python_tests.yml @@ -141,24 +141,3 @@ jobs: shell: bash -l {0} run: | pytest -s -v ./tests/python - - - name: Rename Python wheel - shell: bash -l {0} - run: | - TAG=macosx_10_15_x86_64.macosx_11_0_x86_64.macosx_12_0_x86_64 - python tests/ci_build/rename_whl.py python-package/dist/*.whl ${{ github.sha }} ${TAG} - - - name: Extract branch name - shell: bash - run: echo "##[set-output name=branch;]$(echo ${GITHUB_REF#refs/heads/})" - id: extract_branch - if: github.ref == 'refs/heads/master' || contains(github.ref, 'refs/heads/release_') - - - name: Upload Python wheel - shell: bash -l {0} - if: github.ref == 'refs/heads/master' || contains(github.ref, 'refs/heads/release_') - run: | - python -m awscli s3 cp python-package/dist/*.whl s3://xgboost-nightly-builds/${{ steps.extract_branch.outputs.branch }}/ --acl public-read - env: - AWS_ACCESS_KEY_ID: ${{ secrets.AWS_ACCESS_KEY_ID_IAM_S3_UPLOADER }} - AWS_SECRET_ACCESS_KEY: ${{ secrets.AWS_SECRET_ACCESS_KEY_IAM_S3_UPLOADER }} diff --git a/.github/workflows/python_wheels.yml b/.github/workflows/python_wheels.yml new file mode 100644 index 000000000000..1a2a8c7f5f10 --- /dev/null +++ b/.github/workflows/python_wheels.yml @@ -0,0 +1,57 @@ +name: XGBoost-Python-Wheels + +on: [push, pull_request] + +jobs: + python-wheels: + name: Build wheel for ${{ matrix.platform_id }} + runs-on: ${{ matrix.os }} + strategy: + matrix: + include: + - os: macos-latest + python: 37 + platform_id: macosx_x86_64 + wheel_tag: macosx_10_15_x86_64.macosx_11_0_x86_64.macosx_12_0_x86_64 + - os: macos-latest + python: 38 + platform_id: macosx_arm64 + wheel_tag: macosx_12_0_arm64 + steps: + - uses: actions/checkout@v2 + with: + submodules: 'true' + - name: Setup Python + uses: actions/setup-python@v2 + with: + python-version: '3.9' + - name: Set env var for ARM64 + shell: bash + run: echo "::set-output name=value::CIBW_TARGET_OSX_ARM64=1" + id: arm64_flag + if: matrix.platform_id == 'macosx_arm64' + - name: Build wheels + env: + CIBW_BUILD: cp${{ matrix.python }}-${{ matrix.platform_id }} + CIBW_ARCHS: all + CIBW_ENVIRONMENT: ${{ steps.arm64_flag.outputs.value }} + CIBW_TEST_SKIP: "*-macosx_arm64" + CIBW_BUILD_VERBOSITY: 3 + run: bash tests/ci_build/build_python_wheels.sh + + - name: Rename Python wheel + run: | + python tests/ci_build/rename_whl.py wheelhouse/*.whl ${{ github.sha }} ${{ matrix.wheel_tag }} + - name: Extract branch name + shell: bash + run: echo "##[set-output name=branch;]$(echo ${GITHUB_REF#refs/heads/})" + id: extract_branch + if: github.ref == 'refs/heads/master' || contains(github.ref, 'refs/heads/release_') + - name: Upload Python wheel + if: github.ref == 'refs/heads/master' || contains(github.ref, 'refs/heads/release_') + run: | + python -m pip install awscli + python -m awscli s3 cp wheelhouse/*.whl s3://xgboost-nightly-builds/${{ steps.extract_branch.outputs.branch }}/ --acl public-read + env: + AWS_ACCESS_KEY_ID: ${{ secrets.AWS_ACCESS_KEY_ID_IAM_S3_UPLOADER }} + AWS_SECRET_ACCESS_KEY: ${{ secrets.AWS_SECRET_ACCESS_KEY_IAM_S3_UPLOADER }} diff --git a/python-package/setup.py b/python-package/setup.py index 33f6c231eb56..0a6b7e135174 100644 --- a/python-package/setup.py +++ b/python-package/setup.py @@ -119,6 +119,9 @@ def build( continue cmake_cmd.append('-D' + arg + '=' + value) + if 'CIBW_TARGET_OSX_ARM64' in os.environ: + cmake_cmd.append("-DCMAKE_OSX_ARCHITECTURES=arm64") + self.logger.info('Run CMake command: %s', str(cmake_cmd)) subprocess.check_call(cmake_cmd, cwd=build_dir) diff --git a/tests/ci_build/build_python_wheels.sh b/tests/ci_build/build_python_wheels.sh new file mode 100644 index 000000000000..103c7ef7de11 --- /dev/null +++ b/tests/ci_build/build_python_wheels.sh @@ -0,0 +1,34 @@ +#!/bin/bash + +set -e +set -x + +# OpenMP is not present on macOS by default +if [[ "$RUNNER_OS" == "macOS" ]]; then + # Make sure to use a libomp version binary compatible with the oldest + # supported version of the macos SDK as libomp will be vendored into the + # XGBoost wheels for macos. + + if [[ "$CIBW_BUILD" == *-macosx_arm64 ]]; then + # arm64 builds must cross compile because CI is on x64 + export PYTHON_CROSSENV=1 + export MACOSX_DEPLOYMENT_TARGET=12.0 + OPENMP_URL="https://anaconda.org/conda-forge/llvm-openmp/11.1.0/download/osx-arm64/llvm-openmp-11.1.0-hf3c4609_1.tar.bz2" + else + export MACOSX_DEPLOYMENT_TARGET=10.13 + OPENMP_URL="https://anaconda.org/conda-forge/llvm-openmp/11.1.0/download/osx-64/llvm-openmp-11.1.0-hda6cdc1_1.tar.bz2" + fi + + sudo conda create -n build $OPENMP_URL + PREFIX="/usr/local/miniconda/envs/build" + + export CC=/usr/bin/clang + export CXX=/usr/bin/clang++ + export CPPFLAGS="$CPPFLAGS -Xpreprocessor -fopenmp" + export CFLAGS="$CFLAGS -I$PREFIX/include" + export CXXFLAGS="$CXXFLAGS -I$PREFIX/include" + export LDFLAGS="$LDFLAGS -Wl,-rpath,$PREFIX/lib -L$PREFIX/lib -lomp" +fi + +python -m pip install cibuildwheel +python -m cibuildwheel python-package --output-dir wheelhouse From d770a9edd1d93167e369cd7b6b39841cf0e135f3 Mon Sep 17 00:00:00 2001 From: Hyunsu Cho Date: Sun, 30 Jan 2022 07:41:01 -0800 Subject: [PATCH 2/7] Use Conda's libomp when running Python tests --- .github/workflows/main.yml | 3 --- .github/workflows/python_tests.yml | 12 +++++------- tests/ci_build/conda_env/macos_cpu_test.yml | 1 + 3 files changed, 6 insertions(+), 10 deletions(-) diff --git a/.github/workflows/main.yml b/.github/workflows/main.yml index 819e8a9a68ac..d7ec12c78cd0 100644 --- a/.github/workflows/main.yml +++ b/.github/workflows/main.yml @@ -21,10 +21,7 @@ jobs: submodules: 'true' - name: Install system packages run: | - # Use libomp 11.1.0: https://github.com/dmlc/xgboost/issues/7039 - wget https://raw.githubusercontent.com/Homebrew/homebrew-core/679923b4eb48a8dc7ecc1f05d06063cd79b3fc00/Formula/libomp.rb -O $(find $(brew --repository) -name libomp.rb) brew install ninja libomp - brew pin libomp - name: Build gtest binary run: | mkdir build diff --git a/.github/workflows/python_tests.yml b/.github/workflows/python_tests.yml index a989c4643ace..2d1fd5af4200 100644 --- a/.github/workflows/python_tests.yml +++ b/.github/workflows/python_tests.yml @@ -17,10 +17,7 @@ jobs: - name: Install osx system dependencies if: matrix.os == 'macos-10.15' run: | - # Use libomp 11.1.0: https://github.com/dmlc/xgboost/issues/7039 - wget https://raw.githubusercontent.com/Homebrew/homebrew-core/679923b4eb48a8dc7ecc1f05d06063cd79b3fc00/Formula/libomp.rb -O $(find $(brew --repository) -name libomp.rb) brew install ninja libomp - brew pin libomp - name: Install Ubuntu system dependencies if: matrix.os == 'ubuntu-latest' run: | @@ -120,13 +117,14 @@ jobs: - name: Build XGBoost on macos run: | - wget https://raw.githubusercontent.com/Homebrew/homebrew-core/679923b4eb48a8dc7ecc1f05d06063cd79b3fc00/Formula/libomp.rb -O $(find $(brew --repository) -name libomp.rb) - brew install ninja libomp - brew pin libomp + brew install ninja mkdir build cd build - cmake .. -GNinja -DGOOGLE_TEST=ON -DUSE_DMLC_GTEST=ON + # Set prefix, to use OpenMP library from Conda env + # See https://github.com/dmlc/xgboost/issues/7039#issuecomment-1025038228 + # to learn why we don't use libomp from Homebrew. + cmake .. -GNinja -DGOOGLE_TEST=ON -DUSE_DMLC_GTEST=ON -DCMAKE_PREFIX_PATH=$CONDA_PREFIX ninja - name: Install Python package diff --git a/tests/ci_build/conda_env/macos_cpu_test.yml b/tests/ci_build/conda_env/macos_cpu_test.yml index 7fed0704a4af..c08d21ca4086 100644 --- a/tests/ci_build/conda_env/macos_cpu_test.yml +++ b/tests/ci_build/conda_env/macos_cpu_test.yml @@ -10,6 +10,7 @@ dependencies: - pylint - numpy - scipy +- llvm-openmp - scikit-learn - pandas - matplotlib From e620d440672daba013298343059c8fe5d33a8d81 Mon Sep 17 00:00:00 2001 From: Hyunsu Cho Date: Sun, 30 Jan 2022 07:53:17 -0800 Subject: [PATCH 3/7] fix --- .github/workflows/python_tests.yml | 1 + 1 file changed, 1 insertion(+) diff --git a/.github/workflows/python_tests.yml b/.github/workflows/python_tests.yml index 2d1fd5af4200..55ece3f0dd99 100644 --- a/.github/workflows/python_tests.yml +++ b/.github/workflows/python_tests.yml @@ -116,6 +116,7 @@ jobs: conda list - name: Build XGBoost on macos + shell: bash -l {0} run: | brew install ninja From 90d5ab763d43e8e7914c74c5ebd6aed9ca7b8c25 Mon Sep 17 00:00:00 2001 From: Hyunsu Cho Date: Tue, 1 Feb 2022 14:15:48 -0800 Subject: [PATCH 4/7] Add comment to explain CIBW_TARGET_OSX_ARM64 --- python-package/setup.py | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/python-package/setup.py b/python-package/setup.py index 0a6b7e135174..6138f41bfa92 100644 --- a/python-package/setup.py +++ b/python-package/setup.py @@ -119,6 +119,10 @@ def build( continue cmake_cmd.append('-D' + arg + '=' + value) + # Flag for cross-compiling for Apple Silicon + # We use environment variable because it's the only way to pass down custom flags + # through the cibuildwheel package, which otherwise calls `python setup.py bdist_wheel` + # command. if 'CIBW_TARGET_OSX_ARM64' in os.environ: cmake_cmd.append("-DCMAKE_OSX_ARCHITECTURES=arm64") From 674d0a283fb7ceff09b334d357e1487ff00ed8c2 Mon Sep 17 00:00:00 2001 From: Hyunsu Cho Date: Tue, 1 Feb 2022 14:18:06 -0800 Subject: [PATCH 5/7] Update release script --- dev/release-py-r.py | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/dev/release-py-r.py b/dev/release-py-r.py index 3a639feb51d6..3d58c83c65b5 100644 --- a/dev/release-py-r.py +++ b/dev/release-py-r.py @@ -80,7 +80,8 @@ def download_py_packages(major: int, minor: int, commit_hash: str): "win_amd64", "manylinux2014_x86_64", "manylinux2014_aarch64", - "macosx_10_14_x86_64.macosx_10_15_x86_64.macosx_11_0_x86_64", + "macosx_10_15_x86_64.macosx_11_0_x86_64.macosx_12_0_x86_64", + "macosx_12_0_arm64" ] dir_URL = PREFIX + str(major) + "." + str(minor) + ".0" + "/" From f46d6827bc39da978d464709894b58913a451598 Mon Sep 17 00:00:00 2001 From: Hyunsu Cho Date: Tue, 1 Feb 2022 14:31:04 -0800 Subject: [PATCH 6/7] Add comments in build_python_wheels.sh --- tests/ci_build/build_python_wheels.sh | 8 ++++++-- 1 file changed, 6 insertions(+), 2 deletions(-) diff --git a/tests/ci_build/build_python_wheels.sh b/tests/ci_build/build_python_wheels.sh index 103c7ef7de11..82180a36276d 100644 --- a/tests/ci_build/build_python_wheels.sh +++ b/tests/ci_build/build_python_wheels.sh @@ -3,14 +3,18 @@ set -e set -x -# OpenMP is not present on macOS by default +# Bundle libomp 11.1.0 when targeting MacOS. +# This is a workaround in order to prevent segfaults when running inside a Conda environment. +# See https://github.com/dmlc/xgboost/issues/7039#issuecomment-1025125003 for more context. +# The workaround is also used by the scikit-learn project. if [[ "$RUNNER_OS" == "macOS" ]]; then # Make sure to use a libomp version binary compatible with the oldest # supported version of the macos SDK as libomp will be vendored into the - # XGBoost wheels for macos. + # XGBoost wheels for MacOS. if [[ "$CIBW_BUILD" == *-macosx_arm64 ]]; then # arm64 builds must cross compile because CI is on x64 + # cibuildwheel will take care of cross-compilation. export PYTHON_CROSSENV=1 export MACOSX_DEPLOYMENT_TARGET=12.0 OPENMP_URL="https://anaconda.org/conda-forge/llvm-openmp/11.1.0/download/osx-arm64/llvm-openmp-11.1.0-hf3c4609_1.tar.bz2" From a8451659d788741c50b0dca4284b7de6cfd059be Mon Sep 17 00:00:00 2001 From: Hyunsu Cho Date: Tue, 1 Feb 2022 14:35:49 -0800 Subject: [PATCH 7/7] Document wheel pipeline --- doc/contrib/ci.rst | 12 ++++++++++++ 1 file changed, 12 insertions(+) diff --git a/doc/contrib/ci.rst b/doc/contrib/ci.rst index 642571199e7a..5c2fb3f2ac63 100644 --- a/doc/contrib/ci.rst +++ b/doc/contrib/ci.rst @@ -25,3 +25,15 @@ requests and every update to branches. A few tests however require manual activa details about noLD. This is a requirement for keeping XGBoost on CRAN (the R package index). To invoke this test suite for a particular pull request, simply add a review comment ``/gha run r-nold-test``. (Ordinary comment won't work. It needs to be a review comment.) + +GitHub Actions is also used to build Python wheels targeting MacOS Intel and Apple Silicon. See +`.github/workflows/python_wheels.yml +`_. The +``python_wheels`` pipeline sets up environment variables prefixed ``CIBW_*`` to indicate the target +OS and processor. The pipeline then invokes the script ``build_python_wheels.sh``, which in turns +calls ``cibuildwheel`` to build the wheel. The ``cibuildwheel`` is a library that sets up a +suitable Python environment for each OS and processor target. Since we don't have Apple Silion +machine in GitHub Actions, cross-compilation is needed; ``cibuildwheel`` takes care of the complex +task of cross-compiling a Python wheel. (Note that ``cibuildwheel`` will call +``setup.py bdist_wheel``. Since XGBoost has a native library component, ``setup.py`` contains +a glue code to call CMake and a C++ compiler to build the native library on the fly.)