【C++】统计代码覆盖率(二)
嗷嗷嗷!!!好激动,我好蠢。不过最后还是解决了。呜呜呜
有些都是东一块西一块查的,如果有侵权欢迎私信我,我注明出处。
一 gcov&CMake
昨天试了下测试代码和被测代码都是c++的情况,直接编译生成gcno文件,再一运行,生成gcda文件。脚本统计,blingbling生成了报表,简直漂亮!
不过我们的工程比较大= =。编译时也需要很多依赖文件。
因此使用场景为:在机器A目录编译,拷贝纯bin文件到B目录上运行。编译方式为CMake
1 修改编译脚本
- 找到CMakeList.txt文件,添加如下内容:
-
# coverage option OPTION (ENABLE_COVERAGE "Use gcov" OFF) MESSAGE(STATUS ENABLE_COVERAGE=${ENABLE_COVERAGE}) IF(ENABLE_COVERAGE) SET(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -fprofile-arcs -ftest-coverage") SET(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -fprofile-arcs -ftest-coverage")
# 其中 -fprofile-arcs 用来生成 gcno 文件; -ftest-coverage 用来在统计时生成 gcda 文件# SET(CMAKE_EXE_LINKER_FLAGS "${CMAKE_EXE_LINKER_FLAGS} -fprofile-arcs -ftest-coverage") ENDIF()
- 编译时使用命令如下:
cmake -DENABLE_COVERAGE=ON ..
- 编译后查看文件目录:build一路向下
-
******/build/CMakeFiles/ad_server.dir/src $ll -rw-rw-r-- 1 mobdev mobdev 1431748 Jun 22 18:24 A.cpp.gcno -rw-rw-r-- 1 mobdev mobdev 2138256 Jun 22 18:24 A.cpp.o -rw-rw-r-- 1 mobdev mobdev 314180 Jun 22 18:25 B.cpp.gcno -rw-rw-r-- 1 mobdev mobdev 871944 Jun 22 18:25 B.cpp.o -rw-rw-r-- 1 mobdev mobdev 2053332 Jun 22 18:25 C.cpp.gcno -rw-rw-r-- 1 mobdev mobdev 3931584 Jun 22 18:25 C.cpp.o -rw-rw-r-- 1 mobdev mobdev 1063036 Jun 22 18:24 D.cpp.gcno -rw-rw-r-- 1 mobdev mobdev 3280472 Jun 22 18:24 D.cpp.o
- 如上表示编译成功
2 拷贝bin文件到目录B,注意保持目录B与目录A的为同一次编译结果(我是检查md5文件)
如果是从机器A到机器B,注意配置gcov_prefix交叉编译。
3 启动你的服务,按原方式执行测试用例。case执行完成后进入下一步
4 生成gcda文件。vim test.sh,内容如下:
-
#!/bin/sh SERVER_NAME=$1 pid=`ps -ef | grep $SERVER_NAME | grep -v "grep" | awk '{print $2}'` echo $pid gdb -q attach $pid <<__EOF__ p __gcov_flush() __EOF__
执行命令:sh test.sh your_servername
5 检查:因为这次我的编译和测试在同台机器的不同目录,因为gcda文件会生成到编译时的gcno同目录,检查其中已有gcda文件。如下:
-
******/build/CMakeFiles/ad_server.dir/src $ll -rw-rw-r-- 1 mobdev mobdev 71772 Jun 22 18:55 A.cpp.gcda -rw-rw-r-- 1 mobdev mobdev 1431748 Jun 22 18:24 A.cpp.gcno -rw-rw-r-- 1 mobdev mobdev 2138256 Jun 22 18:24 A.cpp.o -rw-rw-r-- 1 mobdev mobdev 18452 Jun 22 18:55 B.cpp.gcda -rw-rw-r-- 1 mobdev mobdev 314180 Jun 22 18:25 B.cpp.gcno -rw-rw-r-- 1 mobdev mobdev 871944 Jun 22 18:25 B.cpp.o
二 生成html页面
进入gcda和gcno文件所在目录,执行 lcov -c -o result.info -b . -d . //生成info文件 genhtml result.info -o Report //生成html文件 tar cvf Report.tar.gz Report //压缩文件 sz Report.tar.gz //下载到windows系统 解压打开其中的index.html即可看到测试代码覆盖率
三 生成xml报告
1 安装gcovr
cd /usr/local wget https://github.com/gcovr/gcovr/archive/3.2.tar.gz tar -xvf 3.2.tar.gz cd gcovr-3.2/scripts cp gcovr /usr/bin
2 在编译路径下执行 gcovr -r .即可查看覆盖率情况
3 为了使其生成Cobertura可用的xml文件,可以使用命令
gcovr -r . --output yourdir/coverage.xml -xml-pretty
vim coverage.xml即可看到xml报告
四 问题
1 描述:无法生成gcda文件
原因如下:
- 用户代码调用 exit 正常结束时,gcov_exit 函数得到调用,其继续调用 __gcov_flush 函数输出统计数据到 *.gcda 文件中
- 若用户进程并非调用 exit 正常退出,覆盖率统计数据就无法输出,也就无从生成报告了。后台服务程序若非专门设计,一旦启动就很少主动退出,用 kill 杀死进程强制退出时就不会调用 exit,因此没有覆盖率统计结果产生。
解决:执行(一)4步骤即可。
2 描述:无法组合gcda和gcno文件:"stamp mismatch with notes file"
原因如下:
- 网上找了好多答案,说是编译版本不一致,导致时间戳不一致,我一直没有理解。
- 确认自己这儿是因为生成gcno文件的编译版本和生成gcda文件的版本不一致。
-
我的操作: 1 编译二进制文件ad.server,生成了gcno文件 2 拷贝ad.server进测试环境,测试生成了gcda 3 中途又编译了一次ad.server,生成了新的gcno文件. 即gcno和gcda文件使用的ad.server并不是同一次编译的结果。 最后通过对比两个地方的ad.server的md5发现了不同,我好蠢= =
解决:解决很简单,确保你的版本就ok了,可以对比二进制文件的md5.
查看时间戳:hexdump -e '"%x\n"' -s8 -n4 A.cpp.gcno 可以看到时间戳。
hexdump -e '"%x\n"' -s8 -n4 A.cpp.gcda 可以看到时间戳
3 描述:想要让gcda文件生成在指定目录
场景:有时候是在机器A编译,机器B运行,这种情况就会不能生成gcda文件,提示找不到目录
解决:gcov的交叉编译
操作如下:
- vim /etc/profile 进行配置
-
export GCOV_PREFIX="/data/ad_server/ad_server.dir" //gcda的目标路径 export GCOV_PREFIX_STRIP=9 //向上数你的路径到需要配置的那个 比如我的编译路径是/data/code/adserver/6/5/4/3/2/*.gcno,我就设置了9,应该也可以设置export GCOV_PREFIX_STRIP=999等很大然后配置全路径,我没试
- source /etc/profile 使其生效
- 重启服务进程
- 执行测试代码,exit()退出,查看profile的设置路径,已经有gcda文件
- 拷贝gcno文件到gcda路径,统计代码覆盖率
- blingbling的就大功告成了!
我的问题:配置了n久该路径一直不生效,最后要哭了。今天偶然发现需要重启进程,忧伤极了
4 描述:无法使用gcovr生成覆盖率--得到的覆盖率为0%
$gcovr -r . ------------------------------------------------------------------------------ GCC Code Coverage Report Directory: . ------------------------------------------------------------------------------ File Lines Exec Cover Missing ------------------------------------------------------------------------------ ------------------------------------------------------------------------------ TOTAL 0 0 --% ------------------------------------------------------------------------------
解决:执行gcovr -r .在编译的大路径下即可,或者在执行时指定source 路径 如 gcovr -r /home/test/mytest/XXXX/
目录不对,只有gcda和gcno、cpp.o文件无法生成覆盖率,需要源码--猜测
5 描述:使用 --remove 后未在指定路径看到 -o 后的新的 info 文件
解决:实际生成了,使用命令 find / -name "new_file.info" 发现在根目录下,不知道为什么会这样,使用时更新为 指定绝对路径 就 ok 了