Skip to content

Commit

Permalink
ci: add code coverage
Browse files Browse the repository at this point in the history
  • Loading branch information
jolars committed Dec 13, 2023
1 parent 7b17034 commit 6c01a86
Show file tree
Hide file tree
Showing 5 changed files with 768 additions and 1 deletion.
7 changes: 6 additions & 1 deletion .github/workflows/ci.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -20,7 +20,7 @@ jobs:
cmake-version: "3.25"

- name: Configure CMake
run: cmake -B build -S . -DBUILD_TESTING=ON -DBUILD_DOCS=OFF
run: cmake -B build -S . -DBUILD_TESTING=ON -DBUILD_DOCS=OFF -DENABLE-COVERAGE

- name: Build
run: cmake --build build
Expand All @@ -30,6 +30,11 @@ jobs:

- name: Test
run: ctest --test-dir build --output-on-failure

- name: Upload coverage reports to Codecov
uses: codecov/codecov-action@v4-beta
env:
CODECOV_TOKEN: ${{ secrets.CODECOV_ORG_TOKEN }}
docs:
needs: build-and-test
runs-on: ubuntu-latest
Expand Down
7 changes: 7 additions & 0 deletions CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@ project(
VERSION ${project_version})

option(BUILD_DOCS "Build documentation" OFF)
option(ENABLE_COVERAGE "Generate coverage report" OFF)

include(FetchContent)
include(CTest)
Expand Down Expand Up @@ -43,6 +44,12 @@ if(BUILD_TESTING)
Eigen3::Eigen)
target_include_directories(tests PUBLIC tests/ "${PROJECT_BINARY_DIR}")

if(ENABLE_COVERAGE)
find_package(codecov)
add_coverage(tests)
coverage_evaluate()
endif()

list(APPEND CMAKE_MODULE_PATH ${catch2_SOURCE_DIR}/extras)
include(Catch)
catch_discover_tests(tests)
Expand Down
157 changes: 157 additions & 0 deletions cmake/FindGcov.cmake
Original file line number Diff line number Diff line change
@@ -0,0 +1,157 @@
# This file is part of CMake-codecov.
#
# Copyright (c) 2015-2020 RWTH Aachen University, Federal Republic of Germany
#
# See the LICENSE file in the package base directory for details
#
# Written by Alexander Haase, [email protected]
#

# include required Modules
include(FindPackageHandleStandardArgs)

# Search for gcov binary.
set(CMAKE_REQUIRED_QUIET_SAVE ${CMAKE_REQUIRED_QUIET})
set(CMAKE_REQUIRED_QUIET ${codecov_FIND_QUIETLY})

get_property(ENABLED_LANGUAGES GLOBAL PROPERTY ENABLED_LANGUAGES)
foreach(LANG ${ENABLED_LANGUAGES})
# Gcov evaluation is dependent on the used compiler. Check gcov support for
# each compiler that is used. If gcov binary was already found for this
# compiler, do not try to find it again.
if(NOT GCOV_${CMAKE_${LANG}_COMPILER_ID}_BIN)
get_filename_component(COMPILER_PATH "${CMAKE_${LANG}_COMPILER}" PATH)

if("${CMAKE_${LANG}_COMPILER_ID}" STREQUAL "GNU")
# Some distributions like OSX (homebrew) ship gcov with the compiler
# version appended as gcov-x. To find this binary we'll build the
# suggested binary name with the compiler version.
string(REGEX MATCH "^[0-9]+" GCC_VERSION
"${CMAKE_${LANG}_COMPILER_VERSION}")

find_program(
GCOV_BIN
NAMES gcov-${GCC_VERSION} gcov
HINTS ${COMPILER_PATH})

