]> git.cameronkatri.com Git - mandoc.git/blobdiff - regress/regress.pl
more details about Mac OS X; information from Sevan Janiyan
[mandoc.git] / regress / regress.pl
index c49743d5819999dccf21743ec4b11a726dbbc6d3..3423ca63395f081989972cc4449c2e863b691ab5 100755 (executable)
@@ -1,8 +1,8 @@
 #!/usr/bin/env perl
 #
-# $Id: regress.pl,v 1.5 2017/03/08 22:54:22 schwarze Exp $
+# $Id: regress.pl,v 1.16 2021/09/19 12:15:34 schwarze Exp $
 #
-# Copyright (c) 2017 Ingo Schwarze <schwarze@openbsd.org>
+# Copyright (c) 2017,2018,2019,2020,2021 Ingo Schwarze <schwarze@openbsd.org>
 #
 # Permission to use, copy, modify, and distribute this software for any
 # purpose with or without fee is hereby granted, provided that the above
@@ -27,6 +27,7 @@ use IPC::Open3 qw(open3);
 # Define this at one place such that it can easily be changed
 # if diff(1) does not support the -a option.
 my @diff = qw(diff -au);
+system @diff, '/dev/null', '/dev/null' and @diff = qw(diff -u);
 
 # --- utility functions ------------------------------------------------
 
@@ -36,12 +37,17 @@ sub usage ($) {
        exit 1;
 }
 
