#!/bin/bash -x # Copyright 2016 The Bazel Authors. All rights reserved. # # Licensed under the Apache License, Version 2.0 (the "License"); # you may not use this file except in compliance with the License. # You may obtain a copy of the License at # # http://www.apache.org/licenses/LICENSE-2.0 # # Unless required by applicable law or agreed to in writing, software # distributed under the License is distributed on an "AS IS" BASIS, # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. # See the License for the specific language governing permissions and # limitations under the License. # Wrapper script for collecting code coverage during test execution. # # Expected environment: # COVERAGE_MANIFEST - mandatory, location of the instrumented file manifest # LCOV_MERGER - mandatory, location of the LcovMerger # COVERAGE_DIR - optional, location of the coverage temp directory # COVERAGE_OUTPUT_FILE - optional, location of the final lcov file # # Script expects that it will be started in the execution root directory and # not in the test's runfiles directory. if [[ -z "$LCOV_MERGER" ]]; then echo -- echo "Coverage collection running in legacy mode." echo "Legacy mode only supports C++ and even then, it's very brittle." COVERAGE_LEGACY_MODE=1 else COVERAGE_LEGACY_MODE= fi if [[ -z "$COVERAGE_MANIFEST" ]]; then echo -- echo Coverage runner: \$COVERAGE_MANIFEST is not set echo Current environment: env | sort exit 1 fi # When collect_coverage.sh is used, test runner must be instructed not to cd # to the test's runfiles directory. ROOT="$PWD" if [[ "$COVERAGE_MANIFEST" != /* ]]; then # Canonicalize the path to coverage manifest so that tests can find it. export COVERAGE_MANIFEST="$ROOT/$COVERAGE_MANIFEST" fi # write coverage data outside of the runfiles tree export COVERAGE_DIR=${COVERAGE_DIR:-"$ROOT/coverage"} # make COVERAGE_DIR an absolute path if ! [[ $COVERAGE_DIR == $ROOT* ]]; then COVERAGE_DIR=$ROOT/$COVERAGE_DIR fi mkdir -p "$COVERAGE_DIR" COVERAGE_OUTPUT_FILE=${COVERAGE_OUTPUT_FILE:-"$COVERAGE_DIR/_coverage.dat"} # make COVERAGE_OUTPUT_FILE an absolute path if ! [[ $COVERAGE_OUTPUT_FILE == $ROOT* ]]; then COVERAGE_OUTPUT_FILE=$ROOT/$COVERAGE_OUTPUT_FILE fi # Java # -------------------------------------- export JAVA_COVERAGE_FILE=$COVERAGE_DIR/jvcov.dat # Let tests know that it is a coverage run export COVERAGE=1 export BULK_COVERAGE_RUN=1 # Only check if file exists when LCOV_MERGER is set if [[ ! "$COVERAGE_LEGACY_MODE" ]]; then for name in "$LCOV_MERGER"; do if [[ ! -e $name ]]; then echo -- echo Coverage runner: cannot locate file $name exit 1 fi done fi if [[ "$COVERAGE_LEGACY_MODE" ]]; then export GCOV_PREFIX_STRIP=3 export GCOV_PREFIX="${COVERAGE_DIR}" export LLVM_PROFILE_FILE="${COVERAGE_DIR}/%h-%p-%m.profraw" fi cd "$TEST_SRCDIR/$TEST_WORKSPACE" "$@" TEST_STATUS=$? # always create output files touch $COVERAGE_OUTPUT_FILE if [[ $TEST_STATUS -ne 0 ]]; then echo -- echo Coverage runner: Not collecting coverage for failed test. echo The following commands failed with status $TEST_STATUS echo "$@" exit $TEST_STATUS fi cd $ROOT USES_LLVM_COV= if stat "${COVERAGE_DIR}"/*.profraw >/dev/null 2>&1; then USES_LLVM_COV=1 fi if [[ "$USES_LLVM_COV" ]]; then "${COVERAGE_GCOV_PATH}" merge -output "${COVERAGE_OUTPUT_FILE}" "${COVERAGE_DIR}"/*.profraw exit $TEST_STATUS # If LCOV_MERGER is not set, use the legacy C++-only method to convert coverage files. elif [[ "$COVERAGE_LEGACY_MODE" ]]; then cat "${COVERAGE_MANIFEST}" | grep ".gcno$" | while read path; do mkdir -p "${COVERAGE_DIR}/$(dirname ${path})" cp "${ROOT}/${path}" "${COVERAGE_DIR}/${path}" done # Symlink the gcov tool such with a link called gcov. Clang comes with a tool # called llvm-cov, which behaves like gcov if symlinked in this way (otherwise # we would need to invoke it with "llvm-cov gcov"). GCOV="${COVERAGE_DIR}/gcov" ln -s "${COVERAGE_GCOV_PATH}" "${GCOV}" # Run lcov over the .gcno and .gcda files to generate the lcov tracefile. # -c - Collect coverage data # --no-external - Do not collect coverage data for system files # --ignore-errors graph - Ignore missing .gcno files; Bazel only instruments some files # -q - Quiet mode # --gcov-tool "${GCOV}" - Pass the local symlink to be uses as gcov by lcov # -b /proc/self/cwd - Use this as a prefix for all source files instead of # the current directory # -d "${COVERAGE_DIR}" - Directory to search for .gcda files # -o "${COVERAGE_OUTPUT_FILE}" - Output file LCOV=$(which lcov) if [[ ! -x $LCOV ]]; then LCOV=/usr/bin/lcov fi $LCOV -c --no-external --ignore-errors graph -q \ --gcov-tool "${GCOV}" -b /proc/self/cwd \ -d "${COVERAGE_DIR}" -o "${COVERAGE_OUTPUT_FILE}" # Fix up the paths to be relative by removing the prefix we specified above. sed -i -e "s*/proc/self/cwd/**g" "${COVERAGE_OUTPUT_FILE}" exit $TEST_STATUS fi export LCOV_MERGER_CMD="${LCOV_MERGER} --coverage_dir=${COVERAGE_DIR} \ --output_file=${COVERAGE_OUTPUT_FILE}" if [[ $DISPLAY_LCOV_CMD ]] ; then echo "Running lcov_merger" echo $LCOV_MERGER_CMD echo "-----------------" fi # JAVA_RUNFILES is set to the runfiles of the test, which does not necessarily # contain a JVM (it does only if the test has a Java binary somewhere). So let # the LCOV merger discover where its own runfiles tree is. JAVA_RUNFILES= exec $LCOV_MERGER_CMD