elseif("${CMAKE_${LANG}_COMPILER_ID}" MATCHES "^(Apple)?Clang$")
# Some distributions like Debian ship llvm-cov with the compiler version
# appended as llvm-cov-x.y. To find this binary we'll build the suggested
# binary name with the compiler version.
string(REGEX MATCH "^[0-9]+.[0-9]+" LLVM_VERSION
"${CMAKE_${LANG}_COMPILER_VERSION}")

# llvm-cov prior version 3.5 seems to be not working with coverage
# evaluation tools, but these versions are compatible with the gcc gcov
# tool.
if(LLVM_VERSION VERSION_GREATER 3.4)
find_program(
LLVM_COV_BIN
NAMES "llvm-cov-${LLVM_VERSION}" "llvm-cov"
HINTS ${COMPILER_PATH})
mark_as_advanced(LLVM_COV_BIN)

if(LLVM_COV_BIN)
find_program(LLVM_COV_WRAPPER "llvm-cov-wrapper"
PATHS ${CMAKE_MODULE_PATH})
if(LLVM_COV_WRAPPER)
set(GCOV_BIN
"${LLVM_COV_WRAPPER}"
CACHE FILEPATH "")

# set additional parameters
set(GCOV_${CMAKE_${LANG}_COMPILER_ID}_ENV
"LLVM_COV_BIN=${LLVM_COV_BIN}"
CACHE STRING "Environment variables for llvm-cov-wrapper.")
mark_as_advanced(GCOV_${CMAKE_${LANG}_COMPILER_ID}_ENV)
endif()
endif()
endif()

if(NOT GCOV_BIN)
# Fall back to gcov binary if llvm-cov was not found or is incompatible.
# This is the default on OSX, but may crash on recent Linux versions.
find_program(GCOV_BIN gcov HINTS ${COMPILER_PATH})
endif()
endif()

if(GCOV_BIN)
set(GCOV_${CMAKE_${LANG}_COMPILER_ID}_BIN
"${GCOV_BIN}"
CACHE STRING "${LANG} gcov binary.")

if(NOT CMAKE_REQUIRED_QUIET)
message("-- Found gcov evaluation for "
"${CMAKE_${LANG}_COMPILER_ID}: ${GCOV_BIN}")
endif()

unset(GCOV_BIN CACHE)
endif()
endif()
endforeach()

# Add a new global target for all gcov targets. This target could be used to
# generate the gcov files for the whole project instead of calling <TARGET>-gcov
# for each target.
if(NOT TARGET gcov)
add_custom_target(gcov)
endif(NOT TARGET gcov)

# This function will add gcov evaluation for target <TNAME>. Only sources of
# this target will be evaluated and no dependencies will be added. It will call
# Gcov on any source file of <TNAME> once and store the gcov file in the same
# directory.
function(add_gcov_target TNAME)
get_target_property(TBIN_DIR ${TNAME} BINARY_DIR)
set(TDIR ${TBIN_DIR}/CMakeFiles/${TNAME}.dir)

# 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(BUFFER "")
set(NULL_DEVICE "/dev/null")
if(WIN32)
set(NULL_DEVICE "NUL")
endif()
foreach(FILE ${SOURCES})
get_filename_component(FILE_PATH "${TDIR}/${FILE}" PATH)

# call gcov
add_custom_command(
OUTPUT ${TDIR}/${FILE}.gcov
COMMAND ${GCOV_ENV} ${GCOV_BIN} -p ${TDIR}/${FILE}.gcno > ${NULL_DEVICE}
DEPENDS ${TNAME} ${TDIR}/${FILE}.gcno
WORKING_DIRECTORY ${CMAKE_CURRENT_BINARY_DIR})

list(APPEND BUFFER ${TDIR}/${FILE}.gcov)
endforeach()

# add target for gcov evaluation of <TNAME>
add_custom_target(${TNAME}-gcov DEPENDS ${BUFFER})

# add evaluation target to the global gcov target.
add_dependencies(gcov ${TNAME}-gcov)
endfunction(add_gcov_target)
Loading

0 comments on commit 6c01a86

Please sign in to comment.