Linux kcov工具

阅读数:205 评论数:0

跳转到新版页面

分类

Linux

正文

代码覆盖率(Code Coverage)是反映测试用例对被测软件覆盖程序的重要指标,kcov是用于随机测试中合适表达代码覆盖率的一种工具。

前置需求

内核需要配置

CONFIG_KCOV=y

CONFIG_KCOV需要231296以上版本的gcc,如果需要收集比较操作

CONFIG_KCOV_ENABLE_COMPARISONS=y

只有挂载了debugfs,才可以访问配置数据。

使用方法

#include <stdio.h>

#include <stddef.h>

#include <stdint.h>

#include <stdlib.h>

#include <sys/types.h>

#include <sys/stat.h>

#include <sys/ioctl.h>

#include <sys/mman.h>

#include <unistd.h>

#include <fcntl.h>



#define KCOV_INIT_TRACE _IOR('c', 1, unsigned long)

#define KCOV_ENABLE _IO('c', 100)

#define KCOV_DISABLE _IO('c', 101)

#define COVER_SIZE (64<<10)



#define KCOV_TRACE_PC 0

#define KCOV_TRACE_CMP 1



int main(int argc, char **argv)

{

int fd;

unsigned long *cover, n, i;



/* A single fd descriptor allows coverage collection on a single

* thread.

*/

fd = open("/sys/kernel/debug/kcov", O_RDWR);

if (fd == -1)

perror("open"), exit(1);

/* Setup trace mode and trace size. */

if (ioctl(fd, KCOV_INIT_TRACE, COVER_SIZE))

perror("ioctl"), exit(1);

/* Mmap buffer shared between kernel- and user-space. */

cover = (unsigned long*)mmap(NULL, COVER_SIZE * sizeof(unsigned long),

PROT_READ | PROT_WRITE, MAP_SHARED, fd, 0);

if ((void*)cover == MAP_FAILED)

perror("mmap"), exit(1);

/* Enable coverage collection on the current thread. */

if (ioctl(fd, KCOV_ENABLE, KCOV_TRACE_PC))

perror("ioctl"), exit(1);

/* Reset coverage from the tail of the ioctl() call. */

__atomic_store_n(&cover[0], 0, __ATOMIC_RELAXED);

/* That's the target syscal call. */

read(-1, NULL, 0);

/* Read number of PCs collected. */

n = __atomic_load_n(&cover[0], __ATOMIC_RELAXED);

for (i = 0; i < n; i++)

printf("0x%lx\n", cover[i + 1]);

/* Disable coverage collection for the current thread. After this call

* coverage can be enabled for a different thread.

*/

if (ioctl(fd, KCOV_DISABLE, 0))

perror("ioctl"), exit(1);

/* Free resources. */

if (munmap(cover, COVER_SIZE * sizeof(unsigned long)))

perror("munmap"), exit(1);

if (close(fd))

perror("close"), exit(1);

return 0;

}

输出信息

SyS_read

fs/read_write.c:562

__fdget_pos

fs/file.c:774

__fget_light

fs/file.c:746

__fget_light

fs/file.c:750

__fget_light

fs/file.c:760

__fdget_pos

fs/file.c:784

SyS_read

fs/read_write.c:562

 

/* Same includes and defines as above. */



/* Number of 64-bit words per record. */

#define KCOV_WORDS_PER_CMP 4



/*

* The format for the types of collected comparisons.

*

* Bit 0 shows whether one of the arguments is a compile-time constant.

* Bits 1 & 2 contain log2 of the argument size, up to 8 bytes.

*/



#define KCOV_CMP_CONST (1 << 0)

#define KCOV_CMP_SIZE(n) ((n) << 1)

#define KCOV_CMP_MASK KCOV_CMP_SIZE(3)



int main(int argc, char **argv)

