-
Notifications
You must be signed in to change notification settings - Fork 98
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
1 parent
0f08aea
commit 81fd663
Showing
3 changed files
with
344 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,119 @@ | ||
name: Build and collect test coverage | ||
|
||
on: | ||
push: | ||
branches: | ||
- main | ||
pull_request: | ||
types: [assigned, opened, synchronize, reopened] | ||
workflow_dispatch: | ||
|
||
defaults: | ||
run: | ||
# This is already the default, except when running inside another Docker | ||
# image, which is the case here. So set it up globally to avoid | ||
# repeating elsewhere. | ||
shell: bash | ||
|
||
env: | ||
# Run apt package manager in the CI in non-interactive mode. | ||
# Otherwise, on Ubuntu 20.04 the installation of tzdata asking question | ||
DEBIAN_FRONTEND: noninteractive | ||
|
||
jobs: | ||
build-repo: | ||
name: Build and Code Coverage | ||
|
||
# By latest GitHub means actually latest LTS only | ||
runs-on: ubuntu-latest | ||
|
||
strategy: | ||
# Run all the test even if there are some which fail | ||
fail-fast: false | ||
|
||
# Run the tests on the Cartesian product of the following | ||
matrix: | ||
ubuntu_version: [ 22.04 ] | ||
|
||
steps: | ||
# Clone the repo and its submodules. Do shallow clone to save clone | ||
# time. | ||
- name: Get the project repository | ||
uses: actions/checkout@v3 | ||
with: | ||
fetch-depth: 2 | ||
submodules: "true" | ||
|
||
- uses: actions/setup-python@v4 | ||
with: | ||
python-version: '3.11' | ||
|
||
- name: Install Python and other packages | ||
# Install cmake here to get the latest version to compile | ||
# LLVM. The Ubuntu 20.04 cmake version is only 3.16.3 | ||
run: | | ||
pip install cmake numpy psutil pybind11 rich | ||
- name: Install Ninja | ||
# Can not use the following since it wants to use `sudo` not | ||
# available in the case of Docker execution | ||
# https://github.com/llvm/actions/tree/main/install-ninja | ||
# uses: llvm/actions/install-ninja@6a57890d0e3f9f35dfc72e7e48bc5e1e527cdd6c | ||
# So just use the specific implementation instead: | ||
run: sudo apt-get install -y ninja-build | ||
|
||
- name: Install llvm-cov | ||
run: sudo apt-get install -y clang lld llvm | ||
|
||
- name: Get LLVM | ||
id: clone-llvm | ||
run: utils/clone-llvm.sh | ||
|
||
- name: Get LLVM commit hash | ||
id: get-llvm-commit-hash | ||
# Get the LLVM commit hash to be used in the ccache database key to | ||
# avoid mixing different binaries compiled by different LLVM versions | ||
run: echo "hash=$(cd llvm ; git log -1 --format='%H')" >> $GITHUB_OUTPUT | ||
|
||
- name: Ccache for C++ compilation | ||
# https://github.com/hendrikmuhs/ccache-action/releases/tag/v1.2.9 | ||
uses: hendrikmuhs/ccache-action@ca3acd2731eef11f1572ccb126356c2f9298d35e | ||
with: | ||
# Since there are now several compilation jobs running in parallel, | ||
# use a different key per job to avoid a ccache writing race condition | ||
key: ${{ runner.os }}-${{ matrix.ubuntu_version }}-${{ steps.get-llvm-commit-hash.outputs.hash }} | ||
max-size: 1G | ||
|
||
- name: Build and install LLVM | ||
run: LLVM_ENABLE_RTTI=ON utils/build-llvm.sh | ||
|
||
- name: Install our python reqs | ||
run: pip install -r python/requirements.txt | ||
|
||
- name: Build and generate coverage (Release) | ||
run: | | ||
mkdir build_release | ||
cd build_release | ||
cmake .. \ | ||
-GNinja \ | ||
-DCMAKE_C_COMPILER=clang -DCMAKE_CXX_COMPILER=clang++ \ | ||
-DCMAKE_C_COMPILER_LAUNCHER=ccache -DCMAKE_CXX_COMPILER_LAUNCHER=ccache \ | ||
-DCMAKE_EXE_LINKER_FLAGS_INIT="-fuse-ld=lld" -DCMAKE_MODULE_LINKER_FLAGS_INIT="-fuse-ld=lld" -DCMAKE_SHARED_LINKER_FLAGS_INIT="-fuse-ld=lld" \ | ||
-DCMAKE_BUILD_TYPE=Release \ | ||
-DCMAKE_PLATFORM_NO_VERSIONED_SONAME=ON \ | ||
-DCMAKE_VISIBILITY_INLINES_HIDDEN=ON \ | ||
-DCMAKE_C_VISIBILITY_PRESET=hidden \ | ||
-DCMAKE_CXX_VISIBILITY_PRESET=hidden \ | ||
-DAIE_COMPILER=NONE \ | ||
-DAIE_LINKER=NONE \ | ||
-DHOST_COMPILER=NONE \ | ||
-DBUILD_INSTRUMENTED_COVERAGE=ON \ | ||
-DLLVM_ENABLE_ASSERTIONS=OFF \ | ||
-DLLVM_ENABLE_RTTI=ON \ | ||
-DCMAKE_MODULE_PATH=`pwd`/../cmake/modulesXilinx \ | ||
-DMLIR_DIR=../llvm/install/lib/cmake/mlir \ | ||
-DLLVM_DIR=../llvm/install/lib/cmake/llvm \ | ||
-DLLVM_USE_LINKER=lld \ | ||
-DLLVM_EXTERNAL_LIT=`pwd`/../llvm/build/bin/llvm-lit | ||
ninja && ninja generate-aie-coverage-report |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,198 @@ | ||
#!/usr/bin/env python | ||
|
||
from __future__ import print_function | ||
|
||
"""Prepare a code coverage artifact. | ||
- Collate raw profiles into one indexed profile. | ||
- Generate html reports for the given binaries. | ||
Caution: The positional arguments to this script must be specified before any | ||
optional arguments, such as --restrict. | ||
""" | ||
|
||
import argparse | ||
import glob | ||
import os | ||
import subprocess | ||
import sys | ||
|
||
|
||
def merge_raw_profiles(host_llvm_profdata, profile_data_dir, preserve_profiles): | ||
print(":: Merging raw profiles...", end="") | ||
sys.stdout.flush() | ||
raw_profiles = glob.glob(f"{profile_data_dir}/**/*.profraw", recursive=True) | ||
manifest_path = os.path.join(profile_data_dir, "profiles.manifest") | ||
profdata_path = os.path.join(profile_data_dir, "Coverage.profdata") | ||
with open(manifest_path, "w") as manifest: | ||
manifest.write("\n".join(raw_profiles)) | ||
subprocess.check_call( | ||
[ | ||
host_llvm_profdata, | ||
"merge", | ||
"-sparse", | ||
"-f", | ||
manifest_path, | ||
"-o", | ||
profdata_path, | ||
] | ||
) | ||
if not preserve_profiles: | ||
for raw_profile in raw_profiles: | ||
os.remove(raw_profile) | ||
# os.remove(manifest_path) | ||
print("Done!") | ||
return profdata_path | ||
|
||
|
||
def prepare_html_report( | ||
host_llvm_cov, profile, report_dir, binaries, restricted_dirs, compilation_dir | ||
): | ||
print(":: Preparing html report for {0}...".format(binaries), end="") | ||
sys.stdout.flush() | ||
objects = [] | ||
for i, binary in enumerate(binaries): | ||
if i == 0: | ||
objects.append(binary) | ||
else: | ||
objects.extend(("-object", binary)) | ||
invocation = ( | ||
[host_llvm_cov, "show"] | ||
+ objects | ||
+ [ | ||
"-format", | ||
"html", | ||
"-instr-profile", | ||
profile, | ||
"-o", | ||
report_dir, | ||
"-show-line-counts-or-regions", | ||
# "-show-directory-coverage", | ||
"-Xdemangler", | ||
"c++filt", | ||
"-Xdemangler", | ||
"-n", | ||
] | ||
+ restricted_dirs | ||
) | ||
if compilation_dir: | ||
invocation += ["-compilation-dir=" + compilation_dir] | ||
subprocess.check_call(invocation) | ||
with open(os.path.join(report_dir, "summary.txt"), "wb") as Summary: | ||
subprocess.check_call( | ||
[host_llvm_cov, "report"] | ||
+ objects | ||
+ ["-instr-profile", profile] | ||
+ restricted_dirs, | ||
stdout=Summary, | ||
) | ||
print("Done!") | ||
|
||
|
||
def prepare_html_reports( | ||
host_llvm_cov, | ||
profdata_path, | ||
report_dir, | ||
binaries, | ||
unified_report, | ||
restricted_dirs, | ||
compilation_dir, | ||
): | ||
if unified_report: | ||
prepare_html_report( | ||
host_llvm_cov, | ||
profdata_path, | ||
report_dir, | ||
binaries, | ||
restricted_dirs, | ||
compilation_dir, | ||
) | ||
else: | ||
for binary in binaries: | ||
binary_report_dir = os.path.join(report_dir, os.path.basename(binary)) | ||
prepare_html_report( | ||
host_llvm_cov, | ||
profdata_path, | ||
binary_report_dir, | ||
[binary], | ||
restricted_dirs, | ||
compilation_dir, | ||
) | ||
|
||
|
||
if __name__ == "__main__": | ||
parser = argparse.ArgumentParser(description=__doc__) | ||
parser.add_argument("host_llvm_profdata", help="Path to llvm-profdata") | ||
parser.add_argument("host_llvm_cov", help="Path to llvm-cov") | ||
parser.add_argument( | ||
"profile_data_dir", help="Path to the directory containing the raw profiles" | ||
) | ||
parser.add_argument( | ||
"report_dir", help="Path to the output directory for html reports" | ||
) | ||
parser.add_argument( | ||
"binaries", | ||
metavar="B", | ||
type=str, | ||
nargs="*", | ||
help="Path to an instrumented binary", | ||
) | ||
parser.add_argument( | ||
"--only-merge", | ||
action="store_true", | ||
help="Only merge raw profiles together, skip report " "generation", | ||
) | ||
parser.add_argument( | ||
"--preserve-profiles", help="Do not delete raw profiles", action="store_true" | ||
) | ||
parser.add_argument( | ||
"--use-existing-profdata", help="Specify an existing indexed profile to use" | ||
) | ||
parser.add_argument( | ||
"--unified-report", | ||
action="store_true", | ||
help="Emit a unified report for all binaries", | ||
) | ||
parser.add_argument( | ||
"--restrict", | ||
metavar="R", | ||
type=str, | ||
nargs="*", | ||
default=[], | ||
help="Restrict the reporting to the given source paths" | ||
" (must be specified after all other positional arguments)", | ||
) | ||
parser.add_argument( | ||
"-C", | ||
"--compilation-dir", | ||
type=str, | ||
default="", | ||
help="The compilation directory of the binary", | ||
) | ||
args = parser.parse_args() | ||
|
||
if args.use_existing_profdata and args.only_merge: | ||
print("--use-existing-profdata and --only-merge are incompatible") | ||
exit(1) | ||
|
||
if args.use_existing_profdata: | ||
profdata_path = args.use_existing_profdata | ||
else: | ||
profdata_path = merge_raw_profiles( | ||
args.host_llvm_profdata, args.profile_data_dir, args.preserve_profiles | ||
) | ||
|
||
if not len(args.binaries): | ||
print("No binaries specified, no work to do!") | ||
exit(1) | ||
|
||
if not args.only_merge: | ||
prepare_html_reports( | ||
args.host_llvm_cov, | ||
profdata_path, | ||
args.report_dir, | ||
args.binaries, | ||
args.unified_report, | ||
args.restrict, | ||
args.compilation_dir, | ||
) |