mirror of
				https://github.com/catchorg/Catch2.git
				synced 2025-11-04 05:59:32 +01:00 
			
		
		
		
	* Every Linux build tracks coverage when running Debug mode * OS X not supported yet (Future WIP) * Our own unit tests, non-default reporters and Clara are ignored
		
			
				
	
	
		
			355 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			CMake
		
	
	
	
	
	
			
		
		
	
	
			355 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			CMake
		
	
	
	
	
	
# This file is part of CMake-codecov.
 | 
						|
#
 | 
						|
# Copyright (c)
 | 
						|
#   2015-2017 RWTH Aachen University, Federal Republic of Germany
 | 
						|
#
 | 
						|
# See the LICENSE file in the package base directory for details
 | 
						|
#
 | 
						|
# Written by Alexander Haase, alexander.haase@rwth-aachen.de
 | 
						|
#
 | 
						|
 | 
						|
 | 
						|
# configuration
 | 
						|
set(LCOV_DATA_PATH "${CMAKE_BINARY_DIR}/lcov/data")
 | 
						|
set(LCOV_DATA_PATH_INIT "${LCOV_DATA_PATH}/init")
 | 
						|
set(LCOV_DATA_PATH_CAPTURE "${LCOV_DATA_PATH}/capture")
 | 
						|
set(LCOV_HTML_PATH "${CMAKE_BINARY_DIR}/lcov/html")
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
# Search for Gcov which is used by Lcov.
 | 
						|
find_package(Gcov)
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
# This function will add lcov evaluation for target <TNAME>. Only sources of
 | 
						|
# this target will be evaluated and no dependencies will be added. It will call
 | 
						|
# geninfo on any source file of <TNAME> once and store the info file in the same
 | 
						|
# directory.
 | 
						|
#
 | 
						|
# Note: This function is only a wrapper to define this function always, even if
 | 
						|
#   coverage is not supported by the compiler or disabled. This function must
 | 
						|
#   be defined here, because the module will be exited, if there is no coverage
 | 
						|
#   support by the compiler or it is disabled by the user.
 | 
						|
function (add_lcov_target TNAME)
 | 
						|
	if (LCOV_FOUND)
 | 
						|
		# capture initial coverage data
 | 
						|
		lcov_capture_initial_tgt(${TNAME})
 | 
						|
 | 
						|
		# capture coverage data after execution
 | 
						|
		lcov_capture_tgt(${TNAME})
 | 
						|
	endif ()
 | 
						|
endfunction (add_lcov_target)
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
# include required Modules
 | 
						|
include(FindPackageHandleStandardArgs)
 | 
						|
 | 
						|
# Search for required lcov binaries.
 | 
						|
find_program(LCOV_BIN lcov)
 | 
						|
find_program(GENINFO_BIN geninfo)
 | 
						|
find_program(GENHTML_BIN genhtml)
 | 
						|
find_package_handle_standard_args(lcov
 | 
						|
	REQUIRED_VARS LCOV_BIN GENINFO_BIN GENHTML_BIN
 | 
						|
)
 | 
						|
 | 
						|
# enable genhtml C++ demangeling, if c++filt is found.
 | 
						|
set(GENHTML_CPPFILT_FLAG "")
 | 
						|
find_program(CPPFILT_BIN c++filt)
 | 
						|
if (NOT CPPFILT_BIN STREQUAL "")
 | 
						|
	set(GENHTML_CPPFILT_FLAG "--demangle-cpp")
 | 
						|
endif (NOT CPPFILT_BIN STREQUAL "")
 | 
						|
 | 
						|
# enable no-external flag for lcov, if available.
 | 
						|
if (GENINFO_BIN AND NOT DEFINED GENINFO_EXTERN_FLAG)
 | 
						|
	set(FLAG "")
 | 
						|
	execute_process(COMMAND ${GENINFO_BIN} --help OUTPUT_VARIABLE GENINFO_HELP)
 | 
						|
	string(REGEX MATCH "external" GENINFO_RES "${GENINFO_HELP}")
 | 
						|
	if (GENINFO_RES)
 | 
						|
		set(FLAG "--no-external")
 | 
						|
	endif ()
 | 
						|
 | 
						|
	set(GENINFO_EXTERN_FLAG "${FLAG}"
 | 
						|
		CACHE STRING "Geninfo flag to exclude system sources.")
 | 
						|
endif ()
 | 
						|
 | 
						|
