blob: d4e1588f8dfd7acb683281a0953778095cd9c15f [file] [log] [blame]
#!/usr/bin/perl
#
# ***** BEGIN LICENSE BLOCK *****
# Version: MPL 1.1/GPL 2.0/LGPL 2.1
#
# The contents of this file are subject to the Mozilla Public License Version
# 1.1 (the "License"); you may not use this file except in compliance with
# the License. You may obtain a copy of the License at
# http://www.mozilla.org/MPL/
#
# Software distributed under the License is distributed on an "AS IS" basis,
# WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
# for the specific language governing rights and limitations under the
# License.
#
# The Original Code is the Netscape Portable Runtime (NSPR).
#
# The Initial Developer of the Original Code is
# Sun Microsystems, Inc.
# Portions created by the Initial Developer are Copyright (C) 2008
# the Initial Developer. All Rights Reserved.
#
# Contributor(s):
# Christophe Ravel <christophe.ravel@sun.com>, Sun Microsystems
# Slavomir Katuscak <slavomir.katuscak@sun.com>, Sun Microsystems
#
# Alternatively, the contents of this file may be used under the terms of
# either the GNU General Public License Version 2 or later (the "GPL"), or
# the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
# in which case the provisions of the GPL or the LGPL are applicable instead
# of those above. If you wish to allow use of your version of this file only
# under the terms of either the GPL or the LGPL, and not to allow others to
# use your version of this file under the terms of the MPL, indicate your
# decision by deleting the provisions above and replace them with the notice
# and other provisions required by the GPL or the LGPL. If you do not delete
# the provisions above, a recipient may use your version of this file under
# the terms of any one of the MPL, the GPL or the LGPL.
#
# ***** END LICENSE BLOCK *****
use POSIX qw(:sys_wait_h);
use POSIX qw(setsid);
use FileHandle;
# Constants
$WINOS = "MSWin32";
$osname = $^O;
use Cwd;
if ($osname =~ $WINOS) {
# Windows
require Win32::Process;
require Win32;
}
# Get environment variables.
$output_file = $ENV{NSPR_TEST_LOGFILE};
$timeout = $ENV{TEST_TIMEOUT};
$timeout = 0 if (!defined($timeout));
sub getTime {
($second, $minute, $hour, $dayOfMonth, $month, $yearOffset, $dayOfWeek, $dayOfYear, $daylightSavings) = localtime();
$year = 1900 + $yearOffset;
$theTime = sprintf("%04d-%02d-%02d %02d:%02d:%02d",$year,$month,$dayOfMonth,$hour,$minute,$second);
return $theTime;
}
sub open_log {
if (!defined($output_file)) {
print "No output file.\n";
# null device
if ($osname =~ $WINOS) {
$output_file = "nul";
} else {
$output_file = "/dev/null";
}
}
# use STDOUT for OF (to print summary of test results)
open(OF, ">&STDOUT") or die "Can't reuse STDOUT for OF\n";
OF->autoflush;
# reassign STDOUT to $output_file (to print details of test results)
open(STDOUT, ">$output_file") or die "Can't open file $output_file for STDOUT\n";
STDOUT->autoflush;
# redirect STDERR to STDOUT
open(STDERR, ">&STDOUT") or die "Can't redirect STDERR to STDOUT\n";
STDERR->autoflush;
# Print header test in summary
$now = getTime;
print OF "\nNSPR Test Results - tests\n";
print OF "\nBEGIN\t\t\t$now\n";
print OF "NSPR_TEST_LOGFILE\t$output_file\n";
print OF "TEST_TIMEOUT\t$timeout\n\n";
print OF "\nTest\t\t\tResult\n\n";
}
sub close_log {
# end of test marker in summary
$now = getTime;
print OF "END\t\t\t$now\n";
close(OF) or die "Can't close file OF\n";
close(STDERR) or die "Can't close STDERR\n";
close(STDOUT) or die "Can't close STDOUT\n";
}
sub print_begin {
$lprog = shift;
# Summary output
print OF "$prog";
# Full output
$now = getTime;
print "BEGIN TEST: $lprog ($now)\n\n";
}
sub print_end {
($lprog, $exit_status, $exit_signal, $exit_core) = @_;
if (($exit_status == 0) && ($exit_signal == 0) && ($exit_core == 0)) {
$str_status = "Passed";
} else {
$str_status = "FAILED";
}
if ($exit_signal != 0) {
$str_signal = " - signal $exit_signal";
} else {
$str_signal = "";
}
if ($exit_core != 0) {
$str_core = " - core dumped";
} else {
$str_core = "";
}
$now = getTime;
# Full output
print "\nEND TEST: $lprog ($now)\n";
print "TEST STATUS: $lprog = $str_status (exit status " . $exit_status . $str_signal . $str_core . ")\n";
print "--------------------------------------------------\n\n";
# Summary output
print OF "\t\t\t$str_status\n";
}
sub ux_start_prog {
# parameters:
$lprog = shift; # command to run
# Create a process group for the child
# so we can kill all of it if needed
setsid or die "setsid failed: $!";
# Start test program
exec("./$lprog");
# We should not be here unless exec failed.
print "Faild to exec $lprog";
exit 1 << 8;
}
sub ux_wait_timeout {
# parameters:
$lpid = shift; # child process id
$ltimeout = shift; # timeout
if ($ltimeout == 0) {
# No timeout: use blocking wait
$ret = waitpid($lpid,0);
# Exit and don't kill
$lstatus = $?;
$ltimeout = -1;
} else {
while ($ltimeout > 0) {
# Check status of child using non blocking wait
$ret = waitpid($lpid, WNOHANG);
if ($ret == 0) {
# Child still running
# print "Time left=$ltimeout\n";
sleep 1;
$ltimeout--;
} else {
# Child has ended
$lstatus = $?;
# Exit the wait loop and don't kill
$ltimeout = -1;
}
}
}
if ($ltimeout == 0) {
# we ran all the timeout: it's time to kill the child
print "Timeout ! Kill child process $lpid\n";
# Kill the child process and group
kill(-9,$lpid);
$lstatus = 9;
}
return $lstatus;
}
sub ux_test_prog {
# parameters:
$prog = shift; # Program to test
$child_pid = fork;
if ($child_pid == 0) {
# we are in the child process
print_begin($prog);
ux_start_prog($prog);
} else {
# we are in the parent process
$status = ux_wait_timeout($child_pid,$timeout);
# See Perlvar for documentation of $?
# exit status = $status >> 8
# exit signal = $status & 127 (no signal = 0)
# core dump = $status & 128 (no core = 0)
print_end($prog, $status >> 8, $status & 127, $status & 128);
}
return $status;
}
sub win_path {
$lpath = shift;
# MSYS drive letter = /c/ -> c:/
$lpath =~ s/^\/(\w)\//$1:\//;
# Cygwin drive letter = /cygdrive/c/ -> c:/
$lpath =~ s/^\/cygdrive\/(\w)\//$1:\//;
# replace / with \\
$lpath =~ s/\//\\\\/g;
return $lpath;
}
sub win_ErrorReport{
print Win32::FormatMessage( Win32::GetLastError() );
}
sub win_test_prog {
# parameters:
$prog = shift; # Program to test
$status = 1;
$curdir = getcwd;
$curdir = win_path($curdir);
$prog_path = "$curdir\\$prog.exe";
print_begin($prog);
Win32::Process::Create($ProcessObj,
"$prog_path",
"$prog",
0,
NORMAL_PRIORITY_CLASS,
".")|| die win_ErrorReport();
$retwait = $ProcessObj->Wait($timeout * 1000);
if ( $retwait == 0) {
# the prog didn't finish after the timeout: kill
$ProcessObj->Kill($status);
print "Timeout ! Process killed with exit status $status\n";
} else {
# the prog finished before the timeout: get exit status
$ProcessObj->GetExitCode($status);
}
# There is no signal, no core on Windows
print_end($prog, $status, 0, 0);
return $status
}
# MAIN ---------------
@progs = (
"accept",
"acceptread",
"acceptreademu",
"affinity",
"alarm",
"anonfm",
"atomic",
"attach",
"bigfile",
"cleanup",
"cltsrv",
"concur",
"cvar",
"cvar2",
"dlltest",
"dtoa",
"errcodes",
"exit",
"fdcach",
"fileio",
"foreign",
"formattm",
"fsync",
"gethost",
"getproto",
"i2l",
"initclk",
"inrval",
"instrumt",
"intrio",
"intrupt",
"io_timeout",
"ioconthr",
"join",
"joinkk",
"joinku",
"joinuk",
"joinuu",
"layer",
"lazyinit",
"libfilename",
"lltest",
"lock",
"lockfile",
"logfile",
"logger",
"many_cv",
"multiwait",
"nameshm1",
"nblayer",
"nonblock",
"ntioto",
"ntoh",
"op_2long",
"op_excl",
"op_filnf",
"op_filok",
"op_nofil",
"parent",
"parsetm",
"peek",
"perf",
"pipeping",
"pipeping2",
"pipeself",
"poll_nm",
"poll_to",
"pollable",
"prftest",
"primblok",
"provider",
"prpollml",
"ranfile",
"randseed",
"reinit",
"rwlocktest",
"sel_spd",
"selct_er",
"selct_nm",
"selct_to",
"selintr",
"sema",
"semaerr",
"semaping",
"sendzlf",
"server_test",
"servr_kk",
"servr_uk",
"servr_ku",
"servr_uu",
"short_thread",
"sigpipe",
"socket",
"sockopt",
"sockping",
"sprintf",
"stack",
"stdio",
"str2addr",
"strod",
"switch",
"system",
"testbit",
"testfile",
"threads",
"timemac",
"timetest",
"tpd",
"udpsrv",
"vercheck",
"version",
"writev",
"xnotify",
"zerolen");
open_log;
foreach $current_prog (@progs) {
if ($osname =~ $WINOS) {
win_test_prog($current_prog);
} else {
ux_test_prog($current_prog);
}
}
close_log;