Do not fork/exec in gruntmaster-compile
[gruntmaster-daemon.git] / gruntmaster-exec
index 69e967e091120763f4e64e11d8ca933232e850de..9590b4e34173feaa8198bfa13e84e424d810bea6 100755 (executable)
-#!/usr/bin/perl -w
+#!/usr/bin/perl
 use v5.14;
 use strict;
-
-use BSD::Resource qw/setrlimit RLIMIT_AS RLIMIT_FSIZE/;
+use warnings;
+
+use constant +{
+       # Accepted
+       AC => 0,
+
+       # Internal server error
+       ERR => -1,
+
+       # All other errors
+       WA => 1,
+       NZX => 2,
+       TLE => 3,
+       OLE => 4,
+       DIED => 5,
+       REJ => 10,
+};
+# These constants are changed by ex/makevm
+use constant USER => 65534;
+use constant GROUP => 65534;
+
+use BSD::Resource qw/setrlimit RLIMIT_AS RLIMIT_FSIZE RLIMIT_NPROC/;
+use IPC::Signal qw/sig_name sig_num/;
 use sigtrap qw/XFSZ/;
 
-##################################################
-
-my ($mlimit, $olimit, @args) = @ARGV;
-
-setrlimit RLIMIT_AS, $mlimit, $mlimit or die $! if $mlimit;
-setrlimit RLIMIT_FSIZE, $olimit, $olimit or die $! if $olimit;
-
-%ENV = (ONLINE_JUDGE => 1, PATH => $ENV{PATH}, HOME => $ENV{HOME});
-exec @args;
-
+use Getopt::Long;
+use POSIX qw//;
+use Text::ParseWords qw/shellwords/;
+use Time::HiRes qw/alarm/;
+
+my (@fds, $timeout, $mlimit, $olimit, $nobody);
+my $close = 1;
+
+GetOptions(
+       "fd=s"      => \@fds,
+       "timeout=f" => \$timeout,
+       "mlimit=i"  => \$mlimit,
+       "olimit=i"  => \$olimit,
+       "close!"    => \$close,
+       "nobody!"   => \$nobody,
+);
+
+my $killuser = $ENV{GRUNTMASTER_KILL_USER};
+my @sudo;
+@sudo = (shellwords ($ENV{GRUNTMASTER_SUDO}), '--') if $ENV{GRUNTMASTER_SUDO} && $nobody;
+undef $mlimit if @sudo; # sudo wants a lot of address space
+
+my $ret = fork // die 'Cannot fork';
+if ($ret) {
+       my $tle;
+       local $SIG{ALRM} = sub {
+               if ($killuser) {
+                       system @sudo, 'pkill', '-KILL', '-u', $killuser;
+               } else {
+                       kill KILL => $ret
+               }
+               $tle = 1
+       };
+       alarm ($timeout || 5);
+       waitpid $ret, 0;
+       alarm 0;
+       if (@sudo) {
+               $? = $? >> 8;
+               $? = $? < 128 || $? > 128+32 ? ($? << 8) : $? - 128;
+       }
+       my $sig = $? & 127;
+       my $signame = sig_name $sig;
+       exit !say TLE,  "\nTime Limit Exceeded"   if $tle;
+       exit !say OLE,  "\nOutput Limit Exceeded" if $sig && $signame eq 'XFSZ';
+       exit !say DIED, "\nCrash (SIG$signame)"   if $sig && $signame ne 'PIPE';
+       exit !say NZX,  "\nNon-zero exit status: " . ($? >> 8) if $? >> 8;
+       exit !say AC,   "\nAll OK";
+} else {
+       $^F = 50;
+       if ($close) {
+               POSIX::close $_ for 0 .. $^F;
+       }
+       for my $fdstring (@fds) {
+               my ($fd, $file) = split ' ', $fdstring, 2;
+               open my $fh, $file or die $!;
+               my $oldfd = fileno $fh;
+               if ($oldfd != $fd) {
+                       POSIX::dup2 $oldfd, $fd or die $!;
+                       POSIX::close $oldfd or die $!;
+               }
+       }
+       my $nproc = $killuser ? 15 : 1;
+       my $debug = $ENV{TEST_VERBOSE};
+       %ENV = (ONLINE_JUDGE => 1, PATH => $ENV{PATH}, HOME => $ENV{HOME});
+       setrlimit RLIMIT_AS, $mlimit, $mlimit or die $! if $mlimit;
+       setrlimit RLIMIT_FSIZE, $olimit, $olimit or die $! if $olimit;
+       setrlimit RLIMIT_NPROC, $nproc, $nproc or die $! if $nobody;
+       POSIX::setgid $nobody ? 65534 : USER;
+       POSIX::setuid $nobody ? 65534 : GROUP;
+       unshift @ARGV, @sudo;
+       say STDERR "Executing: ", join ' ', map { "'$_'" } @ARGV if $debug;
+       exec @ARGV;
+}
+
+1;
 __END__
 
 =encoding utf-8
This page took 0.011737 seconds and 4 git commands to generate.