# If Lcov was not found, exit module now.
 | 
						|
if (NOT LCOV_FOUND)
 | 
						|
	return()
 | 
						|
endif (NOT LCOV_FOUND)
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
# Create directories to be used.
 | 
						|
file(MAKE_DIRECTORY ${LCOV_DATA_PATH_INIT})
 | 
						|
file(MAKE_DIRECTORY ${LCOV_DATA_PATH_CAPTURE})
 | 
						|
 | 
						|
set(LCOV_REMOVE_PATTERNS "")
 | 
						|
 | 
						|
# This function will merge lcov files to a single target file. Additional lcov
 | 
						|
# flags may be set with setting LCOV_EXTRA_FLAGS before calling this function.
 | 
						|
function (lcov_merge_files OUTFILE ...)
 | 
						|
	# Remove ${OUTFILE} from ${ARGV} and generate lcov parameters with files.
 | 
						|
	list(REMOVE_AT ARGV 0)
 | 
						|
 | 
						|
	# Generate merged file.
 | 
						|
	string(REPLACE "${CMAKE_BINARY_DIR}/" "" FILE_REL "${OUTFILE}")
 | 
						|
	add_custom_command(OUTPUT "${OUTFILE}.raw"
 | 
						|
		COMMAND cat ${ARGV} > ${OUTFILE}.raw
 | 
						|
		DEPENDS ${ARGV}
 | 
						|
		COMMENT "Generating ${FILE_REL}"
 | 
						|
	)
 | 
						|
 | 
						|
	add_custom_command(OUTPUT "${OUTFILE}"
 | 
						|
		COMMAND ${LCOV_BIN} --quiet -a ${OUTFILE}.raw --output-file ${OUTFILE}
 | 
						|
			--base-directory ${PROJECT_SOURCE_DIR} ${LCOV_EXTRA_FLAGS}
 | 
						|
		COMMAND ${LCOV_BIN} --quiet -r ${OUTFILE} ${LCOV_REMOVE_PATTERNS}
 | 
						|
			--output-file ${OUTFILE} ${LCOV_EXTRA_FLAGS}
 | 
						|
		DEPENDS ${OUTFILE}.raw
 | 
						|
		COMMENT "Post-processing ${FILE_REL}"
 | 
						|
	)
 | 
						|
endfunction ()
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
# Add a new global target to generate initial coverage reports for all targets.
 | 
						|
# This target will be used to generate the global initial info file, which is
 | 
						|
# used to gather even empty report data.
 | 
						|
if (NOT TARGET lcov-capture-init)
 | 
						|
	add_custom_target(lcov-capture-init)
 | 
						|
	set(LCOV_CAPTURE_INIT_FILES "" CACHE INTERNAL "")
 | 
						|
endif (NOT TARGET lcov-capture-init)
 | 
						|
 | 
						|
 | 
						|
# This function will add initial capture of coverage data for target <TNAME>,
 | 
						|
# which is needed to get also data for objects, which were not loaded at
 | 
						|
# execution time. It will call geninfo for every source file of <TNAME> once and
 | 
						|
# store the info file in the same directory.
 | 
						|
