|
|
|
# Distributed under the OSI-approved BSD 3-Clause License. See accompanying
|
|
|
|
# file Copyright.txt or https://cmake.org/licensing for details.
|
|
|
|
|
|
|
|
#.rst:
|
|
|
|
# CTestCoverageCollectGCOV
|
|
|
|
# ------------------------
|
|
|
|
#
|
|
|
|
# This module provides the ``ctest_coverage_collect_gcov`` function.
|
|
|
|
#
|
|
|
|
# This function runs gcov on all .gcda files found in the binary tree
|
|
|
|
# and packages the resulting .gcov files into a tar file.
|
|
|
|
# This tarball also contains the following:
|
|
|
|
#
|
|
|
|
# * *data.json* defines the source and build directories for use by CDash.
|
|
|
|
# * *Labels.json* indicates any :prop_sf:`LABELS` that have been set on the
|
|
|
|
# source files.
|
|
|
|
# * The *uncovered* directory holds any uncovered files found by
|
|
|
|
# :variable:`CTEST_EXTRA_COVERAGE_GLOB`.
|
|
|
|
#
|
|
|
|
# After generating this tar file, it can be sent to CDash for display with the
|
|
|
|
# :command:`ctest_submit(CDASH_UPLOAD)` command.
|
|
|
|
#
|
|
|
|
# .. command:: cdash_coverage_collect_gcov
|
|
|
|
#
|
|
|
|
# ::
|
|
|
|
#
|
|
|
|
# ctest_coverage_collect_gcov(TARBALL <tarfile>
|
|
|
|
# [SOURCE <source_dir>][BUILD <build_dir>]
|
|
|
|
# [GCOV_COMMAND <gcov_command>]
|
|
|
|
# [GCOV_OPTIONS <options>...]
|
|
|
|
# )
|
|
|
|
#
|
|
|
|
# Run gcov and package a tar file for CDash. The options are:
|
|
|
|
#
|
|
|
|
# ``TARBALL <tarfile>``
|
|
|
|
# Specify the location of the ``.tar`` file to be created for later
|
|
|
|
# upload to CDash. Relative paths will be interpreted with respect
|
|
|
|
# to the top-level build directory.
|
|
|
|
#
|
|
|
|
# ``SOURCE <source_dir>``
|
|
|
|
# Specify the top-level source directory for the build.
|
|
|
|
# Default is the value of :variable:`CTEST_SOURCE_DIRECTORY`.
|
|
|
|
#
|
|
|
|
# ``BUILD <build_dir>``
|
|
|
|
# Specify the top-level build directory for the build.
|
|
|
|
# Default is the value of :variable:`CTEST_BINARY_DIRECTORY`.
|
|
|
|
#
|
|
|
|
# ``GCOV_COMMAND <gcov_command>``
|
|
|
|
# Specify the full path to the ``gcov`` command on the machine.
|
|
|
|
# Default is the value of :variable:`CTEST_COVERAGE_COMMAND`.
|
|
|
|
#
|
|
|
|
# ``GCOV_OPTIONS <options>...``
|
|
|
|
# Specify options to be passed to gcov. The ``gcov`` command
|
|
|
|
# is run as ``gcov <options>... -o <gcov-dir> <file>.gcda``.
|
|
|
|
# If not specified, the default option is just ``-b``.
|
|
|
|
#
|
|
|
|
# ``GLOB``
|
|
|
|
# Recursively search for .gcda files in build_dir rather than
|
|
|
|
# determining search locations by reading TargetDirectories.txt.
|
|
|
|
#
|
|
|
|
# ``DELETE``
|
|
|
|
# Delete coverage files after they've been packaged into the .tar.
|
|
|
|
#
|
|
|
|
# ``QUIET``
|
|
|
|
# Suppress non-error messages that otherwise would have been
|
|
|
|
# printed out by this function.
|
|
|
|
|
|
|
|
include(CMakeParseArguments)
|
|
|
|
function(ctest_coverage_collect_gcov)
|
|
|
|
set(options QUIET GLOB DELETE)
|
|
|
|
set(oneValueArgs TARBALL SOURCE BUILD GCOV_COMMAND)
|
|
|
|
set(multiValueArgs GCOV_OPTIONS)
|
|
|
|
cmake_parse_arguments(GCOV "${options}" "${oneValueArgs}"
|
|
|
|
"${multiValueArgs}" "" ${ARGN} )
|
|
|
|
if(NOT DEFINED GCOV_TARBALL)
|
|
|
|
message(FATAL_ERROR
|
|
|
|
"TARBALL must be specified. for ctest_coverage_collect_gcov")
|
|
|
|
endif()
|
|
|
|
if(NOT DEFINED GCOV_SOURCE)
|
|
|
|
set(source_dir "${CTEST_SOURCE_DIRECTORY}")
|
|
|
|
else()
|
|
|
|
set(source_dir "${GCOV_SOURCE}")
|
|
|
|
endif()
|
|
|
|
if(NOT DEFINED GCOV_BUILD)
|
|
|
|
set(binary_dir "${CTEST_BINARY_DIRECTORY}")
|
|
|
|
else()
|
|
|
|
set(binary_dir "${GCOV_BUILD}")
|
|
|
|
endif()
|
|
|
|
if(NOT DEFINED GCOV_GCOV_COMMAND)
|
|
|
|
set(gcov_command "${CTEST_COVERAGE_COMMAND}")
|
|
|
|
else()
|
|
|
|
set(gcov_command "${GCOV_GCOV_COMMAND}")
|
|
|
|
endif()
|
|
|
|
# run gcov on each gcda file in the binary tree
|
|
|
|
set(gcda_files)
|
|
|
|
set(label_files)
|
|
|
|
if (GCOV_GLOB)
|
|
|
|
file(GLOB_RECURSE gfiles RELATIVE ${binary_dir} "${binary_dir}/*.gcda")
|
|
|
|
list(LENGTH gfiles len)
|
|
|
|
# if we have gcda files then also grab the labels file for that target
|
|
|
|
if(${len} GREATER 0)
|
|
|
|
file(GLOB_RECURSE lfiles RELATIVE ${binary_dir} "${binary_dir}/Labels.json")
|
|
|
|
list(APPEND gcda_files ${gfiles})
|
|
|
|
list(APPEND label_files ${lfiles})
|
|
|
|
endif()
|
|
|
|
else()
|
|
|
|
# look for gcda files in the target directories
|
|
|
|
# this will be faster and only look where the files will be
|
|
|
|
file(STRINGS "${binary_dir}/CMakeFiles/TargetDirectories.txt" target_dirs
|
|
|
|
ENCODING UTF-8)
|
|
|
|
foreach(target_dir ${target_dirs})
|
|
|
|
file(GLOB_RECURSE gfiles RELATIVE ${binary_dir} "${target_dir}/*.gcda")
|
|
|
|
list(LENGTH gfiles len)
|
|
|
|
# if we have gcda files then also grab the labels file for that target
|
|
|
|
if(${len} GREATER 0)
|
|
|
|
file(GLOB_RECURSE lfiles RELATIVE ${binary_dir}
|
|
|
|
"${target_dir}/Labels.json")
|
|
|
|
list(APPEND gcda_files ${gfiles})
|
|
|
|
list(APPEND label_files ${lfiles})
|
|
|
|
endif()
|
|
|
|
endforeach()
|
|
|
|
endif()
|
|
|
|
# return early if no coverage files were found
|
|
|
|
list(LENGTH gcda_files len)
|
|
|
|
if(len EQUAL 0)
|
|
|
|
if (NOT GCOV_QUIET)
|
|
|
|
message("ctest_coverage_collect_gcov: No .gcda files found, "
|
|
|
|
"ignoring coverage request.")
|
|
|
|
endif()
|
|
|
|
return()
|
|
|
|
endif()
|
|
|
|
# setup the dir for the coverage files
|
|
|
|
set(coverage_dir "${binary_dir}/Testing/CoverageInfo")
|
|
|
|
file(MAKE_DIRECTORY "${coverage_dir}")
|
|
|
|
# call gcov on each .gcda file
|
|
|
|
foreach (gcda_file ${gcda_files})
|
|
|
|
# get the directory of the gcda file
|
|
|
|
get_filename_component(gcda_file ${binary_dir}/${gcda_file} ABSOLUTE)
|
|
|
|
get_filename_component(gcov_dir ${gcda_file} DIRECTORY)
|
|
|
|
# run gcov, this will produce the .gcov file in the current
|
|
|
|
# working directory
|
|
|
|
if(NOT DEFINED GCOV_GCOV_OPTIONS)
|
|
|
|
set(GCOV_GCOV_OPTIONS -b)
|
|
|
|
endif()
|
|
|
|
execute_process(COMMAND
|
|
|
|
${gcov_command} ${GCOV_GCOV_OPTIONS} -o ${gcov_dir} ${gcda_file}
|
|
|
|
OUTPUT_VARIABLE out
|
|
|
|
RESULT_VARIABLE res
|
|
|
|
WORKING_DIRECTORY ${coverage_dir})
|
|
|
|
|
|
|
|
if (GCOV_DELETE)
|
|
|
|
file(REMOVE ${gcda_file})
|
|
|
|
endif()
|
|
|
|
|
|
|
|
endforeach()
|
|
|
|
if(NOT "${res}" EQUAL 0)
|
|
|
|
if (NOT GCOV_QUIET)
|
|
|
|
message(STATUS "Error running gcov: ${res} ${out}")
|
|
|
|
endif()
|
|
|
|
endif()
|
|
|
|
# create json file with project information
|
|
|
|
file(WRITE ${coverage_dir}/data.json
|
|
|
|
"{
|
|
|
|
\"Source\": \"${source_dir}\",
|
|
|
|
\"Binary\": \"${binary_dir}\"
|
|
|
|
}")
|
|
|
|
# collect the gcov files
|
|
|
|
set(unfiltered_gcov_files)
|
|
|
|
file(GLOB_RECURSE unfiltered_gcov_files RELATIVE ${binary_dir} "${coverage_dir}/*.gcov")
|
|
|
|
|
|
|
|
# if CTEST_EXTRA_COVERAGE_GLOB was specified we search for files
|
|
|
|
# that might be uncovered
|
|
|
|
if (DEFINED CTEST_EXTRA_COVERAGE_GLOB)
|
|
|
|
set(uncovered_files)
|
|
|
|
foreach(search_entry IN LISTS CTEST_EXTRA_COVERAGE_GLOB)
|
|
|
|
if(NOT GCOV_QUIET)
|
|
|
|
message("Add coverage glob: ${search_entry}")
|
|
|
|
endif()
|
|
|
|
file(GLOB_RECURSE matching_files "${source_dir}/${search_entry}")
|
|
|
|
if (matching_files)
|
|
|
|
list(APPEND uncovered_files "${matching_files}")
|
|
|
|
endif()
|
|
|
|
endforeach()
|
|
|
|
endif()
|
|
|
|
|
|
|
|
set(gcov_files)
|
|
|
|
foreach(gcov_file ${unfiltered_gcov_files})
|
|
|
|
file(STRINGS ${binary_dir}/${gcov_file} first_line LIMIT_COUNT 1 ENCODING UTF-8)
|
|
|
|
|
|
|
|
set(is_excluded false)
|
|
|
|
if(first_line MATCHES "^ -: 0:Source:(.*)$")
|
|
|
|
set(source_file ${CMAKE_MATCH_1})
|
|
|
|
elseif(NOT GCOV_QUIET)
|
|
|
|
message(STATUS "Could not determine source file corresponding to: ${gcov_file}")
|
|
|
|
endif()
|
|
|
|
|
|
|
|
foreach(exclude_entry IN LISTS CTEST_CUSTOM_COVERAGE_EXCLUDE)
|
|
|
|
if(source_file MATCHES "${exclude_entry}")
|
|
|
|
set(is_excluded true)
|
|
|
|
|
|
|
|
if(NOT GCOV_QUIET)
|
|
|
|
message("Excluding coverage for: ${source_file} which matches ${exclude_entry}")
|
|
|
|
endif()
|
|
|
|
|
|
|
|
break()
|
|
|
|
endif()
|
|
|
|
endforeach()
|
|
|
|
|
|
|
|
get_filename_component(resolved_source_file "${source_file}" ABSOLUTE)
|
|
|
|
foreach(uncovered_file IN LISTS uncovered_files)
|
|
|
|
get_filename_component(resolved_uncovered_file "${uncovered_file}" ABSOLUTE)
|
|
|
|
if (resolved_uncovered_file STREQUAL resolved_source_file)
|
|
|
|
list(REMOVE_ITEM uncovered_files "${uncovered_file}")
|
|
|
|
endif()
|
|
|
|
endforeach()
|
|
|
|
|
|
|
|
if(NOT is_excluded)
|
|
|
|
list(APPEND gcov_files ${gcov_file})
|
|
|
|
endif()
|
|
|
|
endforeach()
|
|
|
|
|
|
|
|
foreach (uncovered_file ${uncovered_files})
|
|
|
|
# Check if this uncovered file should be excluded.
|
|
|
|
set(is_excluded false)
|
|
|
|
foreach(exclude_entry IN LISTS CTEST_CUSTOM_COVERAGE_EXCLUDE)
|
|
|
|
if(uncovered_file MATCHES "${exclude_entry}")
|
|
|
|
set(is_excluded true)
|
|
|
|
if(NOT GCOV_QUIET)
|
|
|
|
message("Excluding coverage for: ${uncovered_file} which matches ${exclude_entry}")
|
|
|
|
endif()
|
|
|
|
break()
|
|
|
|
endif()
|
|
|
|
endforeach()
|
|
|
|
if(is_excluded)
|
|
|
|
continue()
|
|
|
|
endif()
|
|
|
|
|
|
|
|
# Copy from source to binary dir, preserving any intermediate subdirectories.
|
|
|
|
get_filename_component(filename "${uncovered_file}" NAME)
|
|
|
|
get_filename_component(relative_path "${uncovered_file}" DIRECTORY)
|
|
|
|
string(REPLACE "${source_dir}" "" relative_path "${relative_path}")
|
|
|
|
if (relative_path)
|
|
|
|
# Strip leading slash.
|
|
|
|
string(SUBSTRING "${relative_path}" 1 -1 relative_path)
|
|
|
|
endif()
|
|
|
|
file(COPY ${uncovered_file} DESTINATION ${binary_dir}/uncovered/${relative_path})
|
|
|
|
if(relative_path)
|
|
|
|
list(APPEND uncovered_files_for_tar uncovered/${relative_path}/${filename})
|
|
|
|
else()
|
|
|
|
list(APPEND uncovered_files_for_tar uncovered/${filename})
|
|
|
|
endif()
|
|
|
|
endforeach()
|
|
|
|
|
|
|
|
# tar up the coverage info with the same date so that the md5
|
|
|
|
# sum will be the same for the tar file independent of file time
|
|
|
|
# stamps
|
|
|
|
string(REPLACE ";" "\n" gcov_files "${gcov_files}")
|
|
|
|
string(REPLACE ";" "\n" label_files "${label_files}")
|
|
|
|
string(REPLACE ";" "\n" uncovered_files_for_tar "${uncovered_files_for_tar}")
|
|
|
|
file(WRITE "${coverage_dir}/coverage_file_list.txt"
|
|
|
|
"${gcov_files}
|
|
|
|
${coverage_dir}/data.json
|
|
|
|
${label_files}
|
|
|
|
${uncovered_files_for_tar}
|
|
|
|
")
|
|
|
|
|
|
|
|
if (GCOV_QUIET)
|
|
|
|
set(tar_opts "cfj")
|
|
|
|
else()
|
|
|
|
set(tar_opts "cvfj")
|
|
|
|
endif()
|
|
|
|
|
|
|
|
execute_process(COMMAND
|
|
|
|
${CMAKE_COMMAND} -E tar ${tar_opts} ${GCOV_TARBALL}
|
|
|
|
"--mtime=1970-01-01 0:0:0 UTC"
|
|
|
|
"--format=gnutar"
|
|
|
|
--files-from=${coverage_dir}/coverage_file_list.txt
|
|
|
|
WORKING_DIRECTORY ${binary_dir})
|
|
|
|
|
|
|
|
if (GCOV_DELETE)
|
|
|
|
foreach(gcov_file ${unfiltered_gcov_files})
|
|
|
|
file(REMOVE ${binary_dir}/${gcov_file})
|
|
|
|
endforeach()
|
|
|
|
file(REMOVE ${coverage_dir}/coverage_file_list.txt)
|
|
|
|
file(REMOVE ${coverage_dir}/data.json)
|
|
|
|
if (EXISTS ${binary_dir}/uncovered)
|
|
|
|
file(REMOVE ${binary_dir}/uncovered)
|
|
|
|
endif()
|
|
|
|
endif()
|
|
|
|
|
|
|
|
endfunction()
|