| #!/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 "$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. |
| export 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 |
| |
| |
| for name in "$LCOV_MERGER"; do |
| if [[ ! -e $name ]]; then |
| echo -- |
| echo Coverage runner: cannot locate file $name |
| exit 1 |
| fi |
| done |
| |
| # Setting up the environment for executing the C++ tests. |
| export GCOV_PREFIX_STRIP=3 |
| export GCOV_PREFIX="${COVERAGE_DIR}" |
| export LLVM_PROFILE_FILE="${COVERAGE_DIR}/%h-%p-%m.profraw" |
| |
| # In coverage mode for Java, we need to merge the runtime classpath before |
| # running the tests. JacocoCoverageRunner uses this merged jar in order |
| # to get coverage data. |
| # |
| # Merge the classpath using SingleJar and save it in the environment |
| # variable JACOCO_METADATA_JAR. The jars on the runtime classpath are listed |
| # in the file $JAVA_RUNTIME_CLASSPATH_FOR_COVERAGE. |
| # |
| # We need to merge the jars here because the merged jar can be an input |
| # too large (the combined merged jars for several big tests in a run |
| # can go over 10G). Not merging the jars and making |
| # JacocoCoverageRunner read every individual jar goes over the shutdown hook |
| # time limit in the coverage runner (~few seconds). |
| # |
| # SINGLE_JAR_TOOL Exec path of SingleJar. |
| # |
| # JAVA_RUNTIME_CLASSPATH_FOR_COVERAGE Exec path of a file that contains the |
| # relative paths of the jars on the runtime |
| # classpath delimited by newline. |
| if [[ ! -z "${JAVA_RUNTIME_CLASSPATH_FOR_COVERAGE}" ]]; then |
| JAVA_RUNTIME_CLASSPATH_FOR_COVERAGE="${PWD}/${JAVA_RUNTIME_CLASSPATH_FOR_COVERAGE}" |
| SINGLE_JAR_TOOL="${PWD}/${SINGLE_JAR_TOOL}" |
| |
| # Create a paramsfile for invoking SingleJar. |
| mkdir -p "${COVERAGE_DIR}" |
| single_jar_params_file="${COVERAGE_DIR}/runtime_classpath.paramsfile" |
| touch "$single_jar_params_file" |
| |
| # Export JACOCO_METADATA_JAR in order for JacocoCoverageRunner to be able |
| # to read it. |
| export JACOCO_METADATA_JAR="${COVERAGE_DIR}/coverage-runtime_merged_instr.jar" |
| |
| echo -e "--output ${JACOCO_METADATA_JAR}\n--sources" >> "$single_jar_params_file" |
| |
| # Append the runfiles prefix to all the relative paths found in |
| # JAVA_RUNTIME_CLASSPATH_FOR_COVERAGE, to invoke SingleJar with the |
| # absolute paths. |
| RUNFILES_PREFIX="$TEST_SRCDIR/$TEST_WORKSPACE/" |
| cat "$JAVA_RUNTIME_CLASSPATH_FOR_COVERAGE" | sed "s@^@$RUNFILES_PREFIX@" >> "$single_jar_params_file" |
| |
| # Invoke SingleJar. This will create JACOCO_METADATA_JAR. |
| "${SINGLE_JAR_TOOL}" "@$single_jar_params_file" |
| fi |
| |
| # TODO(bazel-team): cd should be avoided. |
| cd "$TEST_SRCDIR/$TEST_WORKSPACE" |
| # Execute the test. |
| "$@" |
| TEST_STATUS=$? |
| |
| # Always create the coverage report. |
| 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 |
| |
| # TODO(bazel-team): cd should be avoided. |
| cd $ROOT |
| |
| # Call the C++ code coverage collection script. |
| if [[ "$CC_CODE_COVERAGE_SCRIPT" ]]; then |
| eval "${CC_CODE_COVERAGE_SCRIPT}" |
| fi |
| |
| |
| # Export the command line that invokes LcovMerger with the flags: |
| # --coverage_dir The absolute path of the directory where the |
| # intermediate coverage reports are located. |
| # CoverageOutputGenerator will search for files with |
| # the .dat and .gcov extension under this directory and |
| # will merge everything it found in the output report. |
| # |
| # --output_file The absolute path of the merged coverage report. |
| # |
| # --filter_sources Filters out the sources that match the given regexes |
| # from the final coverage report. This is needed |
| # because some coverage tools (e.g. gcov) do not have |
| # any way of specifying what sources to exclude when |
| # generating the code coverage report (in this case the |
| # syslib sources). |
| # |
| # --source_file_manifest The absolute path of the coverage source file |
| # manifest. CoverageOutputGenerator uses this file to |
| # keep only the C++ sources found in the manifest. |
| # For other languages the sources in the manifest are |
| # ignored. |
| LCOV_MERGER_CMD="${LCOV_MERGER} --coverage_dir=${COVERAGE_DIR} \ |
| --output_file=${COVERAGE_OUTPUT_FILE} \ |
| --filter_sources=/usr/bin/.+ \ |
| --filter_sources=/usr/lib/.+ \ |
| --filter_sources=/usr/include.+ \ |
| --filter_sources=.*external/.+ \ |
| --source_file_manifest=${COVERAGE_MANIFEST}" |
| |
| if [[ $COVERAGE_REPORTED_TO_ACTUAL_SOURCES_FILE ]]; then |
| LCOV_MERGER_CMD="$LCOV_MERGER_CMD\ |
| --sources_to_replace_file=$ROOT/$COVERAGE_REPORTED_TO_ACTUAL_SOURCES_FILE" |
| fi |
| |
| 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 |