Imre Kis | e6bf9e6 | 2020-01-20 11:34:24 +0100 | [diff] [blame] | 1 | # |
| 2 | # Copyright (c) 2020, Arm Limited. All rights reserved. |
| 3 | # |
| 4 | # SPDX-License-Identifier: BSD-3-Clause |
| 5 | # |
| 6 | |
| 7 | include_guard(DIRECTORY) |
| 8 | |
| 9 | # Checking GCC |
| 10 | if (NOT CMAKE_CXX_COMPILER_ID STREQUAL "GNU") |
| 11 | message(FATAL_ERROR "Coverage measurement is only supported when using GCC") |
| 12 | endif() |
| 13 | |
| 14 | # Checking lcov |
| 15 | find_program(LCOV_COMMAND "lcov") |
| 16 | if (NOT LCOV_COMMAND) |
| 17 | message(FATAL_ERROR "Please install lcov") |
| 18 | endif() |
| 19 | |
| 20 | # Checking c-picker-coverage-mapper |
| 21 | find_program(CPICKER_COVERAGE_MAPPER_COMMAND "c-picker-coverage-mapper") |
| 22 | if (NOT CPICKER_COVERAGE_MAPPER_COMMAND) |
| 23 | message(FATAL_ERROR "Please install c-picker-coverage-mapper using pip (part of c-picker)") |
| 24 | endif() |
| 25 | |
| 26 | # Checking genhtml |
| 27 | find_program(GENHTML_COMMAND "genhtml") |
| 28 | if (NOT GENHTML_COMMAND) |
| 29 | message(FATAL_ERROR "Please install genhtml with genhtml (part of lcov)") |
| 30 | endif() |
| 31 | |
| 32 | # Including this file enables code coverage measurement by adding the necessary compiler and |
| 33 | # linker flags. |
| 34 | set(CMAKE_C_FLAGS "-g -O0 -fprofile-arcs -ftest-coverage") |
| 35 | set(CMAKE_CXX_FLAGS "-g -O0 -fprofile-arcs -ftest-coverage -fno-exceptions") |
| 36 | set(CMAKE_EXE_LINKER_FLAGS "-fprofile-arcs -ftest-coverage") |
| 37 | |
| 38 | # Adding custom targets |
| 39 | add_custom_target(coverage) |
| 40 | add_custom_target(coverage_report) |
| 41 | |
Imre Kis | c73346f | 2020-02-07 15:26:27 +0100 | [diff] [blame^] | 42 | # Adds a file to the dependency list of the target by inserting an accessory |
| 43 | # custom target. The name of the custom target is properly escaped. |
| 44 | function(add_coverage_dependency) |
| 45 | set(_OPTIONS_ARGS) |
| 46 | set(_ONE_VALUE_ARGS TARGET DEPENDS) |
| 47 | set(_MULTI_VALUE_ARGS) |
| 48 | cmake_parse_arguments(_MY_PARAMS "${_OPTIONS_ARGS}" "${_ONE_VALUE_ARGS}" "${_MULTI_VALUE_ARGS}" ${ARGN}) |
| 49 | |
| 50 | set(TARGET ${_MY_PARAMS_TARGET}) |
| 51 | set(DEPENDS ${_MY_PARAMS_DEPENDS}) |
| 52 | |
| 53 | string(REGEX REPLACE "\\/" "_" CUSTOM_TARGET_SUFFIX ${DEPENDS}) |
| 54 | |
| 55 | add_custom_target(${TARGET}_target_${CUSTOM_TARGET_SUFFIX} DEPENDS ${DEPENDS}) |
| 56 | add_dependencies(${TARGET} ${TARGET}_target_${CUSTOM_TARGET_SUFFIX}) |
| 57 | endfunction() |
| 58 | |
Imre Kis | e6bf9e6 | 2020-01-20 11:34:24 +0100 | [diff] [blame] | 59 | # Generates LCOV coverage info file by processing the .gcda and .gcno files. |
| 60 | # The function also maps coverage of the c-picker generated files to the original source lines |
| 61 | # Global dependencies: |
| 62 | # CPICKER_CACHE_PATH: root directory of the c-picker generated files |
| 63 | function(coverage_generate) |
| 64 | set(_OPTIONS_ARGS) |
| 65 | set(_ONE_VALUE_ARGS NAME OUTPUT_FILE) |
| 66 | set(_MULTI_VALUE_ARGS) |
| 67 | cmake_parse_arguments(_MY_PARAMS "${_OPTIONS_ARGS}" "${_ONE_VALUE_ARGS}" "${_MULTI_VALUE_ARGS}" ${ARGN}) |
| 68 | |
| 69 | set(TEST_NAME ${_MY_PARAMS_NAME}) |
Imre Kis | c73346f | 2020-02-07 15:26:27 +0100 | [diff] [blame^] | 70 | set(TEMP_FILE ${_MY_PARAMS_OUTPUT_FILE}_temp) |
Imre Kis | e6bf9e6 | 2020-01-20 11:34:24 +0100 | [diff] [blame] | 71 | set(OUTPUT_FILE ${_MY_PARAMS_OUTPUT_FILE}) |
| 72 | |
Imre Kis | e6bf9e6 | 2020-01-20 11:34:24 +0100 | [diff] [blame] | 73 | # Collecting information from .gcda and .gcno files into an lcov .info file |
| 74 | # Mapping c-picker generated files' coverage info to the original source lines |
| 75 | add_custom_command( |
| 76 | OUTPUT ${TEMP_FILE} ${OUTPUT_FILE} |
| 77 | COMMAND ${LCOV_COMMAND} |
| 78 | --capture |
| 79 | --test-name ${TEST_NAME} |
| 80 | --directory ${CMAKE_CURRENT_BINARY_DIR} |
| 81 | --base-directory ${TF_A_PATH} |
| 82 | --output-file ${TEMP_FILE} |
| 83 | COMMAND ${CPICKER_COVERAGE_MAPPER_COMMAND} |
| 84 | --input ${TEMP_FILE} |
| 85 | --output ${OUTPUT_FILE} |
| 86 | --mapping-path ${CPICKER_CACHE_PATH} |
Imre Kis | e6bf9e6 | 2020-01-20 11:34:24 +0100 | [diff] [blame] | 87 | ) |
Imre Kis | c73346f | 2020-02-07 15:26:27 +0100 | [diff] [blame^] | 88 | |
| 89 | add_coverage_dependency( |
| 90 | TARGET coverage |
| 91 | DEPENDS ${OUTPUT_FILE} |
| 92 | ) |
Imre Kis | e6bf9e6 | 2020-01-20 11:34:24 +0100 | [diff] [blame] | 93 | endfunction() |
| 94 | |
| 95 | # Filters coverage info of files from the matching directory |
| 96 | function(coverage_filter) |
| 97 | set(_OPTIONS_ARGS) |
| 98 | set(_ONE_VALUE_ARGS INPUT_FILE OUTPUT_FILE INCLUDE_DIRECTORY) |
| 99 | set(_MULTI_VALUE_ARGS) |
| 100 | cmake_parse_arguments(_MY_PARAMS "${_OPTIONS_ARGS}" "${_ONE_VALUE_ARGS}" "${_MULTI_VALUE_ARGS}" ${ARGN}) |
| 101 | |
Imre Kis | c73346f | 2020-02-07 15:26:27 +0100 | [diff] [blame^] | 102 | set(INPUT_FILE ${_MY_PARAMS_INPUT_FILE}) |
| 103 | set(OUTPUT_FILE ${_MY_PARAMS_OUTPUT_FILE}) |
| 104 | set(INCLUDE_DIRECTORY ${_MY_PARAMS_INCLUDE_DIRECTORY}) |
| 105 | |
Imre Kis | e6bf9e6 | 2020-01-20 11:34:24 +0100 | [diff] [blame] | 106 | # The pattern must be an absolute path ending with an asterisk |
Imre Kis | c73346f | 2020-02-07 15:26:27 +0100 | [diff] [blame^] | 107 | get_filename_component(INCLUDE_DIRECTORY_ABSPATH "${INCLUDE_DIRECTORY}" ABSOLUTE) |
Imre Kis | e6bf9e6 | 2020-01-20 11:34:24 +0100 | [diff] [blame] | 108 | set(INCLUDE_DIRECTORY_ABSPATH "${INCLUDE_DIRECTORY_ABSPATH}/*") |
| 109 | |
| 110 | add_custom_command( |
Imre Kis | c73346f | 2020-02-07 15:26:27 +0100 | [diff] [blame^] | 111 | OUTPUT ${OUTPUT_FILE} |
Imre Kis | e6bf9e6 | 2020-01-20 11:34:24 +0100 | [diff] [blame] | 112 | COMMAND ${LCOV_COMMAND} |
Imre Kis | c73346f | 2020-02-07 15:26:27 +0100 | [diff] [blame^] | 113 | --extract ${INPUT_FILE} \"${INCLUDE_DIRECTORY_ABSPATH}\" |
| 114 | --output-file ${OUTPUT_FILE} |
| 115 | DEPENDS ${INPUT_FILE} |
Imre Kis | e6bf9e6 | 2020-01-20 11:34:24 +0100 | [diff] [blame] | 116 | ) |
| 117 | |
Imre Kis | c73346f | 2020-02-07 15:26:27 +0100 | [diff] [blame^] | 118 | add_coverage_dependency( |
| 119 | TARGET coverage |
| 120 | DEPENDS ${OUTPUT_FILE} |
| 121 | ) |
Imre Kis | e6bf9e6 | 2020-01-20 11:34:24 +0100 | [diff] [blame] | 122 | endfunction() |
| 123 | |
| 124 | # Generated an HTML report from the LCOV info file |
| 125 | function(coverage_generate_report) |
| 126 | set(_OPTIONS_ARGS) |
| 127 | set(_ONE_VALUE_ARGS INPUT_FILE OUTPUT_DIRECTORY) |
| 128 | set(_MULTI_VALUE_ARGS) |
| 129 | cmake_parse_arguments(_MY_PARAMS "${_OPTIONS_ARGS}" "${_ONE_VALUE_ARGS}" "${_MULTI_VALUE_ARGS}" ${ARGN}) |
| 130 | |
Imre Kis | c73346f | 2020-02-07 15:26:27 +0100 | [diff] [blame^] | 131 | set(INPUT_FILE ${_MY_PARAMS_INPUT_FILE}) |
| 132 | set(OUTPUT_DIRECTORY ${_MY_PARAMS_OUTPUT_DIRECTORY}) |
| 133 | |
Imre Kis | e6bf9e6 | 2020-01-20 11:34:24 +0100 | [diff] [blame] | 134 | add_custom_command( |
Imre Kis | c73346f | 2020-02-07 15:26:27 +0100 | [diff] [blame^] | 135 | OUTPUT ${OUTPUT_DIRECTORY} |
| 136 | COMMAND genhtml ${INPUT_FILE} |
Imre Kis | e6bf9e6 | 2020-01-20 11:34:24 +0100 | [diff] [blame] | 137 | --show-details |
Imre Kis | c73346f | 2020-02-07 15:26:27 +0100 | [diff] [blame^] | 138 | --output-directory ${OUTPUT_DIRECTORY} |
| 139 | DEPENDS ${INPUT_FILE} |
Imre Kis | e6bf9e6 | 2020-01-20 11:34:24 +0100 | [diff] [blame] | 140 | ) |
| 141 | |
Imre Kis | c73346f | 2020-02-07 15:26:27 +0100 | [diff] [blame^] | 142 | add_coverage_dependency( |
| 143 | TARGET coverage_report |
| 144 | DEPENDS ${OUTPUT_DIRECTORY} |
| 145 | ) |
Imre Kis | e6bf9e6 | 2020-01-20 11:34:24 +0100 | [diff] [blame] | 146 | endfunction() |