blob: 5aff63cb4689a80465000887d3df85d1e6212c7d [file] [log] [blame]
// Copyright 2014 Google Inc. All rights reserved.
//
// Licensed under the Apache License, Version 2.0 (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.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
// process-wrapper runs a subprocess with a given timeout (optional),
// redirecting stdout and stderr to given files. Upon exit, whether
// from normal termination or timeout, the subprocess (and any of its children)
// is killed.
//
// The exit status of this program is whatever the child process returned,
// unless process-wrapper receives a signal. ie, on SIGTERM this program will
// die with raise(SIGTERM) even if the child process handles SIGTERM with
// exit(0).
#define _GNU_SOURCE
#include <errno.h>
#include <fcntl.h>
#include <math.h>
#include <signal.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/time.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <sys/wait.h>
#include <unistd.h>
// Not in headers on OSX.
extern char **environ;
static int global_pid; // Returned from fork().
static int global_signal = -1;
static double global_kill_delay = 0.0;
#define DIE(args...) { \
fprintf(stderr, args); \
fprintf(stderr, " --- "); \
perror(NULL); \
fprintf(stderr, "\n"); \
exit(EXIT_FAILURE); \
}
#define CHECK_CALL(x) if (x != 0) { perror(#x); exit(1); }
// Make sure the process and all subprocesses are killed.
static void KillEverything(int pgrp) {
kill(-pgrp, SIGTERM);
// Round up fractional seconds in this polling implementation.
int kill_delay = (int)(global_kill_delay+0.999) ;
// If the process is still alive, give it some time to die gracefully.
while (kill(-pgrp, 0) == 0 && kill_delay-- > 0) {
sleep(1);
}
kill(-pgrp, SIGKILL);
}
// Called when timeout or Signal occurs.
static void OnSignal(int sig) {
global_signal = sig;
if (sig == SIGALRM) {
// SIGALRM represents a timeout, so we should give the process a bit of
// time to die gracefully if it needs it.
KillEverything(global_pid);
} else {
// Signals should kill the process quickly, as it's typically blocking
// the return of the prompt after a user hits "Ctrl-C".
kill(-global_pid, SIGKILL);
}
}
// Set up a signal handler which kills all subprocesses when the
// given signal is triggered.
static void InstallSignalHandler(int sig) {
struct sigaction sa = {};
sa.sa_handler = OnSignal;
sigemptyset(&sa.sa_mask);
CHECK_CALL(sigaction(sig, &sa, NULL));
}
// Revert signal handler to default.
static void UnHandle(int sig) {
struct sigaction sa = {};
sa.sa_handler = SIG_DFL;
sigemptyset(&sa.sa_mask);
CHECK_CALL(sigaction(sig, &sa, NULL));
}
// Enable the given timeout, or no-op if the timeout is non-positive.
static void EnableAlarm(double timeout) {
if (timeout <= 0) return;
struct itimerval timer = {};
timer.it_interval.tv_sec = 0;
timer.it_interval.tv_usec = 0;
double int_val, fraction_val;
fraction_val = modf(timeout, &int_val);
timer.it_value.tv_sec = (long) int_val;
timer.it_value.tv_usec = (long) (fraction_val * 1e6);
CHECK_CALL(setitimer(ITIMER_REAL, &timer, NULL));
}
static void ClearSignalMask() {
// Use an empty signal mask and default signal handlers in the
// subprocess.
sigset_t sset;
sigemptyset(&sset);
sigprocmask(SIG_SETMASK, &sset, NULL);
for (int i = 1; i < NSIG; ++i) {
if (i == SIGKILL || i == SIGSTOP) continue;
struct sigaction sa = {};
sa.sa_handler = SIG_DFL;
sigemptyset(&sa.sa_mask);
sigaction(i, &sa, NULL);
}
}
static int WaitChild(pid_t pid, const char *name) {
int err = 0;
int status = 0;
do {
err = waitpid(pid, &status, 0);
} while (err == -1 && errno == EINTR);
if (err == -1) {
DIE("wait on %s (pid %d) failed", name, pid);
}
return status;
}
// Usage: process-wrapper
// <timeout_sec> <kill_delay_sec> <stdout file> <stderr file>
// [cmdline]
int main(int argc, char *argv[]) {
if (argc <= 5) {
DIE("Not enough cmd line arguments to process-wrapper");
}
// Parse the cmdline args to get the timeout and redirect files.
argv++;
double timeout;
if (sscanf(*argv++, "%lf", &timeout) != 1) {
DIE("timeout_sec is not a real number.");
}
if (sscanf(*argv++, "%lf", &global_kill_delay) != 1) {
DIE("kill_delay_sec is not a real number.");
}
char *stdout_path = *argv++;
char *stderr_path = *argv++;
if (strcmp(stdout_path, "-")) {
// Redirect stdout and stderr.
int fd_out = open(stdout_path, O_WRONLY|O_CREAT|O_TRUNC, 0666);
if (fd_out == -1) {
DIE("Could not open %s for stdout", stdout_path);
}
if (dup2(fd_out, STDOUT_FILENO) == -1) {
DIE("dup2 failed for stdout");
}
CHECK_CALL(close(fd_out));
}
if (strcmp(stderr_path, "-")) {
int fd_err = open(stderr_path, O_WRONLY|O_CREAT|O_TRUNC, 0666);
if (fd_err == -1) {
DIE("Could not open %s for stderr", stderr_path);
}
if (dup2(fd_err, STDERR_FILENO) == -1) {
DIE("dup2 failed for stderr");
}
CHECK_CALL(close(fd_err));
}
global_pid = fork();
if (global_pid < 0) {
DIE("Fork failed");
} else if (global_pid == 0) {
// In child.
if (setsid() == -1) {
DIE("Could not setsid from child");
}
ClearSignalMask();
// Force umask to include read and execute for everyone, to make
// output permissions predictable.
umask(022);
execvp(argv[0], argv); // Does not return.
DIE("execvpe %s failed", argv[0]);
} else {
// In parent.
InstallSignalHandler(SIGALRM);
InstallSignalHandler(SIGTERM);
InstallSignalHandler(SIGINT);
EnableAlarm(timeout);
int status = WaitChild(global_pid, argv[0]);
// The child is done, but may have grandchildren.
kill(-global_pid, SIGKILL);
if (global_signal > 0) {
// Don't trust the exit code if we got a timeout or signal.
UnHandle(global_signal);
raise(global_signal);
} else if (WIFEXITED(status)) {
exit(WEXITSTATUS(status));
} else {
int sig = WTERMSIG(status);
UnHandle(sig);
raise(sig);
}
}
}