function (lcov_capture_initial_tgt TNAME)
 | 
						|
	# We don't have to check, if the target has support for coverage, thus this
 | 
						|
	# will be checked by add_coverage_target in Findcoverage.cmake. Instead we
 | 
						|
	# have to determine which gcov binary to use.
 | 
						|
	get_target_property(TSOURCES ${TNAME} SOURCES)
 | 
						|
	set(SOURCES "")
 | 
						|
	set(TCOMPILER "")
 | 
						|
	foreach (FILE ${TSOURCES})
 | 
						|
		codecov_path_of_source(${FILE} FILE)
 | 
						|
		if (NOT "${FILE}" STREQUAL "")
 | 
						|
			codecov_lang_of_source(${FILE} LANG)
 | 
						|
			if (NOT "${LANG}" STREQUAL "")
 | 
						|
				list(APPEND SOURCES "${FILE}")
 | 
						|
				set(TCOMPILER ${CMAKE_${LANG}_COMPILER_ID})
 | 
						|
			endif ()
 | 
						|
		endif ()
 | 
						|
	endforeach ()
 | 
						|
 | 
						|
	# If no gcov binary was found, coverage data can't be evaluated.
 | 
						|
	if (NOT GCOV_${TCOMPILER}_BIN)
 | 
						|
		message(WARNING "No coverage evaluation binary found for ${TCOMPILER}.")
 | 
						|
		return()
 | 
						|
	endif ()
 | 
						|
 | 
						|
	set(GCOV_BIN "${GCOV_${TCOMPILER}_BIN}")
 | 
						|
	set(GCOV_ENV "${GCOV_${TCOMPILER}_ENV}")
 | 
						|
 | 
						|
 | 
						|
	set(TDIR ${CMAKE_CURRENT_BINARY_DIR}/CMakeFiles/${TNAME}.dir)
 | 
						|
	set(GENINFO_FILES "")
 | 
						|
	foreach(FILE ${SOURCES})
 | 
						|
		# generate empty coverage files
 | 
						|
		set(OUTFILE "${TDIR}/${FILE}.info.init")
 | 
						|
		list(APPEND GENINFO_FILES ${OUTFILE})
 | 
						|
 | 
						|
		add_custom_command(OUTPUT ${OUTFILE} COMMAND ${GCOV_ENV} ${GENINFO_BIN}
 | 
						|
				--quiet --base-directory ${PROJECT_SOURCE_DIR} --initial
 | 
						|
				--gcov-tool ${GCOV_BIN} --output-filename ${OUTFILE}
 | 
						|
				${GENINFO_EXTERN_FLAG} ${TDIR}/${FILE}.gcno
 | 
						|
			DEPENDS ${TNAME}
 | 
						|
			COMMENT "Capturing initial coverage data for ${FILE}"
 | 
						|
		)
 | 
						|
	endforeach()
 | 
						|
 | 
						|
	# Concatenate all files generated by geninfo to a single file per target.
 | 
						|
	set(OUTFILE "${LCOV_DATA_PATH_INIT}/${TNAME}.info")
 | 
						|
	set(LCOV_EXTRA_FLAGS "--initial")
 | 
						|
	lcov_merge_files("${OUTFILE}" ${GENINFO_FILES})
 | 
						|
	add_custom_target(${TNAME}-capture-init ALL DEPENDS ${OUTFILE})
 | 
						|
 | 
						|
	# add geninfo file generation to global lcov-geninfo target
 | 
						|
	add_dependencies(lcov-capture-init ${TNAME}-capture-init)
 | 
						|
	set(LCOV_CAPTURE_INIT_FILES "${LCOV_CAPTURE_INIT_FILES}"
 | 
						|
		"${OUTFILE}" CACHE INTERNAL ""
 | 
						|
	)
 | 
						|
endfunction (lcov_capture_initial_tgt)
 | 
						|
 | 
						|
 | 
						|
# This function will generate the global info file for all targets. It has to be
 | 
						|
# called after all other CMake functions in the root CMakeLists.txt file, to get
 | 
						|
# a full list of all targets that generate coverage data.
 | 
						|
function (lcov_capture_initial)
 | 
						|
	# Skip this function (and do not create the following targets), if there are
 | 
						|
	# no input files.
 | 
						|
	if ("${LCOV_CAPTURE_INIT_FILES}" STREQUAL "")
 | 
						|
		return()
 | 
						|
	endif ()
 | 
						|
 | 
						|
	# Add a new target to merge the files of all targets.
 | 
						|
	set(OUTFILE "${LCOV_DATA_PATH_INIT}/all_targets.info")
 | 
						|
	lcov_merge_files("${OUTFILE}" ${LCOV_CAPTURE_INIT_FILES})
 | 
						|
	add_custom_target(lcov-geninfo-init ALL	DEPENDS ${OUTFILE}
 | 
						|
		lcov-capture-init
 | 
						|
	)
 | 
						|
endfunction (lcov_capture_initial)
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
# Add a new global target to generate coverage reports for all targets. This
 | 
						|
# target will be used to generate the global info file.
 | 
						|
if (NOT TARGET lcov-capture)
 | 
						|
	add_custom_target(lcov-capture)
 | 
						|
	set(LCOV_CAPTURE_FILES "" CACHE INTERNAL "")
 | 
						|
endif (NOT TARGET lcov-capture)
 | 
						|
 | 
						|
 | 
						|
# This function will add capture of coverage data for target <TNAME>, which is
 | 
						|
# needed to get also data for objects, which were not loaded at execution time.
 | 
						|
# It will call geninfo for every source file of <TNAME> once and store the info
 | 
						|
# file in the same directory.
 | 
						|
