CMake编译项目集成gcov/lcov代码覆盖率测试
转自https://blog.csdn.net/lostaway/article/details/40948841,并针对自己项目做了小幅修改,在此感谢原作者
完整演示项目Github地址:https://github.com/lostaway/EasonCodeShare/tree/master/CMakeGcovSupport
初始项目目录结构:
CMakeGcovSupport
├── CMakeLists.txt
├── bin
├── build
├── include
│ └── name.h
├── libgreeting
│ ├── CMakeLists.txt
│ ├── include
│ │ └── greeting.h
│ └── src
│ ├── CMakeLists.txt
│ ├── greeting.cpp
│ └── yelp.cpp
└── src
├── CMakeGcovSupport.cpp
├── CMakeLists.txt
└── name.cpp
示例程序是个很简单的输出一行问候语的程序。为了演示复杂目录结构下CMake集成Gcov/lcov的方法,故意将输出问候语的函数单独放到了 libgreeting 静态库中。执行如下编译命令:
$ cd CMakeGcovSupport
$ mkdir build
$ cd build
$ cmake -DENABLE_COVERAGE=ON ..
$ gmake all
执行 CMake 外部编译之后,CMake 在 build 目录内为我们生成了 .gcno 文件:
CMakeGcovSupport
├── CMakeLists.txt
├── bin
│ └── CMakeGcovSupport
├── build
│ ├── ......
│ ├── libgreeting
│ │ ├── ......
│ │ └── src
│ │ ├── CMakeFiles
│ │ │ ├── ......
│ │ │ ├── greeting.dir
│ │ │ │ ├── ......
│ │ │ │ ├── greeting.cpp.gcno
│ │ │ │ ├── greeting.cpp.o
│ │ │ │ ├── ......
│ │ │ │ ├── yelp.cpp.gcno
│ │ │ │ └── yelp.cpp.o
│ │ │ └── ......
│ │ └── ......
│ └── src
│ ├── CMakeFiles
│ │ ├── ......
│ │ ├── CMakeGcovSupport.dir
│ │ │ ├── CMakeGcovSupport.cpp.gcno
│ │ │ ├── CMakeGcovSupport.cpp.o
│ │ │ ├── ......
│ │ │ ├── name.cpp.gcno
│ │ │ ├── name.cpp.o
│ │ │ └── ......
│ │ └── progress.marks
│ └── ......
├── include
├── libgreeting
└── src
为了避免接下来执行程序过程中,未覆盖的源码文件的覆盖率信息丢失,我们需要对覆盖率信息进行初始化操作:
$ cd CMakeGcovSupport
$ lcov -d build -z
$ lcov -d build -b . --no-external --initial -c -o CMakeGcovSupportInitialCoverage.info
然后我们执行 bin 中的 CMakeGcovSupport, main() 函数中将会调用 Greeting() 和 Name() 函数,而不会调用到 Yelp() 函数。
$ cd CMakeGcovSupport
$ cd bin
$ ./CMakeGcovSupport
$ Hello, gcov.
此时我们会看到在 .gcno 所在目录,有同名的 .gcda 覆盖率数据文件生成了。执行以下命令,生成覆盖率测试报告:
$ cd CMakeGcovSupport
$ lcov -d build -b . --no-external -c -o CMakeGcovSupportCoverage.info
$ genhtml -o CMakeGcovSupportCoverageReport --prefix='pwd' CMakeGcovSupportInitialCoverage.info CMakeGcovSupportCoverage.info
用浏览器打开 CMakeGcovSupportCoverageReport 目录中的 index.html 查看覆盖率报告。
以下部分为具体实施时可能会遇到的问题:
1.报告中不包含branch覆盖率信息
原因在于lcov 1.10以后版本默认不包含branch coverage信息,需要通过修改 vim /etc/lcovrc 文件默认打开branch分支信息的输出,具体修改如下:
# Specify if branch coverage data should be collected andprocessed.
lcov_branch_coverage = 1 #去掉注释,值改为1
# Include branch coverage datadisplay (can be disabled by the --no-branch-coverage option of genhtml)
genhtml_branch_coverage = 1 #去掉注释,值改为1
---------------------------------------------------------------
或者在lcov命令后加上参数 lcov -d <gcda目录位置> -b <测试代码路径> -c -o result.info --rc lcov_branch_coverage=1
再在genhtml命令后加上参数 genhtml -o result result.info --branch-coverage
2.对于复杂项目,不想包含某个文件夹内的文件覆盖率信息,即反向去除不需要的文件,可以使用 --remove 参数
lcov --remove all.info '*/lib/*' -o result.info
正向提取需要的文件,可以使用 --extract 参数
lcov --extract all.info '*/src/*' -o result.info
注意:lcov 不允许同时使用--extract 和 --remove
3.使用 --no-external 参数排除外部库
也可以将命令写成脚本文件:
# InitialCoverage.sh 此脚本文件在编译动作完成后运行,需使用chmod +x InitialCoverage.sh将其转变为可执行文件
$!/bin/sh
echo "Start initialize lcov coverage"
lcov -d cmake-build-release/src -z
lcov -d cmake-build-release/src -b . --no-external --initial -c -o InitialCoverage.info
lcov --remove InitialCoverage.info '*/lib/*' -o InitCoverage.info
rm -rf InitialCoverage.info
exit 1
# ReportCoverage.sh 此脚本文件在runtest或gtest后运行
#!/bin/sh
echo "Start report lcov coverage"
lcov -d cmake-build-release/src -b . --no-external -c -o ReportCoverage.info
lcov --remove ReportCoverage.info '*/lib/*' -o ResultCoverage.info
genhtml -o CoverageReport --prefix='pwd' InitCoverage.info ResultCoverage.info
rm -rf InitCoverage.info
rm -rf ReportCoverage.info
rm -rf ResultCoverage.info
exit 1
另外,CMakeLists.txt中需要添加以下几行:
set(ENABLE_COVERAGE TRUE)
if(ENABLE_COVERAGE)
message(STATUS "Use gcov for code coverage test")
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -fprofile-arcs -ftest-coverage")
endif()