forked from github/verilator
Update internal code coverage framework.
This commit is contained in:
parent
29bcbb0417
commit
1d0a726437
@ -29,7 +29,7 @@ before_install:
|
||||
# Not listing Bit::Vector as slow to install, and only skips one test
|
||||
- touch temp.cpp ; g++ -E -dM -c temp.cpp | sort ; rm -rf temp.cpp
|
||||
- yes yes | sudo cpan -fi Unix::Processors Parallel::Forker
|
||||
- sudo apt-get install gdb gtkwave
|
||||
- sudo apt-get install gdb gtkwave lcov
|
||||
- sudo apt-get install libfl-dev || true
|
||||
- sudo apt-get install libgoogle-perftools-dev
|
||||
before_script:
|
||||
|
@ -98,7 +98,7 @@ need to be present to run Verilator:
|
||||
|
||||
Those developing Verilator itself may also want these (see internals.adoc):
|
||||
|
||||
sudo apt-get install gdb asciidoctor graphviz cmake clang clang-format gprof
|
||||
sudo apt-get install gdb asciidoctor graphviz cmake clang clang-format gprof lcov
|
||||
cpan install Pod::Perldoc
|
||||
cpan install Unix::Processors
|
||||
cpan install Parallel::Forker
|
||||
|
@ -12,7 +12,10 @@ use Pod::Usage;
|
||||
use strict;
|
||||
use vars qw($Debug);
|
||||
|
||||
our $Opt_Stop;
|
||||
our $Exclude_Line_Regexp;
|
||||
our $Remove_Gcda_Regexp;
|
||||
|
||||
our @Remove_Sources;
|
||||
our @Source_Globs;
|
||||
|
||||
@ -27,7 +30,8 @@ Getopt::Long::config("no_auto_abbrev");
|
||||
if (! GetOptions(
|
||||
"debug" => sub { $Debug = 1; },
|
||||
"<>" => sub { die "%Error: Unknown parameter: $_[0]\n"; },
|
||||
"stage=i" => \$Opt_Stage,
|
||||
"stage=i" => \$Opt_Stage, # starting stage number
|
||||
"stop!" => \$Opt_Stop, # stop/do not stop on error in tests
|
||||
)) {
|
||||
die "%Error: Bad usage, try 'install_test --help'\n";
|
||||
}
|
||||
@ -42,45 +46,85 @@ sub test {
|
||||
require "./nodist/code_coverage.dat";
|
||||
|
||||
if ($Opt_Stage <= 0) {
|
||||
travis_fold_start("configure");
|
||||
print "Stage 0: configure (coverage on)\n";
|
||||
run("make distclean");
|
||||
run("make distclean || true");
|
||||
run("./configure --enable-longtests CXX='g++ --coverage'");
|
||||
run("make -k");
|
||||
# The optimized versions will not collect good coverage, overwrite them
|
||||
run("cp bin/verilator_bin_dbg bin/verilator_bin");
|
||||
run("cp bin/verilator_coverage_bin_dbg bin/verilator_coverage_bin");
|
||||
travis_fold_end();
|
||||
}
|
||||
|
||||
if ($Opt_Stage <= 1) {
|
||||
travis_fold_start("test");
|
||||
print "Stage 1: make examples (with coverage on)\n";
|
||||
run("make examples");
|
||||
run("make test_regress");
|
||||
run("make test_regress"
|
||||
. ($Opt_Stop ? '' : ' || true'));
|
||||
travis_fold_end();
|
||||
}
|
||||
|
||||
my $cc_dir = "nodist/obj_dir/coverage";
|
||||
if ($Opt_Stage <= 2) {
|
||||
travis_fold_start("info");
|
||||
print "Stage 2: Create info files under $cc_dir\n";
|
||||
mkpath($cc_dir);
|
||||
mkpath("$cc_dir/info");
|
||||
my $dats = `find . -print | grep .gcda`;
|
||||
my %dirs;
|
||||
my %dats;
|
||||
foreach my $dat (split '\n', $dats) {
|
||||
$dat =~ s!/[^/]+$!!;
|
||||
$dirs{$dat} = 1;
|
||||
$dats{$dat} = 1;
|
||||
}
|
||||
my %dirs;
|
||||
my %gcnos;
|
||||
foreach my $dat (sort keys %dats) {
|
||||
(my $gcno = $dat) =~ s!\.gcda$!.gcno!;
|
||||
if ($dat =~ /$Remove_Gcda_Regexp/) {
|
||||
# Remove .gcda/.gcno for files we don't care about before we slowly
|
||||
# read them
|
||||
unlink $dat;
|
||||
unlink $gcno;
|
||||
delete $dats{$dat};
|
||||
next;
|
||||
}
|
||||
(my $gbase = $gcno) =~ s!.*/!!;
|
||||
if (!$gcnos{$gbase} && -r $gcno) {
|
||||
$gcnos{$gbase} = $gcno;
|
||||
}
|
||||
}
|
||||
# We need a matching .gcno for every .gcda, try to find a matching file elsewhere
|
||||
foreach my $dat (sort keys %dats) {
|
||||
(my $gcno = $dat) =~ s!\.gcda$!.gcno!;
|
||||
(my $gbase = $gcno) =~ s!.*/!!;
|
||||
if (!-r $gcno) {
|
||||
if ($gcnos{$gbase}) {
|
||||
cp($gcnos{$gbase}, $gcno);
|
||||
} else {
|
||||
warn "MISSING .gcno for a .gcda: $gcno\n";
|
||||
}
|
||||
}
|
||||
(my $dir = $dat) =~ s!/[^/]+$!!;
|
||||
$dirs{$dir} = 1;
|
||||
}
|
||||
foreach my $dir (sort keys %dirs) {
|
||||
(my $outname = $dir) =~ s![^a-zA-Z0-9]+!_!g;
|
||||
run("cd $cc_dir/info ; lcov -c -d ../../../../$dir -o app_test_${outname}.info");
|
||||
}
|
||||
travis_fold_end();
|
||||
}
|
||||
|
||||
if ($Opt_Stage <= 3) {
|
||||
travis_fold_start("clone");
|
||||
# lcov doesn't have a control file to override single lines, so replicate the sources
|
||||
print "Stage 3: Clone sources under $cc_dir\n";
|
||||
clone_sources($cc_dir);
|
||||
travis_fold_end();
|
||||
}
|
||||
|
||||
if ($Opt_Stage <= 4) {
|
||||
travis_fold_start("copy");
|
||||
print "Stage 4: Copy .gcno files\n";
|
||||
my $dats = `find . -print | grep .gcno`;
|
||||
foreach my $dat (split '\n', $dats) {
|
||||
@ -89,9 +133,11 @@ sub test {
|
||||
#print "cp $dat, $outdat);\n";
|
||||
cp($dat, $outdat);
|
||||
}
|
||||
travis_fold_end();
|
||||
}
|
||||
|
||||
if ($Opt_Stage <= 5) {
|
||||
travis_fold_start("combine");
|
||||
print "Stage 5: Combine data files\n";
|
||||
run("cd $cc_dir ; lcov -c -i -d src/obj_dbg -o app_base.info");
|
||||
run("cd $cc_dir ; lcov -a app_base.info -o app_total.info");
|
||||
@ -106,22 +152,35 @@ sub test {
|
||||
$comb = "";
|
||||
}
|
||||
}
|
||||
travis_fold_end();
|
||||
}
|
||||
|
||||
if ($Opt_Stage <= 6) {
|
||||
travis_fold_start("filter");
|
||||
print "Stage 6: Filter processed source files\n";
|
||||
my $cmd = '';
|
||||
foreach my $glob (@Remove_Sources) {
|
||||
$cmd .= " '$glob'";
|
||||
}
|
||||
run("cd $cc_dir ; lcov --remove app_total.info $cmd -o app_total.info");
|
||||
travis_fold_end();
|
||||
}
|
||||
|
||||
if ($Opt_Stage <= 7) {
|
||||
if ($Opt_Stage <= 8) {
|
||||
travis_fold_start("report");
|
||||
print "Stage 7: Create HTML\n";
|
||||
cleanup_abs_paths($cc_dir, "$cc_dir/app_total.info", "$cc_dir/app_total.info");
|
||||
run("cd $cc_dir ; genhtml app_total.info --demangle-cpp"
|
||||
." --rc lcov_branch_coverage=1 --rc genhtml_hi_limit=100 --output-directory html");
|
||||
travis_fold_end();
|
||||
}
|
||||
|
||||
if ($Opt_Stage <= 9) {
|
||||
travis_fold_start("upload");
|
||||
print "Stage 9: Upload\n";
|
||||
my $cmd = 'bash <(curl -s https://codecov.io/bash) -f $cc_dir/app_total.info';
|
||||
print "print: Not running: $cmd\n";
|
||||
travis_fold_end();
|
||||
}
|
||||
|
||||
if ($Opt_Stage <= 9) {
|
||||
@ -184,12 +243,13 @@ sub cleanup_abs_paths {
|
||||
sub exclude_line_regexp {
|
||||
$Exclude_Line_Regexp = shift;
|
||||
}
|
||||
|
||||
sub remove_gcda_regexp {
|
||||
$Remove_Gcda_Regexp = shift;
|
||||
}
|
||||
sub remove_source {
|
||||
my @srcs = @_;
|
||||
push @Remove_Sources, @srcs;
|
||||
}
|
||||
|
||||
sub source_globs {
|
||||
my @dirs = @_;
|
||||
push @Source_Globs, @dirs;
|
||||
@ -206,6 +266,15 @@ sub run {
|
||||
($status == 0) or die "%Error: Command Failed $command, $status, stopped";
|
||||
}
|
||||
|
||||
our $_Travis_Action;
|
||||
sub travis_fold_start {
|
||||
$_Travis_Action = shift;
|
||||
print "travis_fold:start:$_Travis_Action\n";
|
||||
}
|
||||
sub travis_fold_end {
|
||||
print "travis_fold:end:$_Travis_Action\n";
|
||||
}
|
||||
|
||||
#######################################################################
|
||||
__END__
|
||||
|
||||
|
@ -22,6 +22,7 @@ source_globs("src/*.cpp",
|
||||
);
|
||||
|
||||
remove_source("/usr/include/*");
|
||||
remove_source("/usr/lib/*");
|
||||
remove_source("*/include/sysc/*");
|
||||
remove_source("*/V3ClkGater.cpp");
|
||||
remove_source("*/V3ClkGater.h");
|
||||
@ -37,7 +38,10 @@ remove_source("*include/gtkwave/*");
|
||||
remove_source("*test_regress/*");
|
||||
remove_source("*examples/*");
|
||||
|
||||
# Remove collected coverage on each little test main file
|
||||
# Would just be removed with remove_source in later step
|
||||
remove_gcda_regexp(qr!test_regress/.*/(Vt_|Vtop_).*\.gcda!);
|
||||
|
||||
exclude_line_regexp(qr/(\bv3fatalSrc\b|\bVL_UNCOVERABLE\b)/);
|
||||
exclude_line_regexp(qr/(\bv3fatalSrc\b|\bVL_UNCOVERABLE\b|\bVL_FATAL)/);
|
||||
|
||||
1;
|
||||
|
@ -306,6 +306,8 @@ sub new {
|
||||
skip_msgs => [],
|
||||
fail_msgs => [],
|
||||
fail_tests => [],
|
||||
# Per-task identifiers
|
||||
running_ids => {},
|
||||
@_};
|
||||
bless $self, $class;
|
||||
return $self;
|
||||
@ -322,14 +324,24 @@ sub one_test {
|
||||
$::Fork->schedule
|
||||
(
|
||||
test_pl_filename => $params{pl_filename},
|
||||
run_pre_start => sub {
|
||||
my $process = shift;
|
||||
# Running in context of parent, before run_on_start
|
||||
# Make an identifier that is unique across all current running jobs
|
||||
my $i = 1; while (exists $self->{running_ids}{$i}) { ++$i; }
|
||||
$process->{running_id} = $i;
|
||||
$self->{running_ids}{$process->{running_id}} = 1;
|
||||
},
|
||||
run_on_start => sub {
|
||||
my $process = shift;
|
||||
# Running in context of child, so can't pass data to parent directly
|
||||
if ($self->{quiet}) {
|
||||
open(STDOUT, ">/dev/null");
|
||||
open(STDERR, ">&STDOUT");
|
||||
}
|
||||
print("="x70,"\n");
|
||||
my $test = VTest->new(@params);
|
||||
my $test = VTest->new(@params,
|
||||
running_id => $process->{running_id});
|
||||
$test->oprint("="x50,"\n");
|
||||
unlink $test->{status_filename};
|
||||
$test->_prep;
|
||||
@ -340,7 +352,9 @@ sub one_test {
|
||||
},
|
||||
run_on_finish => sub {
|
||||
# Running in context of parent
|
||||
my $test = VTest->new(@params);
|
||||
my $process = shift;
|
||||
my $test = VTest->new(@params,
|
||||
running_id => $process->{running_id});
|
||||
$test->_read_status;
|
||||
if ($test->ok) {
|
||||
$self->{ok_cnt}++;
|
||||
@ -374,6 +388,7 @@ sub one_test {
|
||||
}
|
||||
$self->{left_cnt}--;
|
||||
$self->print_summary;
|
||||
delete $self->{running_ids}{$process->{running_id}};
|
||||
},
|
||||
)->ready();
|
||||
}
|
||||
@ -1062,6 +1077,7 @@ sub compile {
|
||||
tee => $param{tee},
|
||||
expect => $param{expect},
|
||||
expect_filename => $param{expect_filename},
|
||||
verilator_run => 1,
|
||||
cmd => \@vlt_cmd) if $::Opt_Verilation;
|
||||
return 1 if $self->errors || $self->skips || $self->unsupporteds;
|
||||
}
|
||||
@ -1077,6 +1093,7 @@ sub compile {
|
||||
tee => $param{tee},
|
||||
expect => $param{expect},
|
||||
expect_filename => $param{expect_filename},
|
||||
verilator_run => 1,
|
||||
cmd => ["cd \"".$self->{obj_dir}."\" && cmake",
|
||||
"\"".$self->{t_dir}."/..\"",
|
||||
"-DTEST_VERILATOR_ROOT=$ENV{VERILATOR_ROOT}",
|
||||
@ -1458,6 +1475,7 @@ sub _run {
|
||||
my $self = (ref $_[0]? shift : $Self);
|
||||
my %param = (tee => 1,
|
||||
#entering => # Print entering directory information
|
||||
#verilator_run => # Move gcov data to parallel area
|
||||
@_);
|
||||
my $command = join(' ',@{$param{cmd}});
|
||||
$command = "time $command" if $opt_benchmark && $command !~ /^cd /;
|
||||
@ -1465,6 +1483,17 @@ sub _run {
|
||||
print " > $param{logfile}" if $param{logfile};
|
||||
print "\n";
|
||||
|
||||
if ($param{verilator_run}) {
|
||||
# Gcov fails when parallel jobs write same data file,
|
||||
# so we make sure output dir is unique across all running jobs.
|
||||
# We can't just put each one in obj_dir as it uses too much disk.
|
||||
$ENV{GCOV_PREFIX_STRIP} = 99;
|
||||
$ENV{GCOV_PREFIX} = "$self->{t_dir}/obj_dist/gcov_$self->{running_id}";
|
||||
} else {
|
||||
delete $ENV{GCOV_PREFIX_STRIP};
|
||||
delete $ENV{GCOV_PREFIX};
|
||||
}
|
||||
|
||||
# Execute command redirecting output, keeping order between stderr and stdout.
|
||||
# Must do low-level IO so GCC interaction works (can't be line-based)
|
||||
my $status;
|
||||
@ -2271,6 +2300,7 @@ sub schedule {
|
||||
my $self = shift;
|
||||
my %params = (@_);
|
||||
|
||||
$params{run_pre_start}->($self);
|
||||
if (my $pid = fork()) { # Parent
|
||||
waitpid($pid, 0);
|
||||
} else { # Child
|
||||
|
@ -16,7 +16,9 @@ scenarios(simulator => 1);
|
||||
|
||||
$DEBUG_QUIET = "--debug --debugi 0 --gdbbt --no-dump-tree";
|
||||
|
||||
run(cmd => ["perl", "../bin/verilator", $DEBUG_QUIET, "-V"]);
|
||||
run(cmd => ["perl", "../bin/verilator", $DEBUG_QUIET, "-V"],
|
||||
verilator_run => 1,
|
||||
);
|
||||
|
||||
compile(
|
||||
verilator_flags2 => [$DEBUG_QUIET, "--trace"],
|
||||
|
@ -25,8 +25,9 @@ inline_checks();
|
||||
|
||||
run(cmd => ["../bin/verilator_coverage",
|
||||
"--annotate", "$Self->{obj_dir}/annotated",
|
||||
"$Self->{obj_dir}/coverage.dat",
|
||||
]);
|
||||
"$Self->{obj_dir}/coverage.dat"],
|
||||
verilator_run => 1,
|
||||
);
|
||||
|
||||
files_identical("$Self->{obj_dir}/annotated/t_cover_line.v", "t/t_cover_line.out");
|
||||
|
||||
|
@ -26,7 +26,9 @@ inline_checks();
|
||||
run(cmd => ["../bin/verilator_coverage",
|
||||
"--annotate", "$Self->{obj_dir}/annotated",
|
||||
"$Self->{obj_dir}/coverage.dat",
|
||||
]);
|
||||
],
|
||||
verilator_run => 1,
|
||||
);
|
||||
|
||||
files_identical("$Self->{obj_dir}/annotated/t_cover_line.v", "t/t_cover_line.out");
|
||||
|
||||
|
@ -23,7 +23,9 @@ execute(
|
||||
run(cmd => ["../bin/verilator_coverage",
|
||||
"--annotate", "$Self->{obj_dir}/annotated",
|
||||
"$Self->{obj_dir}/coverage.dat",
|
||||
]);
|
||||
],
|
||||
verilator_run => 1,
|
||||
);
|
||||
|
||||
files_identical("$Self->{obj_dir}/annotated/t_cover_line.v", "t/t_cover_line.out");
|
||||
vcd_identical("$Self->{obj_dir}/simx.vcd", $Self->{golden_filename});
|
||||
|
@ -23,6 +23,7 @@ foreach my $prog (
|
||||
"--help"],
|
||||
logfile => "$Self->{obj_dir}/t_help.log",
|
||||
tee => 0,
|
||||
verilator_run => 1,
|
||||
);
|
||||
file_grep("$Self->{obj_dir}/t_help.log", qr/DISTRIBUTION/i);
|
||||
}
|
||||
|
@ -24,6 +24,7 @@ foreach my $prog (
|
||||
tee => $self->{verbose},
|
||||
logfile => "$Self->{obj_dir}/t_help.log",
|
||||
expect => qr/^Verilator/,
|
||||
verilator_run => 1,
|
||||
);
|
||||
|
||||
run(fails => 0,
|
||||
@ -32,6 +33,7 @@ foreach my $prog (
|
||||
tee => $self->{verbose},
|
||||
logfile => "$Self->{obj_dir}/t_help.log",
|
||||
expect => qr/^Verilator/,
|
||||
verilator_run => 1,
|
||||
);
|
||||
|
||||
run(fails => 0,
|
||||
@ -39,6 +41,7 @@ foreach my $prog (
|
||||
"-V"],
|
||||
logfile => "$Self->{obj_dir}/t_help.log",
|
||||
expect => qr/^Verilator/,
|
||||
verilator_run => 1,
|
||||
);
|
||||
}
|
||||
|
||||
|
@ -40,7 +40,9 @@ execute(
|
||||
run(cmd => ["$ENV{VERILATOR_ROOT}/bin/verilator_gantt",
|
||||
"$Self->{obj_dir}/profile_threads.dat",
|
||||
"--vcd $Self->{obj_dir}/profile_threads.vcd",
|
||||
"> $Self->{obj_dir}/gantt.log"]);
|
||||
"> $Self->{obj_dir}/gantt.log"],
|
||||
verilator_run => 1,
|
||||
);
|
||||
|
||||
# We should have three lines of gantt chart, each with
|
||||
# an even number of mtask-bars (eg "[123--]")
|
||||
|
@ -13,8 +13,9 @@ scenarios(vlt => 1);
|
||||
lint();
|
||||
|
||||
foreach my $file (glob("$Self->{obj_dir}/*")) {
|
||||
next if $file =~ /\.log/; # Made by driver.pl, not Verilator
|
||||
next if $file =~ /\.status/; # Made by driver.pl, not Verilator
|
||||
next if $file =~ /\.log/; # Made by driver.pl, not Verilator sources
|
||||
next if $file =~ /\.status/; # Made by driver.pl, not Verilator sources
|
||||
next if $file =~ /\.gcda/; # Made by gcov, not Verilator sources
|
||||
error("%Error: Created $file, but --lint-only shouldn't create files");
|
||||
}
|
||||
|
||||
|
@ -35,7 +35,9 @@ while (1) {
|
||||
$secret_dir,
|
||||
"--protect-lib",
|
||||
$secret_prefix,
|
||||
"t/t_prot_lib_secret.v"]);
|
||||
"t/t_prot_lib_secret.v"],
|
||||
verilator_run => 1,
|
||||
);
|
||||
last if $Self->{errors};
|
||||
|
||||
run(logfile => "$secret_dir/secret_gcc.log",
|
||||
|
@ -37,7 +37,9 @@ while (1) {
|
||||
"-GGATED_CLK=1",
|
||||
"--protect-lib",
|
||||
$secret_prefix,
|
||||
"t/t_prot_lib_secret.v"]);
|
||||
"t/t_prot_lib_secret.v"],
|
||||
verilator_run => 1,
|
||||
);
|
||||
last if $Self->{errors};
|
||||
|
||||
run(logfile => "$secret_dir/secret_gcc.log",
|
||||
|
@ -20,6 +20,7 @@ foreach my $basename ("t_vlcov_data_a.dat",
|
||||
"--debugi 9",
|
||||
],
|
||||
tee => $Self->{verbose},
|
||||
verilator_run => 1,
|
||||
);
|
||||
}
|
||||
|
||||
|
@ -16,7 +16,9 @@ run(cmd => ["../bin/verilator_coverage",
|
||||
"t/t_vlcov_data_b.dat",
|
||||
"t/t_vlcov_data_c.dat",
|
||||
"t/t_vlcov_data_d.dat",
|
||||
]);
|
||||
],
|
||||
verilator_run => 1,
|
||||
);
|
||||
|
||||
# Older clib's didn't properly sort maps, but the coverage data doesn't
|
||||
# really care about ordering. So avoid false failures by sorting.
|
||||
|
@ -15,6 +15,7 @@ run(fails => 1,
|
||||
"t/t_NOT_FOUND",],
|
||||
logfile => $Self->{run_log_filename},
|
||||
expect_filename => $Self->{golden_filename},
|
||||
verilator_run => 1,
|
||||
);
|
||||
|
||||
ok(1);
|
||||
|
@ -19,6 +19,7 @@ run(cmd => ["../bin/verilator_coverage",
|
||||
],
|
||||
logfile => "$Self->{obj_dir}/vlcov.log",
|
||||
tee => 0,
|
||||
verilator_run => 1,
|
||||
);
|
||||
|
||||
files_identical("$Self->{obj_dir}/vlcov.log", $Self->{golden_filename});
|
||||
|
@ -20,6 +20,7 @@ foreach my $basename ("t_vlcov_data_a.dat",
|
||||
"--write", "$Self->{obj_dir}/${basename}"
|
||||
],
|
||||
tee => 0,
|
||||
verilator_run => 1,
|
||||
);
|
||||
files_identical("$Self->{obj_dir}/${basename}", "t/${basename}");
|
||||
}
|
||||
|
Loading…
Reference in New Issue
Block a user