function (lcov_capture_tgt TNAME)
 | 
						|
	# We don't have to check, if the target has support for coverage, thus this
 | 
						|
	# will be checked by add_coverage_target in Findcoverage.cmake. Instead we
 | 
						|
	# have to determine which gcov binary to use.
 | 
						|
	get_target_property(TSOURCES ${TNAME} SOURCES)
 | 
						|
	set(SOURCES "")
 | 
						|
	set(TCOMPILER "")
 | 
						|
	foreach (FILE ${TSOURCES})
 | 
						|
		codecov_path_of_source(${FILE} FILE)
 | 
						|
		if (NOT "${FILE}" STREQUAL "")
 | 
						|
			codecov_lang_of_source(${FILE} LANG)
 | 
						|
			if (NOT "${LANG}" STREQUAL "")
 | 
						|
				list(APPEND SOURCES "${FILE}")
 | 
						|
				set(TCOMPILER ${CMAKE_${LANG}_COMPILER_ID})
 | 
						|
			endif ()
 | 
						|
		endif ()
 | 
						|
	endforeach ()
 | 
						|
 | 
						|
	# If no gcov binary was found, coverage data can't be evaluated.
 | 
						|
	if (NOT GCOV_${TCOMPILER}_BIN)
 | 
						|
		message(WARNING "No coverage evaluation binary found for ${TCOMPILER}.")
 | 
						|
		return()
 | 
						|
	endif ()
 | 
						|
 | 
						|
	set(GCOV_BIN "${GCOV_${TCOMPILER}_BIN}")
 | 
						|
	set(GCOV_ENV "${GCOV_${TCOMPILER}_ENV}")
 | 
						|
 | 
						|
 | 
						|
	set(TDIR ${CMAKE_CURRENT_BINARY_DIR}/CMakeFiles/${TNAME}.dir)
 | 
						|
	set(GENINFO_FILES "")
 | 
						|
	foreach(FILE ${SOURCES})
 | 
						|
		# Generate coverage files. If no .gcda file was generated during
 | 
						|
		# execution, the empty coverage file will be used instead.
 | 
						|
		set(OUTFILE "${TDIR}/${FILE}.info")
 | 
						|
		list(APPEND GENINFO_FILES ${OUTFILE})
 | 
						|
 | 
						|
		add_custom_command(OUTPUT ${OUTFILE}
 | 
						|
			COMMAND test -f "${TDIR}/${FILE}.gcda"
 | 
						|
				&& ${GCOV_ENV} ${GENINFO_BIN} --quiet --base-directory
 | 
						|
					${PROJECT_SOURCE_DIR} --gcov-tool ${GCOV_BIN}
 | 
						|
					--output-filename ${OUTFILE} ${GENINFO_EXTERN_FLAG}
 | 
						|
					${TDIR}/${FILE}.gcda
 | 
						|
				|| cp ${OUTFILE}.init ${OUTFILE}
 | 
						|
			DEPENDS ${TNAME} ${TNAME}-capture-init
 | 
						|
			COMMENT "Capturing coverage data for ${FILE}"
 | 
						|
		)
 | 
						|
	endforeach()
 | 
						|
 | 
						|
	# Concatenate all files generated by geninfo to a single file per target.
 | 
						|
	set(OUTFILE "${LCOV_DATA_PATH_CAPTURE}/${TNAME}.info")
 | 
						|
	lcov_merge_files("${OUTFILE}" ${GENINFO_FILES})
 | 
						|
	add_custom_target(${TNAME}-geninfo DEPENDS ${OUTFILE})
 | 
						|
 | 
						|
	# add geninfo file generation to global lcov-capture target
 | 
						|
	add_dependencies(lcov-capture ${TNAME}-geninfo)
 | 
						|
	set(LCOV_CAPTURE_FILES "${LCOV_CAPTURE_FILES}" "${OUTFILE}" CACHE INTERNAL
 | 
						|
		""
 | 
						|
	)
 | 
						|
 | 
						|
	# Add target for generating html output for this target only.
 | 
						|
	file(MAKE_DIRECTORY ${LCOV_HTML_PATH}/${TNAME})
 | 
						|
	add_custom_target(${TNAME}-genhtml
 | 
						|
		COMMAND ${GENHTML_BIN} --quiet --sort --prefix ${PROJECT_SOURCE_DIR}
 | 
						|
			--baseline-file ${LCOV_DATA_PATH_INIT}/${TNAME}.info
 | 
						|
			--output-directory ${LCOV_HTML_PATH}/${TNAME}
 | 
						|
			--title "${CMAKE_PROJECT_NAME} - target ${TNAME}"
 | 
						|
			${GENHTML_CPPFILT_FLAG} ${OUTFILE}
 | 
						|
		DEPENDS ${TNAME}-geninfo ${TNAME}-capture-init
 | 
						|
	)
 | 
						|
