# Part of the Carbon Language project, under the Apache License v2.0 with LLVM # Exceptions. See /LICENSE for license information. # SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception name: test on: push: branches: [trunk, action-test] pull_request: merge_group: permissions: contents: read # For actions/checkout. pull-requests: read # For dorny/paths-filter to read pull requests. # Cancel previous workflows on the PR when there are multiple fast commits. # https://docs.github.com/en/actions/using-workflows/workflow-syntax-for-github-actions#concurrency concurrency: group: ${{ github.workflow }}-${{ github.head_ref || github.ref }} cancel-in-progress: true jobs: test: strategy: matrix: # On PRs and in the merge queue test a recent version of each supported # OS. On push (post-submit), also run on `macos-12` to get Intel macOS # coverage. runner: ${{ fromJSON(github.event_name != 'push' && '["ubuntu-22.04", "macos-14"]' || '["ubuntu-22.04", "macos-14", "macos-12"]') }} build_mode: [fastbuild, opt] include: # The clang-tidy config doesn't work on macos (missing `truncate`). - runner: ubuntu-22.04 build_mode: clang-tidy runs-on: ${{ matrix.runner }} env: os: ${{ startsWith(matrix.runner, 'ubuntu') && 'ubuntu' || 'macos' }} steps: - name: Harden Runner uses: step-security/harden-runner@63c24ba6bd7ba022e95695ff85de572c04a18142 # v2.7.0 with: egress-policy: audit # Ubuntu images start with 23GB available, and this adds 14GB more. For # comparison, MacOS images have >100GB free. # # Although we could delete more, if we run into a limit, not deleting # everything provides a little flexibility to get space while trying # to shrink the build. - name: Free up disk space (Ubuntu) if: env.os == 'ubuntu' uses: jlumbroso/free-disk-space@54081f138730dfa15788a46383842cd2f914a1be # v1.3.1 with: android: true dotnet: true haskell: true # Checkout the pull request head or the branch. - name: Checkout pull request if: github.event_name == 'pull_request' uses: actions/checkout@b4ffde65f46336ab88eb53be808477a3936bae11 # v4.1.1 with: ref: ${{ github.event.pull_request.head.sha }} - name: Checkout branch if: github.event_name != 'pull_request' uses: actions/checkout@b4ffde65f46336ab88eb53be808477a3936bae11 # v4.1.1 # Tests should only run on applicable paths, but we still need to have an # action run for the merge queue. We filter steps based on the paths here, # and condition steps on the output. - id: filter uses: dorny/paths-filter@de90cc6fb38fc0963ad72b210f1f284cd68cea36 # v3.0.2 with: filters: | has_code: - '!{**/*.md,LICENSE,CODEOWNERS,.git*}' # Setup Python and related tools. - uses: actions/setup-python@82c7e631bb3cdc910f68e0081d67478d79c6982d # v5.1.0 if: steps.filter.outputs.has_code == 'true' with: # Match the min version listed in docs/project/contribution_tools.md # or the oldest version available on the OS. python-version: ${{ matrix.runner == 'macos-14' && '3.11' || '3.9' }} # Install and cache LLVM 16 from Homebrew. # TODO: We can potentially remove this and simplify things when the # Homebrew version of LLVM updates to 16 here: # https://github.com/actions/runner-images/blob/main/images/macos/macos-12-Readme.md - name: Cache Homebrew (macOS) if: steps.filter.outputs.has_code == 'true' && env.os == 'macos' id: cache-homebrew-macos uses: actions/cache@0c45773b623bea8c8e75f6c82b208c3cf94ea4f9 # v4.0.2 env: cache-name: cache-homebrew with: # Cover all the critical parts of Homebrew here. Homebrew on Arm macOS # uses its own prefix making this easy to cover, but we need a few # different paths for Intel. path: | ${{ runner.arch == 'ARM64' && '/opt/homebrew' || ' /usr/local/Homebrew /usr/local/Cellar /usr/local/Frameworks /usr/local/bin /usr/local/opt ' }} # Note the key needs to include all the packages we're adding. key: Homebrew-Cache-${{ matrix.runner }}-${{ runner.arch }} - name: Install LLVM and Clang with Homebrew (macOS) if: steps.filter.outputs.has_code == 'true' && env.os == 'macos' && steps.cache-homebrew-macos.outputs.cache-hit != 'true' run: | echo '*** Prune brew leaves' # We prune all the leaf packages to have a minimal environment. This # both minimizes the install space and avoids accidental dependencies # on installed packages. brew leaves LEAVES=$(brew leaves | egrep -v '^(bazelisk|gh|git|git-lfs|gnu-tar|go@.*|jq|pipx|node@.*|openssl@.*|wget|yq|zlib)$') brew uninstall -f --ignore-dependencies $LEAVES echo '*** Installing LLVM deps' brew install --force-bottle --only-dependencies llvm@16 echo '*** Installing LLVM itself' brew install --force-bottle --force --verbose llvm@16 echo '*** brew info llvm@16' brew info llvm@16 echo '*** brew autoremove' brew autoremove echo '*** brew info' brew info echo '*** brew leaves' brew leaves echo '*** brew config' brew config - name: Setup LLVM and Clang (macOS) if: steps.filter.outputs.has_code == 'true' && env.os == 'macos' run: | LLVM_PATH="$(brew --prefix llvm@16)" echo "Using ${LLVM_PATH}" echo "${LLVM_PATH}/bin" >> $GITHUB_PATH echo '*** ls "${LLVM_PATH}"' ls "${LLVM_PATH}" echo '*** ls "${LLVM_PATH}/bin"' ls "${LLVM_PATH}/bin" # Cache and install a recent version of LLVM. This uses the GitHub action # cache to avoid directly downloading on each iteration and improve # reliability. - name: Cache LLVM and Clang installation (Ubuntu) if: steps.filter.outputs.has_code == 'true' && env.os == 'ubuntu' id: cache-llvm-ubuntu uses: actions/cache@0c45773b623bea8c8e75f6c82b208c3cf94ea4f9 # v4.0.2 env: cache-name: cache-llvm with: path: ~/llvm key: LLVM-16-Cache-${{ env.os }}-${{ runner.arch }} - name: Download LLVM and Clang installation (Ubuntu) if: steps.filter.outputs.has_code == 'true' && env.os == 'ubuntu' && steps.cache-llvm-ubuntu.outputs.cache-hit != 'true' run: | cd ~ LLVM_RELEASE=clang+llvm-16.0.4-x86_64-linux-gnu-ubuntu-22.04 echo "*** Downloading $LLVM_RELEASE" wget --show-progress=off "https://github.com/llvm/llvm-project/releases/download/llvmorg-16.0.4/$LLVM_RELEASE.tar.xz" echo "*** Extracting $LLVM_RELEASE" tar -xJf "$LLVM_RELEASE.tar.xz" echo "*** Moving to 'llvm'" mv "$LLVM_RELEASE" llvm echo "*** Testing `clang++ --version`" ~/llvm/bin/clang++ --version # The installation contains *huge* parts of LLVM we don't need for the # toolchain. Prune them here to keep our cache small. echo "*** Cleaning the 'llvm' directory" rm llvm/lib/{*.a,*.so,*.so.*,*.bc} rm llvm/bin/{flang-*,mlir-*,clang-{scan-deps,check,repl},*-test,llvm-{lto*,reduce,bolt*,exegesis,jitlink},bugpoint,opt,llc} echo "*** Size of the 'llvm' directory" du -hs llvm - name: Setup LLVM and Clang paths (Ubuntu) if: steps.filter.outputs.has_code == 'true' && env.os == 'ubuntu' run: | LLVM_PATH=~/llvm echo "Using ${LLVM_PATH}" echo "${LLVM_PATH}/bin" >> $GITHUB_PATH echo '*** ls "${LLVM_PATH}"' ls "${LLVM_PATH}" echo '*** ls "${LLVM_PATH}/bin"' ls "${LLVM_PATH}/bin" # Print the various tool paths and versions to help in debugging. - name: Print tool debugging info if: steps.filter.outputs.has_code == 'true' run: | echo '*** PATH' echo $PATH echo '*** bazelisk' which bazelisk bazelisk --version echo '*** run_bazel.py' ./scripts/run_bazel.py --version echo '*** python' which python python --version echo '*** clang' which clang clang --version echo '*** clang++' which clang++ clang++ --version # Disable uploads when the remote cache is read-only. - name: Set up remote cache access (read-only) if: steps.filter.outputs.has_code == 'true' && github.event_name == 'pull_request' run: | echo "remote_cache_upload=--remote_upload_local_results=false" \ >> $GITHUB_ENV # Provide a cache key when the remote cache is read-write. - name: Set up remote cache access (read-write) if: steps.filter.outputs.has_code == 'true' && github.event_name != 'pull_request' env: REMOTE_CACHE_KEY: ${{ secrets.CARBON_BUILDS_GITHUB }} run: | echo "$REMOTE_CACHE_KEY" | base64 -d > $HOME/remote_cache_key.json echo "remote_cache_upload=--google_credentials=$HOME/remote_cache_key.json" \ >> $GITHUB_ENV # Add our bazel configuration and print basic info to ease debugging. - name: Configure Bazel and print info if: steps.filter.outputs.has_code == 'true' env: # Add a cache version for changes that bazel won't otherwise detect, # like llvm version changes. CACHE_VERSION: 1 run: | cat >user.bazelrc <$TARGETS_FILE # Compute the set of possible rules impacted by this change using # Bazel-based diffing. This lets PRs and the merge queue have a much more # efficient test CI action by avoiding even enumerating (and downloading) # all of the unaffected Bazel targets. - name: Compute impacted pull request targets if: steps.filter.outputs.has_code == 'true' && github.event_name != 'push' env: # Compute the base SHA from the different event structures. GIT_BASE_SHA: ${{ github.event_name == 'pull_request' && github.event.pull_request.base.sha || github.event.merge_group.base_sha }} TARGETS_FILE: ${{ runner.temp }}/targets run: | # First fetch the relevant base into the git repository. git fetch --depth=1 origin $GIT_BASE_SHA # Then use `target-determinator` as wrapped by our script. ./scripts/target_determinator.py $GIT_BASE_SHA >$TARGETS_FILE # Bazel requires a test target to run the test command. There may be # no targets or there may only be non-test targets that we want to # build, so simply inject an explicit no-op test target. echo "//scripts:no_op_test" >> $TARGETS_FILE # Build and run just the tests impacted by the PR or merge group. - name: Test (${{ matrix.build_mode }}) if: steps.filter.outputs.has_code == 'true' && matrix.build_mode != 'clang-tidy' env: # 'libtool_check_unique failed to generate' workaround. # https://github.com/bazelbuild/bazel/issues/14113#issuecomment-999794586 BAZEL_USE_CPP_ONLY_TOOLCHAIN: 1 TARGETS_FILE: ${{ runner.temp }}/targets run: | # Decrease the jobs sharply if we see repeated failures to try to # work around transient network errors even if it makes things # slower. ./scripts/run_bazel.py \ --attempts=5 --jobs-on-last-attempt=4 \ test -c ${{ matrix.build_mode }} \ --target_pattern_file=$TARGETS_FILE # Run in the clang-tidy config. This is done as part of tests so that we # aren't duplicating bazel/llvm setup. # # The `-k` flag is used to print all clang-tidy errors. - name: clang-tidy if: steps.filter.outputs.has_code == 'true' && matrix.build_mode == 'clang-tidy' env: TARGETS_FILE: ${{ runner.temp }}/targets run: | ./scripts/run_bazel.py \ --attempts=5 \ build --config=clang-tidy -k \ --target_pattern_file=$TARGETS_FILE # See "Disk space before build". - name: Disk space after build if: steps.filter.outputs.has_code == 'true' run: df -h