make/tests/thelp.pl
Paul Smith 11444fb001 [SV 62654] Support GNU Make on z/OS
Original patches provided by Igor Todorovski <itodorov@ca.ibm.com>
Reworked by Paul Smith <psmith@gnu.org>.
Thanks to IBM for providing a test system.

* NEWS: Announce support.
* AUTHORS: Ditto.
* README.zOS: Provide details on building GNU Make on z/OS.
* build.sh (get_mk_var): z/OS sh has a strange bug which causes it to
generate extra lines of output: rework the function to print output
as we compute it instead of collecting it into a variable, which
works around this bug.
* src/makeint.h: Declare MK_OS_ZOS if we're building for z/OS.
* src/arscan.c: Don't include <ar.h> on z/OS.
* src/job.c: We can't change environ in ASCII mode on z/OS.
* src/main.c: Ditto.  Also we can't use pselect() on z/OS.
* src/posixos.c: pselect() seems to hang on z/OS: don't use it.
* tests/run_make_tests.pl: Handle different exit codes on z/OS.
* tests/test_driver.pl: Preserve some special z/OS env.vars.
Add special checks to output comparisons when on z/OS.
* tests/scripts/features/archives: Don't validate names.  Don't
try to compile empty files as IBM compilers complain.
* tests/scripts/features/shell_assignment: Fix octal value of #.
* tests/scripts/features/temp_stdin: Don't print "term".
* tests/scripts/functions/shell: Handle shell exit codes.
* tests/scripts/targets/ONESHELL: Ditto.
* tests/scripts/targets/POSIX: sh -x prints differently.
* tests/scripts/variables/SHELL: Ditto.
2023-01-08 10:45:38 -05:00

136 lines
3.4 KiB
Perl
Executable file

#!/usr/bin/env perl
# -*-perl-*-
#
# This script helps us write tests in a portable way, without relying on a lot
# of shell features. Since we already have Perl to run the tests, use that.
#
# The arguments represent a set of steps that will be run one at a time.
# Each step consists of an operator and argument.
#
# It supports the following operators:
# out <word> : echo <word> to stdout with a newline
# raw <word> : echo <word> to stdout without adding anything
# env <word> : echo the value of the env.var. <word>, or "<unset>"
# file <word> : echo <word> to stdout AND create the file <word>
# dir <word> : echo <word> to stdout AND create the directory <word>
# rm <word> : echo <word> to stdout AND delete the file/directory <word>
# wait <word> : wait for a file named <word> to exist
# tmout <secs> : Change the timeout for waiting. Default is 4 seconds.
# sleep <secs> : Sleep for <secs> seconds then echo <secs>
# term <pid> : send SIGTERM to PID <pid>
# fail <err> : echo <err> to stdout then exit with error code err
#
# If given -q only the "out", "raw", and "env" commands generate output.
# Force flush
$| = 1;
my $quiet = 0;
my $timeout = 10;
sub op {
my ($op, $nm) = @_;
defined $nm or die "Missing value for $op\n";
if ($op eq 'out') {
print "$nm\n";
return 1;
}
if ($op eq 'raw') {
print "$nm";
return 1;
}
if ($op eq 'env') {
print "$nm=" unless $quiet;
if (exists $ENV{$nm}) {
print "$ENV{$nm}\n";
} else {
print "<unset>\n";
}
return 1;
}
# Show the output before creating the file
if ($op eq 'file') {
print "file $nm\n" unless $quiet;
open(my $fh, '>', $nm) or die "$nm: open: $!\n";
close(my $fh);
return 1;
}
# Show the output before creating the directory
if ($op eq 'dir') {
print "dir $nm\n" unless $quiet;
mkdir($nm) or die "$nm: mkdir: $!\n";
return 1;
}
# Show the output after removing the file
if ($op eq 'rm') {
if (-f $nm) {
unlink($nm) or die "$nm: unlink: $!\n";
} elsif (-d $nm) {
rmdir($nm) or die "$nm: rmdir: $!\n";
} else {
die "$nm: not file or directory: $!\n";
}
print "rm $nm\n" unless $quiet;
return 1;
}
if ($op eq 'tmout') {
$timeout = $nm;
print "tmout $nm\n" unless $quiet;
return 1;
}
# Show the output after the file exists
if ($op eq 'wait') {
my $start = time();
my $end = $start + $timeout;
while (time() <= $end) {
if (-f $nm) {
print "wait $nm\n" unless $quiet;
return 1;
}
select(undef, undef, undef, 0.1);
}
die "wait $nm: timeout after ".(time()-$start-1)." seconds\n";
}
# Show the output after sleeping
if ($op eq 'sleep') {
sleep($nm);
print "sleep $nm\n" unless $quiet;
return 1;
}
if ($op eq 'term') {
print "term $nm\n" unless $quiet;
kill('TERM', $nm);
return 1;
}
if ($op eq 'fail') {
print "fail $nm\n" unless $quiet;
exit($nm);
}
die("Invalid command: $op $nm\n");
}
if (@ARGV && $ARGV[0] eq '-q') {
$quiet = 1;
shift;
}
while (@ARGV) {
if (op($ARGV[0], $ARGV[1])) {
shift;
shift;
}
}
exit(0);