116 lines
2.8 KiB
Python
116 lines
2.8 KiB
Python
#!/usr/bin/env python
|
|
#
|
|
# Copyright (c) 2011, 2012, Simon Howard
|
|
#
|
|
# Permission to use, copy, modify, and/or distribute this software
|
|
# for any purpose with or without fee is hereby granted, provided
|
|
# that the above copyright notice and this permission notice appear
|
|
# in all copies.
|
|
#
|
|
# THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL
|
|
# WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED
|
|
# WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE
|
|
# AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR
|
|
# CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
|
|
# LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT,
|
|
# NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN
|
|
# CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
|
#
|
|
|
|
from re import match
|
|
from os import rename, popen, getcwd, chdir
|
|
from os.path import exists, dirname, basename
|
|
from os.path import join as path_join
|
|
from glob import glob
|
|
import sys
|
|
|
|
def parse_stats(text):
|
|
m = match(r"(.*)% of (.*)", text)
|
|
|
|
pct = m.group(1)
|
|
lines = int(m.group(2))
|
|
cov_lines = int(float(pct) * lines / 100 + 0.5)
|
|
uncov_lines = lines - cov_lines
|
|
|
|
return (pct + "%", uncov_lines, cov_lines, lines)
|
|
|
|
|
|
def gcov(filename):
|
|
|
|
# gcov must be run from within the same directory as the source
|
|
# file that we are analysing.
|
|
|
|
old_wd = getcwd()
|
|
src_dir = dirname(filename)
|
|
chdir(src_dir)
|
|
s = popen("gcov %s" % basename(filename))
|
|
chdir(old_wd)
|
|
|
|
# Process output from gcov.
|
|
# This may cover multiple files when one .c file includes another.
|
|
|
|
results = {}
|
|
filename = None
|
|
|
|
# File 'lha_file_header.c'
|
|
# Lines executed:88.97% of 136
|
|
|
|
for line in s:
|
|
m = match(r"File '(.*)'", line)
|
|
if m:
|
|
filename = m.group(1)
|
|
|
|
m = match(r"Lines executed:(.*)", line)
|
|
if m:
|
|
full_path = path_join(src_dir, filename)
|
|
results[full_path] = parse_stats(m.group(1))
|
|
|
|
s.close()
|
|
|
|
return results
|
|
|
|
def format_output(filename, stats):
|
|
print " %-35s%7s%7s%7s%7s" % ((filename, ) + stats)
|
|
|
|
print
|
|
format_output("Filename", ("Percent", "Uncov", "Cov", "Total"))
|
|
print " " + ("-" * 65)
|
|
|
|
for filename in sorted(sys.argv[1:]):
|
|
gcno = filename.replace(".c", ".gcno")
|
|
gcda = filename.replace(".c", ".gcda")
|
|
|
|
xgcno = glob(path_join(dirname(gcno), "*-" + basename(gcno)))
|
|
xgcda = glob(path_join(dirname(gcda), "*-" + basename(gcda)))
|
|
|
|
# Must rename eg. liblhasatest_a-foo.gcno to foo.gcno:
|
|
|
|
if len(xgcno) > 0:
|
|
rename(xgcno[0], gcno)
|
|
|
|
if len(xgcda) > 0:
|
|
rename(xgcda[0], gcda)
|
|
|
|
if not exists(gcno) or not exists(gcda):
|
|
continue
|
|
|
|
# Run gcov and parse output:
|
|
|
|
results = gcov(filename)
|
|
|
|
if len(results) > 0:
|
|
if filename in results:
|
|
format_output(filename, results[filename])
|
|
else:
|
|
format_output(filename, ("", "", "", ""))
|
|
|
|
for subfile in sorted(results.keys()):
|
|
if subfile.endswith(".h"):
|
|
continue
|
|
|
|
if subfile != filename:
|
|
format_output(" -> " + subfile, results[subfile])
|
|
|
|
print
|
|
|