endfunction (lcov_capture_tgt)
 | 
						|
 | 
						|
 | 
						|
# This function will generate the global info file for all targets. It has to be
 | 
						|
# called after all other CMake functions in the root CMakeLists.txt file, to get
 | 
						|
# a full list of all targets that generate coverage data.
 | 
						|
function (lcov_capture)
 | 
						|
	# Skip this function (and do not create the following targets), if there are
 | 
						|
	# no input files.
 | 
						|
	if ("${LCOV_CAPTURE_FILES}" STREQUAL "")
 | 
						|
		return()
 | 
						|
	endif ()
 | 
						|
 | 
						|
	# Add a new target to merge the files of all targets.
 | 
						|
	set(OUTFILE "${LCOV_DATA_PATH_CAPTURE}/all_targets.info")
 | 
						|
	lcov_merge_files("${OUTFILE}" ${LCOV_CAPTURE_FILES})
 | 
						|
	add_custom_target(lcov-geninfo DEPENDS ${OUTFILE} lcov-capture)
 | 
						|
 | 
						|
	# Add a new global target for all lcov targets. This target could be used to
 | 
						|
	# generate the lcov html output for the whole project instead of calling
 | 
						|
	# <TARGET>-geninfo and <TARGET>-genhtml for each target. It will also be
 | 
						|
	# used to generate a html site for all project data together instead of one
 | 
						|
	# for each target.
 | 
						|
	if (NOT TARGET lcov)
 | 
						|
		file(MAKE_DIRECTORY ${LCOV_HTML_PATH}/all_targets)
 | 
						|
		add_custom_target(lcov
 | 
						|
			COMMAND ${GENHTML_BIN} --quiet --sort
 | 
						|
				--baseline-file ${LCOV_DATA_PATH_INIT}/all_targets.info
 | 
						|
				--output-directory ${LCOV_HTML_PATH}/all_targets
 | 
						|
				--title "${CMAKE_PROJECT_NAME}" --prefix "${PROJECT_SOURCE_DIR}"
 | 
						|
				${GENHTML_CPPFILT_FLAG} ${OUTFILE}
 | 
						|
			DEPENDS lcov-geninfo-init lcov-geninfo
 | 
						|
		)
 | 
						|
	endif ()
 | 
						|
endfunction (lcov_capture)
 | 
						|
 | 
						|
 | 
						|
 | 
						|
 | 
						|
# Add a new global target to generate the lcov html report for the whole project
 | 
						|
# instead of calling <TARGET>-genhtml for each target (to create an own report
 | 
						|
# for each target). Instead of the lcov target it does not require geninfo for
 | 
						|
# all targets, so you have to call <TARGET>-geninfo to generate the info files
 | 
						|
# the targets you'd like to have in your report or lcov-geninfo for generating
 | 
						|
# info files for all targets before calling lcov-genhtml.
 | 
						|
file(MAKE_DIRECTORY ${LCOV_HTML_PATH}/selected_targets)
 | 
						|
if (NOT TARGET lcov-genhtml)
 | 
						|
	add_custom_target(lcov-genhtml
 | 
						|
		COMMAND ${GENHTML_BIN}
 | 
						|
			--quiet
 | 
						|
			--output-directory ${LCOV_HTML_PATH}/selected_targets
 | 
						|
			--title \"${CMAKE_PROJECT_NAME} - targets  `find
 | 
						|
				${LCOV_DATA_PATH_CAPTURE} -name \"*.info\" ! -name
 | 
						|
				\"all_targets.info\" -exec basename {} .info \\\;`\"
 | 
						|
			--prefix ${PROJECT_SOURCE_DIR}
 | 
						|
			--sort
 | 
						|
			${GENHTML_CPPFILT_FLAG}
 | 
						|
			`find ${LCOV_DATA_PATH_CAPTURE} -name \"*.info\" ! -name
 | 
						|
				\"all_targets.info\"`
 | 
						|
	)
 | 
						|
endif (NOT TARGET lcov-genhtml)
 |