gcov lcov genhtml工具

     gcov伴随gcc 发布。gcc编译加入-fprofile-arcs -ftest-coverage 参数生成二进制程序,执行测试用例生成代码覆盖率信息。

使用这些工具产生代码覆盖率的统计,需要经过:编译参数设置&编译、运行、gcov生成覆盖信息、locv生成 html报表数据、genhtml根据报表数据生成可视化的html报告 这几个步骤

编译参数设置&编译

    用GCC编译的时候加上-fprofile-arcs -ftest-coverage选项,链接的时候也加上。
    fprofile-arcs参数使gcc创建一个程序的流图,之后找到适合图的生成树。只有不在生成树中的弧被操纵(instrumented):gcc添加了代码来清点这些弧执行的次数。当这段弧是一个块的唯一出口或入口时,操纵工具代码(instrumentation code)将会添加到块中,否则创建一个基础块来包含操纵工具代码。

    编译成功后,会生成各*.c对应的 *.gcno文件。*.gcno是由-ftest-coverage产生的,它包含了重建基本块图和相应的块的源码的行号的信息。

运行

    也就是执行单元测试进程,如果编译参数中加了gcov相关的参数后,执行结果会生成 各 *.c文件对应的 *.gcda文件。*gcda是由加了-fprofile-arcs编译参数的编译后的文件运行所产生的,它包含了弧跳变的次数和其他的概要信息(而gcda只能在程序运行完毕后才能产生的)。

 

gcov生成覆盖信息

     Gcov执行函数覆盖、语句覆盖和分支覆盖。gcov工具会生成各*.c文件对应的 *.c.gcov 文件,不过*.c.gcov在lcov命令执行后,会被删除。*.c.gcov包含了函数和代码执行次数的信息。

locv生成 html报表数据

Lcov则是上的gcov 结果展现的一个前端,可以将覆盖率信息转换成html展现,它用来解析*.c.gcov中晦涩的字符,生成代码覆盖的信息,并输出到文件(*.info),genhtml根据*.info文件生成html报告。

lcov -d ./ -c  -o $(LCOV_INFO) -b ./

 

genhtml根据报表数据生成可视化的html报告

genhtml -o $(COV_REPORT) $(LCOV_INFO) --show-details –legend

 

最终的报告,*.html文件

 

 

完整的makfile

# makefile for sample cpputest

