#!./perl
my
$t0
=
time
();
my
$deparse_failures
;
my
$deparse_skips
;
my
$deparse_skip_file
=
'../Porting/deparse-skips.txt'
;
my
%dir_to_switch
=
(
base
=>
''
,
comp
=>
''
,
run
=>
''
,
'../ext/File-Glob/t'
=>
'-I.. -MTestInit'
,
);
my
%abs
= (
'../cpan/Archive-Tar'
=> 1,
'../cpan/AutoLoader'
=> 1,
'../cpan/CPAN'
=> 1,
'../cpan/Encode'
=> 1,
'../cpan/ExtUtils-Constant'
=> 1,
'../cpan/ExtUtils-Install'
=> 1,
'../cpan/ExtUtils-MakeMaker'
=> 1,
'../cpan/ExtUtils-Manifest'
=> 1,
'../cpan/File-Fetch'
=> 1,
'../cpan/IPC-Cmd'
=> 1,
'../cpan/IPC-SysV'
=> 1,
'../cpan/Module-Load'
=> 1,
'../cpan/Module-Load-Conditional'
=> 1,
'../cpan/Pod-Simple'
=> 1,
'../cpan/Test-Simple'
=> 1,
'../cpan/podlators'
=> 1,
'../dist/Cwd'
=> 1,
'../dist/Devel-PPPort'
=> 1,
'../dist/ExtUtils-ParseXS'
=> 1,
'../dist/Tie-File'
=> 1,
);
my
%temp_no_core
= (
'../cpan/Compress-Raw-Bzip2'
=> 1,
'../cpan/Compress-Raw-Zlib'
=> 1,
'../cpan/Devel-PPPort'
=> 1,
'../cpan/Getopt-Long'
=> 1,
'../cpan/IO-Compress'
=> 1,
'../cpan/MIME-Base64'
=> 1,
'../cpan/parent'
=> 1,
'../cpan/Pod-Simple'
=> 1,
'../cpan/podlators'
=> 1,
'../cpan/Test-Simple'
=> 1,
'../cpan/Tie-RefHash'
=> 1,
'../cpan/Unicode-Collate'
=> 1,
'../dist/Unicode-Normalize'
=> 1,
);
my
@bad_env_vars
=
qw(
PERL5LIB PERLLIB PERL5OPT PERL_UNICODE
PERL_YAML_BACKEND PERL_JSON_BACKEND
)
;
for
my
$envname
(
@bad_env_vars
) {
my
$override
=
$ENV
{
"${envname}_TEST"
};
if
(
defined
$override
) {
warn
"$0: $envname=$override\n"
;
$ENV
{
$envname
} =
$override
;
}
else
{
delete
$ENV
{
$envname
};
}
}
our
$Valgrind_Log
;
my
%skip
= (
'.'
=> 1,
'..'
=> 1,
'CVS'
=> 1,
'RCS'
=> 1,
'SCCS'
=> 1,
'.svn'
=> 1,
);
if
($::do_nothing) {
return
1;
}
$| = 1;
my
$OS
=
$ENV
{FAKE_OS} || $^O;
my
$is_vms
=
$OS
eq
"VMS"
;
my
$is_win32
=
$OS
eq
"MSWin32"
;
my
$is_os2
=
$OS
eq
"os2"
;
@ARGV
=
grep
(
$_
,
@ARGV
)
if
$is_vms
;
our
$show_elapsed_time
=
$ENV
{HARNESS_TIMER} || 0;
my
$dump_tests
= 0;
{
my
%opt_vars
= (
benchmark
=> \$::benchmark,
core
=> \$::core,
v
=> \$::verbose,
torture
=> \$::torture,
utf8
=> \$::with_utf8,
utf16
=> \$::with_utf16,
taintwarn
=> \$::taintwarn,
dumptests
=> \
$dump_tests
,
);
my
@argv
= ();
foreach
my
$idx
(0..
$#ARGV
) {
my
$opt
;
if
(
$ARGV
[
$idx
] =~ /^-?-(\S+)$/) {
$opt
= $1;
}
else
{
push
@argv
,
$ARGV
[
$idx
];
next
;
}
if
(
my
$ref
=
$opt_vars
{
$opt
}) {
$$ref
= 1;
}
elsif
(
$opt
=~ /^deparse(,.+)?$/) {
$::deparse = 1;
$::deparse_opts = $1;
_process_deparse_config();
}
else
{
die
"Unknown option '$opt'\n"
;
}
}
@ARGV
=
@argv
;
}
chdir
't'
if
-f
't/TEST'
;
if
(-f
'TEST'
&& -f
'harness'
&& -d
'../lib'
) {
@INC
=
'../lib'
;
}
die
"You need to run \"make test_prep\" first to set things up.\n"
unless
-e
'perl'
or -e
'perl.exe'
or -e
'perl.pm'
;
if
(
$is_win32
) {
my
@argv
;
if
(
eval
'@argv = map glob, @ARGV; 1'
) {
@ARGV
=
@argv
;
}
else
{
die
"Failed to glob \@ARGV: $@"
;
}
}
$ENV
{PERL_DESTRUCT_LEVEL} = 2
unless
exists
$ENV
{PERL_DESTRUCT_LEVEL};
$ENV
{PERL_DL_NONLAZY} = 1
unless
exists
$ENV
{PERL_DL_NONLAZY};
$ENV
{EMXSHELL} =
'sh'
;
my
%timings
= ();
our
@found
;
sub
_find_tests {
@found
=();
push
@ARGV
, _find_files(
'\.t$'
,
$_
[0]) }
sub
_find_files {
my
(
$patt
,
@dirs
) =
@_
;
for
my
$dir
(
@dirs
) {
opendir
DIR,
$dir
or
die
"Trouble opening $dir: $!"
;
foreach
my
$f
(
sort
{
$a
cmp
$b
}
readdir
DIR) {
next
if
$skip
{
$f
};
$dir
=~ s/(?<!\^)\.dir(;1)?$//i
if
$is_vms
;
my
$fullpath
=
"$dir/$f"
;
if
(-d
$fullpath
) {
_find_files(
$patt
,
$fullpath
);
}
elsif
(
$f
=~ /
$patt
/) {
push
@found
,
$fullpath
;
}
}
}
@found
;
}
sub
_scan_test {
my
(
$test
,
$type
) =
@_
;
open
(
my
$script
,
"<"
,
$test
) or
die
"Can't read $test.\n"
;
my
$first_line
= <
$script
>;
$first_line
=~
tr
/\0//d
if
$::with_utf16;
my
$switch
=
""
;
if
(
$first_line
=~ /
$switch
=
"-$1"
;
}
else
{
if
($::taintwarn) {
$switch
=
'-t'
;
}
else
{
$switch
=
''
;
}
}
my
$file_opts
=
""
;
if
(
$type
eq
'deparse'
) {
while
(<
$script
>) {
$file_opts
=
$file_opts
.
",-f$3$4"
if
/^
}
}
close
$script
;
my
$perl
=
$is_win32
?
'.\perl'
:
'./perl'
;
my
$lib
=
'../lib'
;
my
$run_dir
;
my
$return_dir
;
$test
=~ /^(.+)\/[^\/]+/;
my
$dir
= $1;
my
$testswitch
=
$dir_to_switch
{
$dir
};
if
(!
defined
$testswitch
) {
if
(
$test
=~ s!^(\.\./(cpan|dist|ext)/[^/]+)/t!t!) {
$run_dir
= $1;
$return_dir
=
'../../t'
;
$lib
=
'../../lib'
;
$perl
=
'../../t/perl'
;
$testswitch
=
"-I../.. -MTestInit=U2T"
;
if
($2 eq
'cpan'
|| $2 eq
'dist'
) {
if
(
$abs
{
$run_dir
}) {
$testswitch
=
$testswitch
.
',A'
;
}
if
(
$temp_no_core
{
$run_dir
}) {
$testswitch
=
$testswitch
.
',NC'
;
}
}
}
elsif
(
$test
=~ m!^\.\./lib!) {
$testswitch
=
'-I.. -MTestInit=U1'
;
}
else
{
$testswitch
=
'-I.. -MTestInit'
;
}
}
my
$utf8
= ($::with_utf8 || $::with_utf16) ?
"-I$lib -Mutf8"
:
''
;
my
%options
= (
perl
=>
$perl
,
lib
=>
$lib
,
test
=>
$test
,
run_dir
=>
$run_dir
,
return_dir
=>
$return_dir
,
testswitch
=>
$testswitch
,
utf8
=>
$utf8
,
file
=>
$file_opts
,
switch
=>
$switch
,
);
return
\
%options
;
}
sub
_cmd {
my
(
$options
,
$type
) =
@_
;
my
$test
=
$options
->{test};
my
$cmd
;
if
(
$type
eq
'deparse'
) {
my
$perl
=
"$options->{perl} $options->{testswitch}"
;
my
$lib
=
$options
->{lib};
$cmd
= (
"$perl $options->{switch} -I$lib -MO=-qq,Deparse,-sv1.,"
.
"-l$::deparse_opts$options->{file} "
.
"$test > $test.dp "
.
"&& $perl $options->{switch} -I$lib $test.dp"
);
}
elsif
(
$type
eq
'perl'
) {
my
$perl
=
$options
->{perl};
my
$redir
=
$is_vms
?
'2>&1'
:
''
;
if
(
$ENV
{PERL_VALGRIND}) {
my
$perl_supp
=
$options
->{return_dir} ?
"$options->{return_dir}/perl.supp"
:
"perl.supp"
;
my
$valgrind_exe
=
$ENV
{VALGRIND} //
'valgrind'
;
if
(
$options
->{run_dir}) {
$Valgrind_Log
= Cwd::abs_path(
"$options->{run_dir}/$Valgrind_Log"
);
}
my
$vg_opts
=
$ENV
{VG_OPTS}
//
"--log-file=$Valgrind_Log "
.
"--suppressions=$perl_supp --leak-check=yes "
.
"--leak-resolution=high --show-reachable=yes "
.
"--num-callers=50 --track-origins=yes"
;
if
(
$vg_opts
!~ /--
log
-file/) {
$vg_opts
=
"--log-file=$Valgrind_Log $vg_opts"
;
}
$perl
=
"$valgrind_exe $vg_opts $perl"
;
}
my
$args
=
"$options->{testswitch} $options->{switch} $options->{utf8}"
;
$cmd
=
$perl
. _quote_args(
$args
) .
" $test $redir"
;
}
return
$cmd
;
}
sub
_before_fork {
my
(
$options
) =
@_
;
if
(
$options
->{run_dir}) {
my
$run_dir
=
$options
->{run_dir};
chdir
$run_dir
or
die
"Can't chdir to '$run_dir': $!"
;
}
my
$test
=
$options
->{test};
(
local
$Valgrind_Log
=
"$test.valgrind-current"
) =~ s/^.*\///;
if
(
$ENV
{PERL_VALGRIND} && -e
$Valgrind_Log
) {
unlink
$Valgrind_Log
or
warn
"$0: Failed to unlink '$Valgrind_Log': $!\n"
;
}
return
;
}
sub
_after_fork {
my
(
$options
) =
@_
;
if
(
$options
->{return_dir}) {
my
$return_dir
=
$options
->{return_dir};
chdir
$return_dir
or
die
"Can't chdir from '$options->{run_dir}' to '$return_dir': $!"
;
}
return
;
}
sub
_run_test {
my
(
$test
,
$type
) =
@_
;
my
$options
= _scan_test(
$test
,
$type
);
_before_fork(
$options
);
my
$cmd
= _cmd(
$options
,
$type
);
open
(
my
$results
,
"$cmd |"
) or
print
"can't run '$cmd': $!.\n"
;
_after_fork(
$options
);
binmode
$results
;
return
$results
;
}
sub
_quote_args {
my
(
$args
) =
@_
;
my
$argstring
=
''
;
foreach
(
split
(/\s+/,
$args
)) {
$_
=
q(")
.
$_
.
q(")
if
$is_vms
&& !/^\"/ &&
length
(
$_
) > 0;
$argstring
=
$argstring
.
' '
.
$_
;
}
return
$argstring
;
}
sub
_populate_hash {
return
unless
defined
$_
[0];
return
map
{
$_
, 1}
split
/\s+/,
$_
[0];
}
sub
_tests_from_manifest {
my
(
$extensions
,
$known_extensions
,
$all
) =
@_
;
s/\bCwd\b/PathTools/, s!\bList/Util\b!Scalar/List/Utils!
for
$extensions
,
$known_extensions
;
my
%skip
;
my
%extensions
= _populate_hash(
$extensions
);
my
%known_extensions
= _populate_hash(
$known_extensions
);
my
%printed_skip_warning
;
foreach
(
keys
%known_extensions
) {
$skip
{
$_
} = 1
unless
$extensions
{
$_
};
}
my
@results
;
my
%non_ext
;
push
@results
, \
%non_ext
if
$all
;
my
$mani
=
'../MANIFEST'
;
if
(
open
(MANI,
$mani
)) {
while
(<MANI>) {
chomp
;
my
(
$file
)=
split
/\t/,
$_
;
if
(
$file
=~ m!^((?:cpan|dist|ext)/(\S+)/+(?:[^/\s]+\.t|test\.pl)|lib/\S+?(?:\.t|test\.pl))\z!) {
my
$t
= $1;
my
$extension
= $2;
if
(
ord
"A"
!= 65
&&
defined
$extension
&&
$extension
=~ m! \b (?:
Archive-Tar/
| Config-Perl-V/
| CPAN-Meta/
| CPAN-Meta-YAML/
| Digest-SHA/
| ExtUtils-MakeMaker/
| HTTP-Tiny/
| IO-Compress/
| JSON-PP/
| libnet/
| MIME-Base64/
| podlators/
| Pod-Simple/
| Pod-Checker/
| Digest-MD5/
| Test-Harness/
| IPC-Cmd/
| Encode/
| Socket/
| ExtUtils-Manifest/
| Module-Metadata/
| PerlIO-via-QuotedPrint/
)
!x)
{
print
STDERR
"Skipping testing of $extension on EBCDIC\n"
unless
$printed_skip_warning
{
$extension
}++;
next
;
}
if
(!$::core ||
$t
=~ m!^lib/[a-z]!) {
if
(
defined
$extension
) {
$extension
=~ s!/t(:?/\S+)*$!!;
next
if
$skip
{
$extension
};
my
$flat_extension
=
$extension
;
$flat_extension
=~ s!-!/!g;
next
if
$skip
{
$flat_extension
};
}
my
$path
=
"../$t"
;
push
@results
,
$path
;
$::path_to_name{
$path
} =
$t
;
}
}
elsif
(
$file
=~m!/(?:test\.pl|[^/\s]+\.t)\z! and
$file
ne
"t/test.pl"
) {
my
$munged
=
$file
;
next
if
$munged
=~m!^(?:t/)?os2/! and !
$is_os2
;
next
if
$munged
=~m!^(?:t/)?win32/! and !
$is_win32
;
next
if
$munged
=~m!^(?:t/)?japh/! and !($::torture or
$ENV
{PERL_TORTURE_TEST});
next
if
$munged
=~m!^(?:t/)?benchmark/! and !($::benchmark or
$ENV
{PERL_BENCHMARK});
next
if
$munged
=~m!^(?:t/)?bigmem/! and !
$ENV
{PERL_TEST_MEMORY};
$munged
=~ s!t/!! or
$munged
=
"../$munged"
;
$non_ext
{
$munged
}++;
}
}
close
MANI;
}
else
{
warn
"$0: cannot open $mani: $!\n"
;
}
return
@results
;
}
sub
dump_tests {
my
(
$ary
) =
@_
;
for
my
$test
(
sort
@$ary
) {
$test
=~s!^\.\./!! or
$test
=~s!^!t/!;
print
"$test\n"
;
}
exit
(0);
}
sub
filter_taint_tests {
my
$tests
=
shift
;
return
unless
$Config::Config
{taint_disabled} eq
"define"
;
my
%known_tainter
=
map
{
$_
=> 0 } (
'../cpan/Test-Harness/t/regression.t'
,
'../cpan/Test-Harness/t/source_handler.t'
,
'../cpan/Test-Harness/t/compat/inc-propagation.t'
,
);
@$tests
=
grep
{
my
$file
=
$_
;
open
my
$ifh
,
"<"
,
$file
or
die
"Failed to read: '$file': $!"
;
my
$line
= <
$ifh
>;
my
$keep
=
$file
=~/taint/ ? 0 : (
$known_tainter
{
$file
} // 1);
if
(
$line
=~/^
my
$switch
= $1;
if
(
$switch
=~ s/[Tt]//) {
$keep
= 0;
}
}
$keep
}
@$tests
;
}
unless
(
@ARGV
) {
foreach
my
$dir
(
qw(base comp run cmd io re opbasic op uni mro class perf test_pl)
) {
_find_tests(
$dir
);
}
unless
($::core) {
_find_tests(
'porting'
);
_find_tests(
"lib"
);
}
_find_tests(
'win32'
)
if
$is_win32
;
_find_tests(
'os2'
)
if
$is_os2
;
my
$configsh
=
'../config.sh'
;
my
(
$extensions
,
$known_extensions
);
if
(-f
$configsh
) {
open
FH,
$configsh
or
die
"Can't open $configsh: $!"
;
while
(<FH>) {
if
(/^extensions=[
'"](.*)['
"]$/) {
$extensions
= $1;
}
elsif
(/^known_extensions=[
'"](.*)['
"]$/) {
$known_extensions
= $1;
}
}
if
(!
defined
$known_extensions
) {
warn
"No known_extensions line found in $configsh"
;
}
if
(!
defined
$extensions
) {
warn
"No extensions line found in $configsh"
;
}
}
push
@ARGV
, _tests_from_manifest(
$extensions
,
$known_extensions
);
unless
($::core) {
_find_tests(
'japh'
)
if
$::torture or
$ENV
{PERL_TORTURE_TEST};
_find_tests(
'benchmark'
)
if
$::benchmark or
$ENV
{PERL_BENCHMARK};
_find_tests(
'bigmem'
)
if
$ENV
{PERL_TEST_MEMORY};
}
}
@ARGV
=
do
{
my
@order
= (
"test_pl"
,
"base"
,
"comp"
,
"run"
,
"cmd"
,
"io"
,
"re"
,
"opbasic"
,
"op"
,
"op/hook"
,
"uni"
,
"mro"
,
"class"
,
"lib"
,
"ext"
,
"dist"
,
"cpan"
,
"perf"
,
"porting"
,
);
my
%order
=
map
{
$order
[
$_
] => 1+
$_
} 0..
$#order
;
my
$idx
= 0;
map
{
$_
->[0]
}
sort
{
$a
->[3] <=>
$b
->[3] ||
$a
->[1] <=>
$b
->[1]
}
map
{
my
$root
= /(\w+)/ ? $1 :
""
;
[
$_
,
$idx
++,
$root
,
$order
{
$root
}||=0 ]
}
@ARGV
;
};
dump_tests(\
@ARGV
)
if
$dump_tests
;
filter_taint_tests(\
@ARGV
);
if
($::deparse) {
_testprogs(
'deparse'
,
''
,
@ARGV
);
}
elsif
($::with_utf16) {
for
my
$e
(0, 1) {
for
my
$b
(0, 1) {
print
STDERR
"# ENDIAN $e BOM $b\n"
;
my
@UARGV
;
for
my
$a
(
@ARGV
) {
my
$u
=
$a
.
"."
. (
$e
?
"l"
:
"b"
) .
"e"
. (
$b
?
"b"
:
""
);
my
$f
=
$e
?
"v"
:
"n"
;
push
@UARGV
,
$u
;
unlink
(
$u
);
if
(
open
(A,
$a
)) {
if
(
open
(U,
">$u"
)) {
print
U
pack
(
"$f"
, 0xFEFF)
if
$b
;
while
(<A>) {
print
U
pack
(
"$f*"
,
unpack
(
"C*"
,
$_
));
}
close
(U);
}
close
(A);
}
}
_testprogs(
'perl'
,
''
,
@UARGV
);
unlink
(
@UARGV
);
}
}
}
else
{
_testprogs(
'perl'
,
''
,
@ARGV
);
}
sub
_testprogs {
my
(
$type
,
$args
,
@tests
) =
@_
;
print
<<'EOT' if ($type eq 'deparse');
------------------------------------------------------------------------------
TESTING DEPARSER
------------------------------------------------------------------------------
EOT
$::bad_files = 0;
foreach
my
$t
(
@tests
) {
unless
(
exists
$::path_to_name{
$t
}) {
my
$tname
=
"t/$t"
;
$::path_to_name{
$t
} =
$tname
;
}
}
my
$maxlen
= 0;
foreach
(@::path_to_name{
@tests
}) {
s/\.\w+\z/ /;
my
$len
=
length
;
$maxlen
=
$len
if
$len
>
$maxlen
;
}
my
$dotdotdot
=
$maxlen
+ 3 ;
my
$grind_ct
= 0;
my
$total_files
=
@tests
;
my
$good_files
= 0;
my
$tested_files
= 0;
my
$totmax
= 0;
my
%failed_tests
;
my
@unexpected_pass
;
my
$toolnm
;
while
(
my
$test
=
shift
@tests
) {
my
(
$test_start_time
,
@starttimes
) = 0;
if
(
$show_elapsed_time
) {
$test_start_time
= Time::HiRes::
time
();
@starttimes
=
times
;
}
if
(
$test
=~ /^$/) {
next
;
}
if
(
$type
eq
'deparse'
&&
$test
=~
$deparse_skips
) {
next
;
}
my
$te
= $::path_to_name{
$test
} .
'.'
x (
$dotdotdot
-
length
($::path_to_name{
$test
})) .
' '
;
if
(!
$is_vms
) {
print
$te
;
$te
=
''
;
}
(
local
$Valgrind_Log
=
"$test.valgrind-current"
) =~ s/^.*\///;
my
$results
= _run_test(
$test
,
$type
);
my
$failure
;
my
$next
= 0;
my
$seen_leader
= 0;
my
$seen_ok
= 0;
my
$trailing_leader
= 0;
my
$max
;
my
%todo
;
while
(<
$results
>) {
next
if
/^\s*$/;
if
(/^1..$/ &&
$is_vms
) {
my
$l2
= <
$results
>;
if
(
$l2
=~ /^\s*$/) {
$l2
= <
$results
>;
}
$_
=
'1..'
.
$l2
;
}
if
($::verbose) {
print
$_
;
}
unless
(/^\
if
(
$trailing_leader
) {
$failure
=
'FAILED--extra output after trailing 1..n'
;
last
;
}
if
(/^1\.\.([0-9]+)( todo ([\d ]+))?/) {
if
(
$seen_leader
) {
$failure
=
'FAILED--seen duplicate leader'
;
last
;
}
$max
= $1;
%todo
=
map
{
$_
=> 1 }
split
/ /, $3
if
$3;
$totmax
=
$totmax
+
$max
;
$tested_files
=
$tested_files
+ 1;
if
(
$seen_ok
) {
$trailing_leader
= 1;
if
(
$next
!=
$max
) {
$failure
=
"FAILED--expected $max tests, saw $next"
;
last
;
}
}
else
{
$next
= 0;
}
$seen_leader
= 1;
}
else
{
if
(/^(not )?ok(?: (\d+))?[^\
unless
(
$seen_leader
) {
unless
(
$seen_ok
) {
$next
= 0;
}
}
$seen_ok
= 1;
$next
=
$next
+ 1;
my
(
$not
,
$num
,
$extra
,
$istodo
) = ($1, $2, $3, 0);
$num
=
$next
unless
$num
;
if
(
$num
==
$next
) {
$extra
and
$istodo
=
$extra
=~ /
$istodo
= 1
if
$todo
{
$num
};
if
(
$not
&& !
$istodo
) {
$failure
=
"FAILED at test $num"
;
last
;
}
}
else
{
$failure
=
"FAILED--expected test $next, saw test $num"
;
last
;
}
}
elsif
(/^Bail out!\s*(.*)/i) {
die
"FAILED--Further testing stopped"
. ($1 ?
": $1\n"
:
".\n"
);
}
else
{
next
if
$test
=~ /^\W*(cpan|dist|ext|lib)\b/;
$failure
=
"FAILED--unexpected output at test $next"
;
last
;
}
}
}
}
my
@junk
= <
$results
>;
close
$results
;
undef
@junk
;
if
(not
defined
$failure
) {
$failure
=
'FAILED--no leader found'
unless
$seen_leader
;
}
_check_valgrind(\
$toolnm
, \
$grind_ct
, \
$test
);
if
(
$type
eq
'deparse'
&& !
$ENV
{KEEP_DEPARSE_FILES}) {
unlink
"./$test.dp"
;
}
if
(not
defined
$failure
and
$next
!=
$max
) {
$failure
=
"FAILED--expected $max tests, saw $next"
;
}
if
( !
defined
$failure
and $? )
{
$failure
=
"FAILED--non-zero wait status: $?"
;
}
if
(
$type
eq
'deparse'
&&
$test
=~
$deparse_failures
) {
if
(!
$failure
) {
push
@unexpected_pass
,
$test
;
}
else
{
print
"${te}skipped\n"
;
$tested_files
=
$tested_files
- 1;
next
;
}
}
if
(
defined
$failure
) {
print
"${te}$failure\n"
;
$::bad_files = $::bad_files + 1;
if
(
$test
=~ /^base/ && !
defined
&DynaLoader::boot_DynaLoader
) {
die
"Failed a basic test ($test) under minitest -- cannot continue.\n"
;
}
$failed_tests
{
$test
} = 1;
}
else
{
if
(
$max
) {
my
(
$elapsed
,
$etms
) = (
""
, 0);
if
(
$show_elapsed_time
) {
$etms
= (Time::HiRes::
time
() -
$test_start_time
) * 1000;
$elapsed
=
sprintf
(
" %8.0f ms"
,
$etms
);
my
(
@endtimes
) =
times
;
$endtimes
[
$_
] -=
$starttimes
[
$_
]
for
0..
$#endtimes
;
splice
@endtimes
, 0, 2;
$_
*= 1000
for
@endtimes
;
$timings
{
$test
} = [
$etms
,
@endtimes
];
$elapsed
.=
sprintf
(
" %5.0f ms"
,
$_
)
for
@endtimes
;
}
print
"${te}ok$elapsed\n"
;
$good_files
=
$good_files
+ 1;
}
else
{
print
"${te}skipped\n"
;
$tested_files
=
$tested_files
- 1;
}
}
}
if
($::bad_files == 0) {
if
(
$good_files
) {
print
"All tests successful.\n"
;
}
else
{
die
"FAILED--no tests were run for some reason.\n"
;
}
}
else
{
my
$pct
=
$tested_files
?
sprintf
(
"%.2f"
, (
$tested_files
- $::bad_files) /
$tested_files
* 100) :
"0.00"
;
my
$s
= $::bad_files == 1 ?
""
:
"s"
;
warn
"Failed $::bad_files test$s out of $tested_files, $pct% okay.\n"
;
for
my
$test
(
sort
keys
%failed_tests
) {
print
"\t$test\n"
;
}
if
(
@unexpected_pass
) {
print
<<EOF;
The following scripts were expected to fail under -deparse (at least
according to $deparse_skip_file), but unexpectedly succeeded:
EOF
print
"\t$_\n"
for
sort
@unexpected_pass
;
print
"\n"
;
}
warn
<<'SHRDLU_1';
### Since not all tests were successful, you may want to run some of
### them individually and examine any diagnostic messages they produce.
### See the INSTALL document's section on "make test".
SHRDLU_1
warn
<<'SHRDLU_2' if $good_files / $total_files > 0.8;
### You have a good chance to get more information by running
### ./perl harness
### in the 't' directory since most (>=80%) of the tests succeeded.
SHRDLU_2
if
(
eval
{
require
Config; Config->
import
; 1}) {
if
($::Config{usedl} && (
my
$p
= $::Config{ldlibpthname})) {
warn
<<SHRDLU_3;
### You may have to set your dynamic library search path,
### $p, to point to the build directory:
SHRDLU_3
if
(
exists
$ENV
{
$p
} &&
$ENV
{
$p
} ne
''
) {
warn
<<SHRDLU_4a;
### setenv $p `pwd`:\$$p; cd t; ./perl harness
### $p=`pwd`:\$$p; export $p; cd t; ./perl harness
### export $p=`pwd`:\$$p; cd t; ./perl harness
SHRDLU_4a
}
else
{
warn
<<SHRDLU_4b;
### setenv $p `pwd`; cd t; ./perl harness
### $p=`pwd`; export $p; cd t; ./perl harness
### export $p=`pwd`; cd t; ./perl harness
SHRDLU_4b
}
warn
<<SHRDLU_5;
### for csh-style shells, like tcsh; or for traditional/modern
### Bourne-style shells, like bash, ksh, and zsh, respectively.
SHRDLU_5
}
}
}
printf
"Elapsed: %d sec\n"
,
time
() -
$t0
;
my
(
$user
,
$sys
,
$cuser
,
$csys
) =
times
;
my
$tot
=
sprintf
(
"u=%.2f s=%.2f cu=%.2f cs=%.2f scripts=%d tests=%d"
,
$user
,
$sys
,
$cuser
,
$csys
,
$tested_files
,
$totmax
);
print
"$tot\n"
;
if
(
$good_files
) {
if
(-d
$show_elapsed_time
) {
my
@dt
=
localtime
;
$dt
[5] += 1900;
$dt
[4] += 1;
my
$fn
=
"$show_elapsed_time/"
.
join
(
'-'
,
@dt
[5,4,3,2,1]).
".ttimes"
;
Storable::store({
perf
=> \
%timings
,
gather_conf_platform_info(),
total
=>
$tot
,
},
$fn
);
print
"wrote storable file: $fn\n"
;
}
}
_cleanup_valgrind(\
$toolnm
, \
$grind_ct
);
}
exit
($::bad_files != 0);
sub
gather_conf_platform_info {
my
(
%conf
,
@platform
) = ();
$conf
{
$_
} =
$Config::Config
{
$_
}
for
grep
/cc|git|config_arg\d+/,
keys
%Config::Config
;
if
(-f
'/proc/cpuinfo'
) {
open
my
$fh
,
'/proc/cpuinfo'
or
warn
"$!: /proc/cpuinfo\n"
;
@platform
=
grep
/name|cpu/, <
$fh
>;
chomp
$_
for
@platform
;
}
unshift
@platform
,
$OS
;
return
(
conf
=> \
%conf
,
platform
=> {
cpu
=> \
@platform
,
mem
=> [
grep
s/\s+/ /,
grep
chomp
, `free` ],
load
=> [
grep
chomp
, `uptime` ],
},
host
=> (
grep
chomp
, `hostname -f`),
version
=>
'0.03'
,
);
}
sub
_check_valgrind {
return
unless
$ENV
{PERL_VALGRIND};
my
(
$toolnm
,
$grind_ct
,
$test
) =
@_
;
$$toolnm
=
$ENV
{VALGRIND};
$$toolnm
=~ s|.*/||;
my
@valgrind
;
if
(-e
$Valgrind_Log
) {
if
(
open
(V,
$Valgrind_Log
)) {
@valgrind
= <V>;
close
V;
}
else
{
warn
"$0: Failed to open '$Valgrind_Log': $!\n"
;
}
}
if
(
$ENV
{VG_OPTS} =~ /(cachegrind)/ or
$$toolnm
=~ /(perf)/) {
$$toolnm
= $1;
if
(
$$toolnm
eq
'perf'
) {
my
(
$sub
) =
split
/\s/,
$ENV
{VG_OPTS};
$$toolnm
.=
"-$sub"
;
}
if
(
rename
$Valgrind_Log
,
"$$test.$$toolnm"
) {
$$grind_ct
++;
}
else
{
warn
"$0: Failed to create '$$test.$$toolnm': $!\n"
;
}
}
elsif
(
@valgrind
) {
my
$leaks
= 0;
my
$errors
= 0;
for
my
$i
(0..
$#valgrind
) {
local
$_
=
$valgrind
[
$i
];
if
(/^==\d+== ERROR SUMMARY: (\d+) errors? /) {
$errors
=
$errors
+ $1;
}
elsif
(/^==\d+== LEAK SUMMARY:/) {
for
my
$off
(1 .. 4) {
if
(
$valgrind
[
$i
+
$off
] =~
/(?:lost|reachable):\s+\d+ bytes in (\d+) blocks/) {
$leaks
=
$leaks
+ $1;
}
}
}
}
if
(
$errors
or
$leaks
) {
if
(
rename
$Valgrind_Log
,
"$$test.valgrind"
) {
$$grind_ct
=
$$grind_ct
+ 1;
}
else
{
warn
"$0: Failed to create '$$test.valgrind': $!\n"
;
}
}
}
else
{
if
(
$ENV
{VG_OPTS} &&
$ENV
{VG_OPTS} !~ /(^|\s)(-
q|--quiet)(\s|
$)/) {
warn
"No valgrind output?\n"
;
}
}
if
(-e
$Valgrind_Log
) {
unlink
$Valgrind_Log
or
warn
"$0: Failed to unlink '$Valgrind_Log': $!\n"
;
}
}
sub
_cleanup_valgrind {
return
unless
$ENV
{PERL_VALGRIND};
my
(
$toolnm
,
$grind_ct
) =
@_
;
my
$s
=
$$grind_ct
== 1 ?
''
:
's'
;
print
"$$grind_ct valgrind report$s created.\n"
, ;
if
(
$$toolnm
eq
'cachegrind'
) {
unlink
_find_files(
'cachegrind.out.\d+$'
,
qw (
../t ../cpan ../ext ../dist/ ));
}
elsif
(
$$toolnm
eq
'valgrind'
) {
unlink
grep
{ -z } _find_files(
'valgrind-current'
,
qw (
../t ../cpan ../ext ../dist/ ));
}
}
sub
_process_deparse_config {
my
@deparse_failures
;
my
@deparse_skips
;
my
$f
=
$deparse_skip_file
;
my
$skips
;
if
(!
open
(
$skips
,
'<'
,
$f
)) {
warn
"Failed to find $f: $!\n"
;
return
;
}
my
$in
;
while
(<
$skips
>) {
if
(/__DEPARSE_FAILURES__/) {
$in
= \
@deparse_failures
;
next
;
}
elsif
(/__DEPARSE_SKIPS__/) {
$in
= \
@deparse_skips
;
next
;
}
elsif
(!
$in
) {
next
;
}
s/
s/\s+$//;
next
unless
$_
;
push
@$in
,
$_
;
warn
"WARNING: $f:$.: excluded file doesn't exist: $_\n"
unless
-f
$_
;
}
for
my
$f
(
@deparse_failures
,
@deparse_skips
) {
if
(
$f
=~ m|/$|) {
$f
=
qr/\Q$f\E.*/
;
}
else
{
$f
=
qr/\Q$f\E/
;
}
}
$deparse_failures
=
join
(
'|'
,
@deparse_failures
);
$deparse_failures
=
qr/^(?:$deparse_failures)$/
;
$deparse_skips
=
join
(
'|'
,
@deparse_skips
);
$deparse_skips
=
qr/^(?:$deparse_skips)$/
;
}