{

int fd;

uint64_t *cover, type, arg1, arg2, is_const, size;

unsigned long n, i;



fd = open("/sys/kernel/debug/kcov", O_RDWR);

if (fd == -1)

perror("open"), exit(1);

if (ioctl(fd, KCOV_INIT_TRACE, COVER_SIZE))

perror("ioctl"), exit(1);

/*

* Note that the buffer pointer is of type uint64_t*, because all

* the comparison operands are promoted to uint64_t.

*/

cover = (uint64_t *)mmap(NULL, COVER_SIZE * sizeof(unsigned long),

PROT_READ | PROT_WRITE, MAP_SHARED, fd, 0);

if ((void*)cover == MAP_FAILED)

perror("mmap"), exit(1);

/* Note KCOV_TRACE_CMP instead of KCOV_TRACE_PC. */

if (ioctl(fd, KCOV_ENABLE, KCOV_TRACE_CMP))

perror("ioctl"), exit(1);

__atomic_store_n(&cover[0], 0, __ATOMIC_RELAXED);

read(-1, NULL, 0);

/* Read number of comparisons collected. */

n = __atomic_load_n(&cover[0], __ATOMIC_RELAXED);

for (i = 0; i < n; i++) {

type = cover[i * KCOV_WORDS_PER_CMP + 1];

/* arg1 and arg2 - operands of the comparison. */

arg1 = cover[i * KCOV_WORDS_PER_CMP + 2];

arg2 = cover[i * KCOV_WORDS_PER_CMP + 3];

/* ip - caller address. */

ip = cover[i * KCOV_WORDS_PER_CMP + 4];

/* size of the operands. */

size = 1 << ((type & KCOV_CMP_MASK) >> 1);

/* is_const - true if either operand is a compile-time constant.*/

is_const = type & KCOV_CMP_CONST;

printf("ip: 0x%lx type: 0x%lx, arg1: 0x%lx, arg2: 0x%lx, "

"size: %lu, %s\n",

ip, type, arg1, arg2, size,

is_const ? "const" : "non-const");

}

if (ioctl(fd, KCOV_DISABLE, 0))

perror("ioctl"), exit(1);

/* Free resources. */

if (munmap(cover, COVER_SIZE * sizeof(unsigned long)))

perror("munmap"), exit(1);

if (close(fd))

perror("close"), exit(1);

return 0;

}



相关推荐

一、概述 1、为什么Linus不使用GPLv3 在 PC 上,只要你得到了某个程序的源代码,就可以自行编译生成二进制程序,然后替换掉原有的二进制程序,你的程序自由很容易得到保证。然而 iPod、iPh

说明:这种方式只是用于方便阅读代码,因为可以在源代间快速索引跳跃。但是最后可能会有一些warning,可以不必关心,如果是强迫症,可以使用下面这种方式来去掉。

一、概述 在Linux系统中,/usr/bin和/usr/local/bin是两个常见的目录,用于存放可执行文件(二进制文件)。 很多应用都安装在/usr/local下面,先看一下automake工具

  一、概述 vmstat命令是最常见的Linux/Unix监控工具,可以监控给定时间间隔服务器的CPU使用率、内存使用、IO情况。相比top命令,可以查看到整个机器的CPU、内存、IO的使用情况,而

一、概述 sar,System Activity Reporter。是目前 Linux 上最为全面的系统性能分析工具之一,可以从多方面对系统的活动进行报告,包括:文件的读写情况、系统调用的使用情况、磁

一、简介 简单来说awk就是把文件逐行的读入,以空格为默认分隔符将每行切片,切开的部分再进行各种分析处理。 二、使用方法 1、基本语法  awk '条件类型1 {动作1} 条件类型2{动作2} ...

一、概述 列出目标目录中所有的子目录和文件。 二、 语法 ls [选项] [目录名] -a, –all 列出目录下的所有文件,包括以 . 开头的隐含文件 -A 同-a,但不列出“.”(表示当前目录)

一、概述 cd全称是change directory,用于切换当前工作目录。 注意的是,cd命令是一个内建命令,它是由 shell 提供的。因此,不同的 shell 可能会有一些差异,但基本的用法和功

一、概述 全称为print working directory,查看”当前工作目录“的完整路径,一般情况下不带任何参数 二、语法 pwd [选项] -L 即logical,逻辑路径 -P 即

一、概述 通过 mkdir 命令可以实现在指定位置创建以 DirName(指定的文件名)命名的文件夹或目录。要创建文件夹或目录的用户必须对所创建的文件夹的父文件夹具有写权限。并且,所创建的文件夹(目录