INC_DIR := -I../inc -I../inc/mysql -I../src/libcas
SRC_FILE := libcas.c cJSON.c qs_thread.c qs_init.c qs_paltask.c qs_cmd.c qs_db.c qs_main.c
SRC_OBJS := libcas.so
SRC_LNAME := cas TEST_GROUP_FILE := test.c mocks.c
TEST_EXE := test_libcas RUN_PREPARE := "cp ../bin/cas.json ." UT_LOG := result.log
CU_PROJECT_NAME := $(shell pwd |awk -F "/" '{print $$NF}' )
LCOV_INFO := $(CU_PROJECT_NAME).info
COV_REPORT := $(CU_PROJECT_NAME)_cover all: $(TEST_EXE) $(SRC_OBJS) CPPUTEST_HOME := /usr/cpputest CC := gcc -Wformat
CCFLAGS := -g -DLINUX $(INC_DIR) -shared -fPIC -fprofile-arcs -ftest-coverage --coverage
CCFLAGS += -D CPPUTEST # 编译测试文件时, 忽略被测代码 *.c 的main函数, test.c 的代码中用了宏CPPUTEST
CCLDFLAGS := -lgcov
CCLDFLAGS += -L../bin -lm -lvst_memory -lvst_shell_log -ldbase_comm -lbase_log -lvst_plat -lpthread \
-loam_proxy -ldbase_my CPP := g++
CPPFLAGS := -g -w -fpermissive -DLINUX -fprofile-arcs -ftest-coverage --coverage
CPPFLAGS += -I$(CPPUTEST_HOME)/include $(INC_DIR)
CPPLDFLAGS := -lgcov
CPPLDFLAGS += -Wl,-rpath,./,-rpath,./../bin -L$(CPPUTEST_HOME)/lib -L../bin -lCppUTest -lCppUTestExt
CPPLDFLAGS += -lssl -ldl -lpthread -lvst_com -lvst_memory -lvst_plat -lvst_encrypt\
-lvst_shell_cli -lvst_shell_log -lm -lvst_thpool -lreadline \
-lvst_thpool -ldbcfg_parse -ldata_parse \
-lipdr -laccpwd -lstat -luas -ldata_stat CPPLDFLAGS += -l$(SRC_LNAME) SRCDIR := $(shell ls -d ../src/*)
vpath
vpath %.c $(SRCDIR) $(SRC_OBJS): $(SRC_FILE)
$(CC) -o $@ $^ $(CCFLAGS) $(CCLDFLAGS) $(TEST_EXE): $(TEST_GROUP_FILE) $(SRC_OBJS)
$(CPP) -o $@ $^ $(CPPFLAGS) $(CPPLDFLAGS) .PHONY: clean
clean:
@echo "clean..."
rm -f $(SRC_OBJS) $(TEST_EXE)
rm -f *.gcda
rm -f *.gcno
rm -f *.gcov
rm -f $(UT_LOG)
rm -rf $(COV_REPORT)
rm -f *.info run:
eval $(RUN_PREPARE)
./$(TEST_EXE) &> $(UT_LOG) cover:
for file in $(SRC_FILE);\
do\
gcov -b "$$file";\
done
lcov -d ./ -c -o $(LCOV_INFO) -b ./
genhtml -o $(COV_REPORT) $(LCOV_INFO) --show-details --legend

 

一些问题

一开始的时候,生成的报告,只有 line和functions的统计,没有branches的统计。

在执行 make cover后,日志只有:

gcov命令执行的日志中,有branch的信息:

开始解决问题的思路:是不是gcov 或lcov的命令参数不对,查了命令及尝试了修改参数后,仍未解决。

后来发现 lcov的命令日志中有如下告警:

按告警信息搜索,发现了 stackoverflow上有个相似的问题:

http://stackoverflow.com/questions/13378640/lcov-can-not-collect-branch-coverage-statistics

参照上面的思路,修改了/etc/lcovrc中

lcov_branch_coverage=1

genhtml_branch_coverage=1

后,最终的html报告中才有branches的统计。

c/c++代码的unit-test中覆盖率的统计的更多相关文章

  1. Mysql基础代码(不断完善中)

    Mysql基础代码,不断完善中~ /* 启动MySQL */ net start mysql /* 连接与断开服务器 */ mysql -h 地址 -P 端口 -u 用户名 -p 密码 /* 跳过权限 ...

  2. php示例代码之类似于C#中的String.Format方法

    php示例代码之类似于C#中的String.Format方法 原文来自于  http://stackoverflow.com/questions/1241177/c-string-format-equ ...

  3. 利用Roslyn把C#代码编译到内存中并进行执行

    Tugberk Ugurlu在其博文<Compiling C# Code Into Memory and Executing It with Roslyn>中给大家介绍了一种使用.NET下 ...

  4. ASP.net中网站访问量统计方法代码(在线人数,本月访问,本日访问,访问流量,累计访问)

    一.建立一个数据表IPStat用于存放用户信息 我在IPStat表中存放的用户信息只包括登录用户的IP(IP_Address),IP来源(IP_Src)和登录时间 (IP_DateTime),些表的信 ...

  5. mac github工具将命令当下来的代码拖入macgithub中就可以

    mac github工具将命令当下来的代码拖入macgithub中就可以,刚開始傻傻的就知道点击那个加入button,总是在当下来的文件夹下创建个文件夹.并且代码不能同步

  6. git使用方法----如何利用git管理代码?如何使用git将代码传到github中去

    ##  在文件夹中打开 git here; 1.git init ===初始化一个仓库(这个仓库会存放,git对我们代码进行备份的文件)2.配置个人信息 -- --在git中设置当前使用的用户是==( ...

  7. .NET/C# 异常处理:写一个空的 try 块代码,而把重要代码写到 finally 中

    不知你是否见过 try { } finally { } 代码中,try 块留空,而只往 finally 中写代码的情况呢?这种写法有其特殊的目的. 本文就来说说这种不一样的写法. 你可以点开这个链接查 ...

  8. Google自动广告,将广告代码放置在 HTML 中的什么位置?

    Google自动广告,将广告代码放置在 HTML 中的什么位置? 为自动广告生成广告代码后,您需要将此代码放置在要展示广告的每个网页中.您应将广告代码放置在网页的 <head> 标记(或正 ...

  9. Java用代码演示String类中的以下方法的用法

    用代码演示String类中的以下方法的用法 (1)boolean isEmpty(): 判断字符串是不是空串,如果是空的就返回true (2)char charAt(int index): 返回索引上 ...

随机推荐

  1. EF添加ADO.NET实体模型处直接选择Oracle数据源

    上一文介绍了如何下载Mysql for vs Tools来进行Mysql的ADO.NET实体模型数据源选择,今天将Oracle的测试了下.步骤如下: 1.在你项目Model层中nuget安装选中项 2 ...

  2. FineReport——自定义登录页

    统一的接口: http://localhost:8075/WebReport/ReportServer?op=fs_load&cmd=sso&fr_username=XX&fr ...

  3. 502 bad gateway,ngix

    1.ftm对nginx的解析出现问题:20171228

  4. Linux非常用命令

    查看系统版本(64位还是32位版本) uname -a 或 more /proc/version 执行结果

  5. 《逐梦旅程 WINDOWS游戏编程之从零开始》笔记4——Direct3D编程基础

    第11章 Direct3D编程基础 2D游戏是贴图的艺术,3D游戏是渲染的艺术.这句话在我学过了之前的GDI编程之后,前一句算是有所体会,现在是来理解后一句的时候了. 安装DirectX SDK配置啥 ...

  6. OpenCL与CUDA,CPU与GPU

    OpenCL OpenCL(全称Open Computing Language,开放运算语言)是第一个面向异构系统通用目的并行编程的开放式.免费标准,也是一个统一的编程环境,便于软件开发人员为高性能计 ...

  7. 【JBPM4】任务节点-任务分配swimlane

    swimlane泳道,几个任务受理人相同的任务节点,可以划分为一个泳道 JPDL <?xml version="1.0" encoding="UTF-8" ...

  8. [水煮 ASP.NET Web API2 方法论](12-2)管理 OData 路由

    问题 如何控制 OData 路由 解决方案 为了注册路由,可以使用  HttpConfigurationExtension 类中 MapODataServiceRoute 的扩展方法.对于单一路由这样 ...

  9. CentOS7.5安装配置conky(极简)

    1.安装epel源 下载地址:http://dl.fedoraproject.org/pub/epel/ 找到epel-release-XXXXXXX.rpm文件,下载解压 rpm -ivh epel ...

  10. ESLint 使用入门

    在团队协作中,为避免低级 Bug.产出风格统一的代码,会预先制定编码规范.使用 Lint 工具和代码风格检测工具,则可以辅助编码规范执行,有效控制代码质量. 在以前的项目中,我们选择 JSHint 和 ...