+# Modifier arguments provided on the command line,
+# inspected by the main program and by the utility functions.
+my %targets;
+
 # Run a command and send STDOUT and STDERR to a file.
 # 1st argument: path to the output file
 # 2nd argument: command name
 # The remaining arguments are passed to the command.
 sub sysout ($@) {
        my $outfile = shift;
+       print "@_\n" if $targets{verbose};
        local *OUT_FH;
        open OUT_FH, '>', $outfile or die "$outfile: $!";
        my $pid = open3 undef, ">&OUT_FH", undef, @_;
@@ -53,6 +59,7 @@ sub sysout ($@) {
 # Simlar, but filter the output as needed for the lint test.
 sub syslint ($@) {
        my $outfile = shift;
+       print "@_\n" if $targets{verbose};
        open my $outfd, '>', $outfile or die "$outfile: $!";
        my $infd;
        my $pid = open3 undef, $infd, undef, @_;
@@ -69,26 +76,32 @@ sub syslint ($@) {
 # Simlar, but filter the output as needed for the html test.
 sub syshtml ($@) {
        my $outfile = shift;
+       print "@_\n" if $targets{verbose};
        open my $outfd, '>', $outfile or die "$outfile: $!";
        my $infd;
        my $pid = open3 undef, $infd, undef, @_;
-       my $state;
+       my $state = 0;
        while (<$infd>) {
                chomp;
                if (!$state && s/.*<math class="eqn">//) {
-                       $state = 1;
+                       $state = 'math';
                        next unless length;
+               } elsif (/BEGINTEST/) {
+                       $state = 'other';
+                       next;
+               } elsif (/ENDTEST/) {
+                       $state = 0;
+                       next;
                }
-               $state = 1 if /^BEGINTEST/;
-               if ($state && s/<\/math>.*//) {
+               if ($state eq 'math') {
                        s/^ *//;
-                       print $outfd "$_\n" if length;
-                       undef $state;
-                       next;
+                       if (s/<\/math>.*//) {
+                               print $outfd "$_\n" if length;
+                               $state = 0;
+                               next;
+                       }
                }
-               s/^ *//;
                print $outfd "$_\n" if $state;
-               undef $state if /^ENDTEST/;
        }
        close $outfd;
        close $infd;
@@ -97,7 +110,7 @@ sub syshtml ($@) {
 }
 
 my @failures;
-sub fail ($$$) {
+sub fail ($$) {
        warn "FAILED: @_\n";
        push @failures, [@_];
 }
@@ -105,32 +118,25 @@ sub fail ($$$) {
 
 # --- process command line arguments -----------------------------------
 
-my ($subdir, $onlytest) = split ':', (shift // '.');
-my $displaylevel = 2;
-my %targets;
+my $onlytest = shift // '';
 for (@ARGV) {
-       if (/^[0123]$/) {
-               $displaylevel = int;
-               next;
-       }
-       /^(all|ascii|utf8|man|html|markdown|lint|clean|verbose)$/
+       /^(all|ascii|tag|man|utf8|html|markdown|lint|clean|verbose)$/
            or usage "$_: invalid modifier";
        $targets{$_} = 1;
 }
 $targets{all} = 1
-    unless $targets{ascii} || $targets{utf8} || $targets{man} ||
-      $targets{html} || $targets{markdown} ||
+    unless $targets{ascii} || $targets{tag} || $targets{man} ||
+      $targets{utf8} || $targets{html} || $targets{markdown} ||
       $targets{lint} || $targets{clean};
-$targets{ascii} = $targets{utf8} = $targets{man} = $targets{html} =
-    $targets{markdown} = $targets{lint} = 1 if $targets{all};
-$displaylevel = 3 if $targets{verbose};
+$targets{ascii} = $targets{tag} = $targets{man} = $targets{utf8} =
+    $targets{html} = $targets{markdown} = $targets{lint} = 1
+    if $targets{all};
 
 
 # --- parse Makefiles --------------------------------------------------
 
-my %vars = (MOPTS => '');
-sub parse_makefile ($) {
-       my $filename = shift;
+sub parse_makefile ($%) {
+       my ($filename, $vars) = @_;
        open my $fh, '<', $filename or die "$filename: $!";
        while (<$fh>) {
                chomp;
@@ -143,242 +149,325 @@ sub parse_makefile ($) {
                my $var = $1;
                my $opt = $2;
                my $val = $3;
-               $val =~ s/\$\{(\w+)\}/$vars{$1}/;
-               $val = "$vars{$var} $val" if $opt eq '+';
-               $vars{$var} = $val
-                   unless $opt eq '?' && defined $vars{$var};
+               $val =~ s/\$\{(\w+)\}/$vars->{$1}/;
+               $val = "$vars->{$var} $val" if $opt eq '+';
+               $vars->{$var} = $val
+                   unless $opt eq '?' && defined $vars->{$var};
        }
        close $fh;
 }
 
-if ($subdir eq '.') {
-       $vars{SUBDIR} = 'roff char mdoc man tbl eqn';
-} else {
-       parse_makefile "$subdir/Makefile";
-       parse_makefile "$subdir/../Makefile.inc"
-           if -e "$subdir/../Makefile.inc";
-}
-
-my @mandoc = '../mandoc';
-my @subdir_names;
-my (@regress_testnames, @utf8_testnames, @lint_testnames);
-my (@html_testnames, @markdown_testnames);
-my (%skip_ascii, %skip_man, %skip_markdown);
-
-push @mandoc, split ' ', $vars{MOPTS} if $vars{MOPTS};
-delete $vars{MOPTS};
-delete $vars{SKIP_GROFF};
-delete $vars{SKIP_GROFF_ASCII};
-delete $vars{TBL};
-delete $vars{EQN};
-if (defined $vars{SUBDIR}) {
-       @subdir_names = split ' ', $vars{SUBDIR};
-       delete $vars{SUBDIR};
-}
-if (defined $vars{REGRESS_TARGETS}) {
-       @regress_testnames = split ' ', $vars{REGRESS_TARGETS};
-       delete $vars{REGRESS_TARGETS};
-}
-if (defined $vars{UTF8_TARGETS}) {
-       @utf8_testnames = split ' ', $vars{UTF8_TARGETS};
-       delete $vars{UTF8_TARGETS};
-}
-if (defined $vars{HTML_TARGETS}) {
-       @html_testnames = split ' ', $vars{HTML_TARGETS};
-       delete $vars{HTML_TARGETS};
-}
-if (defined $vars{MARKDOWN_TARGETS}) {
-       @markdown_testnames = split ' ', $vars{MARKDOWN_TARGETS};
-       delete $vars{MARKDOWN_TARGETS};
-}
-if (defined $vars{LINT_TARGETS}) {
-       @lint_testnames = split ' ', $vars{LINT_TARGETS};
-       delete $vars{LINT_TARGETS};
-}
-if (defined $vars{SKIP_ASCII}) {
-       for (split ' ', $vars{SKIP_ASCII}) {
-               $skip_ascii{$_} = 1;
-               $skip_man{$_} = 1;
+my (@regress_tests, @utf8_tests, @lint_tests, @html_tests);
+my (%tag_tests, %skip_ascii, %skip_man, %skip_markdown);
+foreach my $module (qw(roff char mdoc man tbl eqn)) {
+       my %modvars;
+       parse_makefile "$module/Makefile", \%modvars;
+       foreach my $subdir (split ' ', $modvars{SUBDIR}) {
+               my %subvars = (MOPTS => '');
+               parse_makefile "$module/$subdir/Makefile", \%subvars;
+               parse_makefile "$module/Makefile.inc", \%subvars;
+               delete $subvars{GOPTS};
+               delete $subvars{SKIP_GROFF};
+               delete $subvars{SKIP_GROFF_ASCII};
+               my @mopts = split ' ', $subvars{MOPTS};
+               delete $subvars{MOPTS};
+               my @regress_testnames;
+               if (defined $subvars{TAG_TARGETS}) {
+                       $tag_tests{"$module/$subdir/$_"} = 1
+                           for split ' ', $subvars{TAG_TARGETS};
+                       delete $subvars{TAG_TARGETS};
+               }
+               if (defined $subvars{REGRESS_TARGETS}) {
+                       push @regress_testnames,
+                           split ' ', $subvars{REGRESS_TARGETS};
+                       push @regress_tests, {
+                           NAME => "$module/$subdir/$_",
+                           MOPTS => \@mopts,
+                       } foreach @regress_testnames;
+                       delete $subvars{REGRESS_TARGETS};
+               }
+               if (defined $subvars{UTF8_TARGETS}) {
+                       push @utf8_tests, {
+                           NAME => "$module/$subdir/$_",
+                           MOPTS => \@mopts,
+                       } foreach split ' ', $subvars{UTF8_TARGETS};
+                       delete $subvars{UTF8_TARGETS};
+               }
+               if (defined $subvars{HTML_TARGETS}) {
+                       push @html_tests, {
+                           NAME => "$module/$subdir/$_",
+                           MOPTS => \@mopts,
+                       } foreach split ' ', $subvars{HTML_TARGETS};
+                       delete $subvars{HTML_TARGETS};
+               }
+               if (defined $subvars{LINT_TARGETS}) {
+                       push @lint_tests, {
+                           NAME => "$module/$subdir/$_",
+                           MOPTS => \@mopts,
+                       } foreach split ' ', $subvars{LINT_TARGETS};
+                       delete $subvars{LINT_TARGETS};
+               }
+               if (defined $subvars{SKIP_ASCII}) {
+                       for (split ' ', $subvars{SKIP_ASCII}) {
+                               $skip_ascii{"$module/$subdir/$_"} = 1;
+                               $skip_man{"$module/$subdir/$_"} = 1;
+                       }
+                       delete $subvars{SKIP_ASCII};
+               }
+               if (defined $subvars{SKIP_TMAN}) {
+                       $skip_man{"$module/$subdir/$_"} = 1
+                           for split ' ', $subvars{SKIP_TMAN};
+                       delete $subvars{SKIP_TMAN};
+               }
+               if (defined $subvars{SKIP_MARKDOWN}) {
+                       $skip_markdown{"$module/$subdir/$_"} = 1
+                           for split ' ', $subvars{SKIP_MARKDOWN};
+                       delete $subvars{SKIP_MARKDOWN};
+               }
+               if (keys %subvars) {
+                       my @vars = keys %subvars;
+                       die "unknown var(s) @vars in dir $module/$subdir";
+               }
+               map {
+                       $skip_ascii{"$module/$subdir/$_"} = 1;
+               } @regress_testnames if $skip_ascii{"$module/$subdir/ALL"};
+               map {
+                       $skip_man{"$module/$subdir/$_"} = 1;
+               } @regress_testnames if $skip_man{"$module/$subdir/ALL"};
+               map {
+                       $skip_markdown{"$module/$subdir/$_"} = 1;
+               } @regress_testnames if $skip_markdown{"$module/$subdir/ALL"};
+       }
+       delete $modvars{SUBDIR};
+       if (keys %modvars) {
+               my @vars = keys %modvars;
+               die "unknown var(s) @vars in module $module";
        }
-       delete $vars{SKIP_ASCII};
-}
-if (defined $vars{SKIP_TMAN}) {
-       $skip_man{$_} = 1 for split ' ', $vars{SKIP_TMAN};
-       delete $vars{SKIP_TMAN};
-}
-if (defined $vars{SKIP_MARKDOWN}) {
-       $skip_markdown{$_} = 1 for split ' ', $vars{SKIP_MARKDOWN};
-       delete $vars{SKIP_MARKDOWN};
-}
-if (keys %vars) {
-       my @vars = keys %vars;
-       die "unknown var(s) @vars";
 }
-map { $skip_ascii{$_} = 1; } @regress_testnames if $skip_ascii{ALL};
-map { $skip_man{$_} = 1; } @regress_testnames if $skip_man{ALL};
-map { $skip_markdown{$_} = 1; } @regress_testnames if $skip_markdown{ALL};
 
 # --- run targets ------------------------------------------------------
 
 my $count_total = 0;
-for my $dirname (@subdir_names) {
-       $count_total++;
-       print "\n" if $targets{verbose};
-       system './regress.pl', "$subdir/$dirname", keys %targets,
-           ($displaylevel ? $displaylevel - 1 : 0),
-           and fail $subdir, $dirname, 'subdir';
-}
-
 my $count_ascii = 0;
+my $count_tag = 0;
 my $count_man = 0;
-for my $testname (@regress_testnames) {
-       next if $onlytest && $testname ne $onlytest;
-       my $i = "$subdir/$testname.in";
-       my $o = "$subdir/$testname.mandoc_ascii";
-       my $w = "$subdir/$testname.out_ascii";
-       if ($targets{ascii} && !$skip_ascii{$testname}) {
+my $count_rm = 0;
+if ($targets{ascii} || $targets{tag} || $targets{man}) {
+       print "Running ascii, tag, and man tests ";
+       print "...\n" if $targets{verbose};
+}
+for my $test (@regress_tests) {
+       my $i = "$test->{NAME}.in";
+       my $o = "$test->{NAME}.mandoc_ascii";
+       my $w = "$test->{NAME}.out_ascii";
+       my $to = "$test->{NAME}.mandoc_tag";
+       my $tos = "$test->{NAME}.mandoc_tag_s";
+       my $tw = "$test->{NAME}.out_tag";
+       my $diff_ascii;
+       if ($targets{tag} && $tag_tests{$test->{NAME}} &&
+           $test->{NAME} =~ /^$onlytest/) {
+               $count_tag++;
+               $count_total++;
+               my @cmd = (qw(../man -l), @{$test->{MOPTS}},
+                   qw(-I os=OpenBSD -T ascii -O),
+                   "outfilename=$o,tagfilename=$to", "$i");
+               print "@cmd\n" if $targets{verbose};
+               system @cmd
+                   and fail $test->{NAME}, 'tag:man';
+               system "sed 's: .*/: :' $to > $tos";
+               system @diff, $tw, $tos
+                   and fail $test->{NAME}, 'tag:diff';
+               print "." unless $targets{verbose};
+               $diff_ascii = $targets{ascii};
+       } elsif ($targets{ascii} && !$skip_ascii{$test->{NAME}} &&
+           $test->{NAME} =~ /^$onlytest/) {
+               sysout $o, '../mandoc', @{$test->{MOPTS}},
+                   qw(-I os=OpenBSD -T ascii), $i
+                   and fail $test->{NAME}, 'ascii:mandoc';
+               $diff_ascii = 1;
+       }
+       if ($diff_ascii) {
                $count_ascii++;
                $count_total++;
-               print "@mandoc -T ascii $i\n" if $targets{verbose};
-               sysout $o, @mandoc, qw(-T ascii), $i
-                   and fail $subdir, $testname, 'ascii:mandoc';
                system @diff, $w, $o
-                   and fail $subdir, $testname, 'ascii:diff';
+                   and fail $test->{NAME}, 'ascii:diff';
+               print "." unless $targets{verbose};
        }
-       my $m = "$subdir/$testname.in_man";
-       my $mo = "$subdir/$testname.mandoc_man";
-       if ($targets{man} && !$skip_man{$testname}) {
+       my $m = "$test->{NAME}.in_man";
+       my $mo = "$test->{NAME}.mandoc_man";
+       if ($targets{man} && !$skip_man{$test->{NAME}} &&
+           $test->{NAME} =~ /^$onlytest/) {
                $count_man++;
                $count_total++;
-               print "@mandoc -T man $i\n" if $targets{verbose};
-               sysout $m, @mandoc, qw(-T man), $i
-                   and fail $subdir, $testname, 'man:man';
-               print "@mandoc -man -T ascii $m\n" if $targets{verbose};
-               sysout $mo, @mandoc, qw(-man -T ascii -O mdoc), $m
-                   and fail $subdir, $testname, 'man:mandoc';
+               sysout $m, '../mandoc', @{$test->{MOPTS}},
+                   qw(-I os=OpenBSD -T man), $i
+                   and fail $test->{NAME}, 'man:man';
+               sysout $mo, '../mandoc', @{$test->{MOPTS}},
+                   qw(-man -I os=OpenBSD -T ascii -O mdoc), $m
+                   and fail $test->{NAME}, 'man:mandoc';
                system @diff, $w, $mo
-                   and fail $subdir, $testname, 'man:diff';
+                   and fail $test->{NAME}, 'man:diff';
+               print "." unless $targets{verbose};
        }
        if ($targets{clean}) {
-               print "rm $o\n"
-                   if $targets{verbose} && !$skip_ascii{$testname};
-               unlink $o;
-               print "rm $m $mo\n"
-                   if $targets{verbose} && !$skip_man{$testname};
-               unlink $m, $mo;
+               print "rm $o $to $tos $m $mo\n" if $targets{verbose};
+               $count_rm += unlink $o, $to, $tos, $m, $mo;
        }
 }
+if ($targets{ascii} || $targets{tag} || $targets{man}) {
+       print "Number of ascii, tag, and man tests:" if $targets{verbose};
+       print " $count_ascii + $count_tag + $count_man tests run.\n";
+}
 
 my $count_utf8 = 0;
-for my $testname (@utf8_testnames) {
-       next if $onlytest && $testname ne $onlytest;
-       my $i = "$subdir/$testname.in";
-       my $o = "$subdir/$testname.mandoc_utf8";
-       my $w = "$subdir/$testname.out_utf8";
-       if ($targets{utf8}) {
+if ($targets{utf8}) {
+       print "Running utf8 tests ";
+       print "...\n" if $targets{verbose};
+}
+for my $test (@utf8_tests) {
+       my $i = "$test->{NAME}.in";
+       my $o = "$test->{NAME}.mandoc_utf8";
+       my $w = "$test->{NAME}.out_utf8";
+       if ($targets{utf8} && $test->{NAME} =~ /^$onlytest/o) {
                $count_utf8++;
                $count_total++;
-               print "@mandoc -T utf8 $i\n" if $targets{verbose};
-               sysout $o, @mandoc, qw(-T utf8), $i
-                   and fail $subdir, $testname, 'utf8:mandoc';
+               sysout $o, '../mandoc', @{$test->{MOPTS}},
+                   qw(-I os=OpenBSD -T utf8), $i
+                   and fail $test->{NAME}, 'utf8:mandoc';
                system @diff, $w, $o
-                   and fail $subdir, $testname, 'utf8:diff';
+                   and fail $test->{NAME}, 'utf8:diff';
+               print "." unless $targets{verbose};
        }
        if ($targets{clean}) {
                print "rm $o\n" if $targets{verbose};
-               unlink $o;
+               $count_rm += unlink $o;
        }
 }
+if ($targets{utf8}) {
+       print "Number of utf8 tests:" if $targets{verbose};
+       print " $count_utf8 tests run.\n";
+}
 
 my $count_html = 0;
-for my $testname (@html_testnames) {
-       next if $onlytest && $testname ne $onlytest;
-       my $i = "$subdir/$testname.in";
-       my $o = "$subdir/$testname.mandoc_html";
-       my $w = "$subdir/$testname.out_html";
-       if ($targets{html}) {
+if ($targets{html}) {
+       print "Running html tests ";
+       print "...\n" if $targets{verbose};
+}
+for my $test (@html_tests) {
+       my $i = "$test->{NAME}.in";
+       my $o = "$test->{NAME}.mandoc_html";
+       my $w = "$test->{NAME}.out_html";
+       if ($targets{html} && $test->{NAME} =~ /^$onlytest/) {
                $count_html++;
                $count_total++;
-               print "@mandoc -T html $i\n" if $targets{verbose};
-               syshtml $o, @mandoc, qw(-T html), $i
-                   and fail $subdir, $testname, 'html:mandoc';
+               syshtml $o, '../mandoc', @{$test->{MOPTS}},
+                   qw(-T html), $i
+                   and fail $test->{NAME}, 'html:mandoc';
                system @diff, $w, $o
-                   and fail $subdir, $testname, 'html:diff';
+                   and fail $test->{NAME}, 'html:diff';
+               print "." unless $targets{verbose};
        }
        if ($targets{clean}) {
                print "rm $o\n" if $targets{verbose};
-               unlink $o;
+               $count_rm += unlink $o;
        }
 }
+if ($targets{html}) {
+       print "Number of html tests:" if $targets{verbose};
+       print " $count_html tests run.\n";
+}
 
 my $count_markdown = 0;
-for my $testname (@regress_testnames) {
-       next if $onlytest && $testname ne $onlytest;
-       my $i = "$subdir/$testname.in";
-       my $o = "$subdir/$testname.mandoc_markdown";
-       my $w = "$subdir/$testname.out_markdown";
-       if ($targets{markdown} && !$skip_markdown{$testname}) {
+if ($targets{markdown}) {
+       print "Running markdown tests ";
+       print "...\n" if $targets{verbose};
+}
+for my $test (@regress_tests) {
+       my $i = "$test->{NAME}.in";
+       my $o = "$test->{NAME}.mandoc_markdown";
+       my $w = "$test->{NAME}.out_markdown";
+       if ($targets{markdown} && !$skip_markdown{$test->{NAME}} &&
+           $test->{NAME} =~ /^$onlytest/) {
                $count_markdown++;
                $count_total++;
-               print "@mandoc -T markdown $i\n" if $targets{verbose};
-               sysout $o, @mandoc, qw(-T markdown), $i
-                   and fail $subdir, $testname, 'markdown:mandoc';
+               sysout $o, '../mandoc', @{$test->{MOPTS}},
+                   qw(-I os=OpenBSD -T markdown), $i
+                   and fail $test->{NAME}, 'markdown:mandoc';
                system @diff, $w, $o
-                   and fail $subdir, $testname, 'markdown:diff';
+                   and fail $test->{NAME}, 'markdown:diff';
+               print "." unless $targets{verbose};
        }
        if ($targets{clean}) {
                print "rm $o\n" if $targets{verbose};
-               unlink $o;
+               $count_rm += unlink $o;
        }
 }
+if ($targets{markdown}) {
+       print "Number of markdown tests:" if $targets{verbose};
+       print " $count_markdown tests run.\n";
+}
 
 my $count_lint = 0;
-for my $testname (@lint_testnames) {
-       next if $onlytest && $testname ne $onlytest;
-       my $i = "$subdir/$testname.in";
-       my $o = "$subdir/$testname.mandoc_lint";
-       my $w = "$subdir/$testname.out_lint";
-       if ($targets{lint}) {
+if ($targets{lint}) {
+       print "Running lint tests ";
+       print "...\n" if $targets{verbose};
+}
+for my $test (@lint_tests) {
+       my $i = "$test->{NAME}.in";
+       my $o = "$test->{NAME}.mandoc_lint";
+       my $w = "$test->{NAME}.out_lint";
+       if ($targets{lint} && $test->{NAME} =~ /^$onlytest/) {
                $count_lint++;
                $count_total++;
-               print "@mandoc -T lint $i\n" if $targets{verbose};
-               syslint $o, @mandoc, qw(-T lint), $i
-                   and fail $subdir, $testname, 'lint:mandoc';
+               syslint $o, '../mandoc', @{$test->{MOPTS}},
+                   qw(-I os=OpenBSD -T lint -W all), $i
+                   and fail $test->{NAME}, 'lint:mandoc';
                system @diff, $w, $o
-                   and fail $subdir, $testname, 'lint:diff';
+                   and fail $test->{NAME}, 'lint:diff';
+               print "." unless $targets{verbose};
        }
        if ($targets{clean}) {
                print "rm $o\n" if $targets{verbose};
-               unlink $o;
+               $count_rm += unlink $o;
        }
 }
-
-exit 0 unless $displaylevel or @failures;
-
-print "\n" if $targets{verbose};
-if ($onlytest) {
-       print "test $subdir:$onlytest finished";
-} else {
-       print "testsuite $subdir finished";
+if ($targets{lint}) {
+       print "Number of lint tests:" if $targets{verbose};
+       print " $count_lint tests run.\n";
 }
-print ' ', (scalar @subdir_names), ' subdirectories' if @subdir_names;
-print " $count_ascii ascii" if $count_ascii;
-print " $count_man man" if $count_man;
-print " $count_utf8 utf8" if $count_utf8;
-print " $count_html html" if $count_html;
-print " $count_markdown markdown" if $count_markdown;
-print " $count_lint lint" if $count_lint;
+
+# --- final report -----------------------------------------------------
 
 if (@failures) {
-       print " (FAIL)\n\nSOME TESTS FAILED:\n\n";
+       print "\nNUMBER OF FAILED TESTS: ", scalar @failures,
+           " (of $count_total tests run.)\n";
        print "@$_\n" for @failures;
        print "\n";
        exit 1;
-} elsif ($count_total == 1) {
-       print " (OK)\n";
+}
+print "\n" if $targets{verbose};
+if ($count_total == 1) {
+       print "Test succeeded.\n";
 } elsif ($count_total) {
-       print " (all $count_total tests OK)\n";
+       print "All $count_total tests OK:";
+       print " $count_ascii ascii" if $count_ascii;
+       print " $count_tag tag" if $count_tag;
+       print " $count_man man" if $count_man;
+       print " $count_utf8 utf8" if $count_utf8;
+       print " $count_html html" if $count_html;
+       print " $count_markdown markdown" if $count_markdown;
+       print " $count_lint lint" if $count_lint;
+       print "\n";
 } else {
-       print " (no tests run)\n";
-} 
+       print "No tests were run.\n";
+}
+if ($targets{clean}) {
+       if ($count_rm) {
+               print "Deleted $count_rm test output files.\n";
+               print "The tree is now clean.\n";
+       } else {
+               print "No test output files were found.\n";
+               print "The tree was already clean.\n";
+       }
+}
 exit 0;