cmake/CodeCoverage.cmake (126 lines of code) (raw):
#from https://github.com/bilke/cmake-modules/blob/master/CodeCoverage.cmake
# - Enable Code Coverage
#
# 2012-01-31, Lars Bilke
#
# USAGE:
# 1. Copy this file into your cmake modules path
# 2. Add the following line to your CMakeLists.txt:
# INCLUDE(CodeCoverage)
#
# 3. Use the function SETUP_TARGET_FOR_COVERAGE to create a custom make target
# which runs your test executable and produces a lcov code coverage report.
#
# 4. Check is coverage available use CODE_COVERAGE_FOUND
#
# Check prereqs
find_package(Perl)
set(CODE_COVERAGE_FOUND "True")
FIND_PROGRAM( GCOV_PATH gcov )
if(WIN32)
if(NOT PERL_FOUND)
message(STATUS "Perl not found! Coverage report will be unavailable")
unset(CODE_COVERAGE_FOUND)
endif(NOT PERL_FOUND)
if(LCOV_ROOT)
find_file(LCOV_PATH lcov PATHS "${LCOV_ROOT}\\bin\\")
set(LCOV_PATH_NATIVE ${LCOV_PATH})
#string(REGEX REPLACE "[/]" "\\\\" LCOV_PATH_NATIVE ${LCOV_PATH})
find_file(GENHTML_PATH genhtml PATHS "${LCOV_ROOT}\\bin\\")
set(GENHTML_PATH_NATIVE ${GENHTML_PATH})
# string(REGEX REPLACE "[/]" "\\\\" GENHTML_PATH_NATIVE ${GENHTML_PATH})
else(LCOV_ROOT)
message(STATUS "LCOV_ROOT is not set! Coverage report will be unavailable")
endif(LCOV_ROOT)
else(WIN32)
FIND_PROGRAM( LCOV_PATH lcov )
FIND_PROGRAM( GENHTML_PATH genhtml )
endif(WIN32)
FIND_PROGRAM( GCOVR_PATH gcovr PATHS ${CMAKE_SOURCE_DIR}/tests)
IF(NOT GCOV_PATH)
MESSAGE(STATUS "gcov not found! Coverage report will be unavailable")
unset(CODE_COVERAGE_FOUND)
ENDIF() # NOT GCOV_PATH
IF(NOT CMAKE_COMPILER_IS_GNUCXX)
MESSAGE(STATUS "Compiler is not GNU gcc! Coverage report will be unavailable")
unset(CODE_COVERAGE_FOUND)
ENDIF() # NOT CMAKE_COMPILER_IS_GNUCXX
IF( NOT CMAKE_BUILD_TYPE STREQUAL "Debug" )
MESSAGE(STATUS "Code coverage results with an optimised (non-Debug) build may be misleading" )
ENDIF() # NOT CMAKE_BUILD_TYPE STREQUAL "Debug"
IF(CODE_COVERAGE_FOUND)
# Setup compiler options
ADD_DEFINITIONS(-fprofile-arcs -ftest-coverage)
LINK_LIBRARIES(gcov)
ENDIF()
# Param _targetname The name of new the custom make target
# Param _testrunner The name of the target which runs the tests
# Param _outputname lcov output is generated as _outputname.info
# HTML report is generated in _outputname/index.html
# Optional fourth parameter is passed as arguments to _testrunner
# Pass them in list form, e.g.: "-j;2" for -j 2
FUNCTION(SETUP_TARGET_FOR_COVERAGE _targetname _testrunner _outputname)
IF(NOT LCOV_PATH)
MESSAGE(WARNING "lcov not found! Coverage report will be unavailable")
ENDIF() # NOT LCOV_PATH
IF(NOT GENHTML_PATH)
MESSAGE(WARNING "genhtml not found! Coverage report will be unavailable")
ENDIF() # NOT GENHTML_PATH
IF(WIN32)
set(LCOV_COMMAND ${PERL_EXECUTABLE})
set(GENHTML_COMMAND ${PERL_EXECUTABLE})
set(LCOV_COMMAND_ARGS ${LCOV_PATH_NATIVE})
set(GENHTML_COMMAND_ARGS ${GENHTML_PATH_NATIVE})
ELSE(WIN32)
set(LCOV_COMMAND ${LCOV_PATH})
set(GENHTML_COMMAND ${GENHTML_PATH})
ENDIF(WIN32)
message(STATUS "LCOV_COMMAND is ${LCOV_COMMAND}")
# Setup target
ADD_CUSTOM_TARGET(${_targetname}
# Cleanup lcov
${LCOV_COMMAND} ${LCOV_COMMAND_ARGS} --directory . --zerocounters
# Run tests
COMMAND ${_testrunner} ${ARGV3}
# Capturing lcov counters and generating report
COMMAND ${LCOV_COMMAND} ${LCOV_COMMAND_ARGS} --directory . --capture --output-file ${_outputname}.info
COMMAND ${LCOV_COMMAND} ${LCOV_COMMAND_ARGS} --remove ${_outputname}.info 'tests\\*' --output-file ${_outputname}.info.cleaned
COMMAND ${GENHTML_COMMAND} ${GENHTML_COMMAND_ARGS} -o ${_outputname} ${_outputname}.info.cleaned
COMMAND ${CMAKE_COMMAND} -E remove ${_outputname}.info ${_outputname}.info.cleaned
WORKING_DIRECTORY ${CMAKE_BINARY_DIR}
COMMENT "Resetting code coverage counters to zero.\nProcessing code coverage counters and generating report."
)
# Show info where to find the report
ADD_CUSTOM_COMMAND(TARGET ${_targetname} POST_BUILD
COMMAND ;
COMMENT "Open ./${_outputname}/index.html in your browser to view the coverage report."
)
ENDFUNCTION() # SETUP_TARGET_FOR_COVERAGE
# Param _targetname The name of new the custom make target
# Param _testrunner The name of the target which runs the tests
# Param _outputname cobertura output is generated as _outputname.xml
# Optional fourth parameter is passed as arguments to _testrunner
# Pass them in list form, e.g.: "-j;2" for -j 2
FUNCTION(SETUP_TARGET_FOR_COVERAGE_COBERTURA _targetname _testrunner _outputname)
IF(NOT PYTHON_EXECUTABLE)
MESSAGE(WARNING "Python not found! Coverage report will be unavailable")
ENDIF() # NOT PYTHON_EXECUTABLE
IF(NOT GCOVR_PATH)
MESSAGE(WARNING "gcovr not found! Coverage report will be unavailable")
ENDIF() # NOT GCOVR_PATH
ADD_CUSTOM_TARGET(${_targetname}
# Run tests
${_testrunner} ${ARGV3}
# Running gcovr
COMMAND ${PYTHON_EXECUTABLE} ${GCOVR_PATH} -r ${CMAKE_SOURCE_DIR} -e '${CMAKE_SOURCE_DIR}/tests/'
COMMAND ${PYTHON_EXECUTABLE} ${GCOVR_PATH} --html -r ${CMAKE_SOURCE_DIR} -e '${CMAKE_SOURCE_DIR}/tests/' -o ${_outputname}.html
COMMAND ${PYTHON_EXECUTABLE} ${GCOVR_PATH} --xml -r ${CMAKE_SOURCE_DIR} -e '${CMAKE_SOURCE_DIR}/tests/' -o ${_outputname}.xml
WORKING_DIRECTORY ${CMAKE_BINARY_DIR}
COMMENT "Running gcovr to produce Cobertura code coverage report."
)
# Show info where to find the report
ADD_CUSTOM_COMMAND(TARGET ${_targetname} POST_BUILD
COMMAND ;
COMMENT "Cobertura code coverage report saved in '${_outputname}.html' and '${_outputname}.xml'"
)
ENDFUNCTION() # SETUP_TARGET_FOR_COVERAGE_COBERTURA