|  | #!/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 --printf='' "${COVERAGE_DIR}"/*.profraw 2>/dev/null; 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 | 
|  |  | 
|  | # Unfortunately, lcov messes up the source file names if it can't find the | 
|  | # files at their relative paths. Workaround by creating empty source files | 
|  | # according to the manifest (i.e., only for files that are supposed to be | 
|  | # instrumented). | 
|  | cat "${COVERAGE_MANIFEST}" | egrep ".(cc|h)$" | while read path; do | 
|  | mkdir -p "${COVERAGE_DIR}/$(dirname ${path})" | 
|  | touch "${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 | 
|  | # --gcov-tool "${GCOV}" - Pass the local symlink to be uses as gcov by lcov | 
|  | # -d "${COVERAGE_DIR}" - Directory to search for .gcda files | 
|  | # -o "${COVERAGE_OUTPUT_FILE}" - Output file | 
|  | /usr/bin/lcov -c --no-external --ignore-errors graph \ | 
|  | --gcov-tool "${GCOV}" \ | 
|  | -d "${COVERAGE_DIR}" -o "${COVERAGE_OUTPUT_FILE}" | 
|  |  | 
|  | # The paths are all wrong, because they point to /tmp. Fix up the paths to | 
|  | # point to the exec root instead (${ROOT}). | 
|  | # This does not work with sandboxing, because ${ROOT} points to the sandbox dir. | 
|  | sed -i -e "s*${COVERAGE_DIR}*${ROOT}*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 |