gcov是一个测试代码覆盖率的程序,正确地使用它搭配 gcc
可以分析、帮助你将代码写得更高效。帮助你优化程序。类似于一个profiling tool,使用 gcov 或者 gprof,可以收集到一些基础的性能统计数据。比如:
fprofile-arcs参数使gcc创建一个程序的流图,之后找到适合图的生成树。只有不在生成树中的弧被操纵(instrumented):gcc添加了代码来清点这些弧执行的次数。当这段弧是一个块的唯一出口或入口时,操纵工具代码(instrumentation code)将会添加到块中,否则创建一个基础块来包含操纵工具代码。
详情请参考这个链接:https://www.cnblogs.com/ChinaHook/p/5508660.html
lcov 是 gcc 测试覆盖率的前端图形展示工具。它通过收集多个源文件的 行、函数和分支的代码覆盖信息(程序执行之后生成gcda、gcno文件,上面的链接有讲) 并且将收集后的信息生成HTML页面。生成HTML需要使用genhtml命令。
首先,在代码编译和链接的时候,需要加上下面两个编译选项。在链接时需要加上gcov链接参数。
-fprofile-arcs
-ftest-coverage
##############################################
# COVERAGE (生成覆盖率选项)
##############################################
if (ENABLE_COVERAGE)
set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -g -O0 -fprofile-arcs -ftest-coverage --coverage")
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -g -O0 -fprofile-arcs -ftest-coverage --coverage")
endif()
在lcov目录(我们的这次测试使用的目录)下存在3个文件,test_case.cpp test_case.h test_main.cpp test_case.ccp 文件
int add (int a , int b ) {
return a+b ;
}
int minus(int a , int b ) {
return a-b ;
}
test_case.h文件
int add (int a , int b );
int minus(int a , int b );
test_main.cpp文件
#include
#include "test_case.h"
int main() {
std::cout << add(10,20) << std::endl ;
return 0 ;
}
这里的a.cpp和a.hpp中定义了两个接口 add 和 minus(暂不考虑溢出问题)。我们在testa.cpp中调用add这个接口。我们的代码覆盖率应该是50%,因为总共两个接口,我们只使用(调用)了其中一个。
推荐的使用流程如下(最后再详述lcov常用的参数的含义):
我们使用lcov时需要在项目的根路径。
# 来自 man lcov
Recommended procedure when capturing data for a test case:
#1. create baseline coverage data file
lcov -c -i -d appdir -o app_base.info
#2. perform test
appdir/test
#3. create test coverage data file
lcov -c -d appdir -o app_test.info
# 4. combine baseline and test coverage data
lcov -a app_base.info -a app_test.info -o app_total.info
按照上述流程,我们使用自己的例子来执行
g++ test_main.cpp test_case.cpp -fprofile-arcs -ftest-coverage -lgcov -o test_cover
这步可有可无,即归零所有执行过的产生覆盖率信息的统计文件:
lcov -d ./ -z
# -c 捕获,-i初始化,-d应用目录,-o输出文件
lcov -c -i -d ./ -o init.info
./test_cover
lcov -c -d ./ -o cover.info
# -a 合并文件
lcov -a init.info -a cover.info -o total.info
# --remove 删除统计信息中如下的代码或文件,支持正则
lcov --remove total.info '*/usr/include/*' '*/usr/lib/*' '*/usr/lib64/*' '*/usr/local/include/*' '*/usr/local/lib/*' '*/usr/local/lib64/*' '*/third/*' 'test_main.cpp' -o final.info
#如果是git目录,可以获取此次版本的commitID,如果不是,忽略此步
# commitId=$(git log | head -n1 | awk '{print $2}')
# 这里可以带上项目名称和提交ID,如果没有,忽略此步
#genhtml -o cover_report --legend --title "${project_name} commit SHA1:${commitId}" --prefix=${curr_path} final.info
# -o 生成的html及相关文件的目录名称,--legend 简单的统计信息说明
# --title 项目名称,--prefix 将要生成的html文件的路径
genhtml -o cover_report --legend --title "project_name" --prefix=./ final.info
目录下所有文件
test_case.gcda、test_case.gcno、test_main.gcda、test_main.gcno就是运行可执行文件后gcov产生的统计信息文件。
cover_report目录就是生成的html信息目录。
这样,我们就可以通过firefox或者chrome打开cover_report/index.html来查看我们的代码覆盖率。
注:如果使用CMake编译和构建的话,可以在指定路径的时候,统一使用项目的编译构建路径,即项目下源文件下创建的build目录作为路径,(运行lcov在项目的根路径)举例如下:
lcov -c -i -d ./build -o init.info
lcov -c -d ./build -o cover.info
... ...
lcov常用的参数
-d 项目路径,即.gcda .gcno所在的路径
-a 合并(归并)多个lcov生成的info文件
-c 捕获,也即收集代码运行后所产生的统计计数信息
--external 捕获其它目录产生的统计计数文件
-i/--initial 初始化所有的覆盖率信息,作为基准数据
-o 生成处理后的文件
-r/--remove 移除不需要关注的覆盖率信息文件
-z 重置所有执行程序所产生的统计信息为0
#!/bin/bash
CUR_DIR=`pwd`
OLD_DIR="$(cd "$(dirname "$0")" && pwd)"
APP_HOME="OLD_DIR/APP_HOME"
function FnCreateCoverage()
{
echo -e "\033[32mCreator Coverage\033[0m"
# 初始化并创建基准数据文件
lcov -c -i -d build -o build.info
# 执行编译后的测试程序
cd ${APP_HOME}/testbin
chmod +x gtest_all
./gtest_all
cd ${OLD_DIR}
#收集测试文件运行后的覆盖率文件
lcov -c -d build -o cover.info
#合并基准数据和执行测试文件后的生成的覆盖率数据
lcov -a build.info -a cover.info -o total.info
# 过滤不关心的源文件路径
lcov --remove total.info '*/usr/include/*' '*/usr/lib/*' '*/usr/lib64/*' '*/usr/local/include/*' '*/usr/local/lib/*' '*/usr/local/lib64/*' '*/third/*' 'test_main.cpp' -o final.info
genhtml -o cover_report --legend --title "project_name" --prefix=./ final.info
rm *.info >/del/null 2>&1
}
function FnDelCoverage()
{
echo -e "\033[32mDelete Coverage\033[0m"
echo ""
rm -rf cover_report >/del/null 2>&1
rm *.info >/del/null 2>&1
}
function Usage()
{
echo "Usage:"
echo -e "\tsh coverage.sh [option]"
echo ""
echo -e "\t del delete coverage"
echo -e "\t gen generate coverage"
}
function ParseCommandLine()
{
if [ -n "$1" ];then
case "$1" in
"gen")
FnCreateCoverage
;;
"del")
FnDelCoverage
;;
*)
FnCreateCoverage
;;
esac
else
FnCreateCoverage
fi
}
function main()
{
ParseCommandLine "$@"
}
main $@
cd ${CUR_DIR}
exit 0