blob: c60de6e4e350ce2e85a1eb19838a33804611a79a [file] [log] [blame]
// Copyright 2014 The Bazel Authors. 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).
#include <err.h>
#include <errno.h>
#include <signal.h>
#include <stdbool.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <sys/wait.h>
#include <unistd.h>
#include <string>
#include <vector>
#include "src/main/tools/logging.h"
#include "src/main/tools/process-tools.h"
static double global_kill_delay;
static pid_t global_child_pid;
static volatile sig_atomic_t global_signal;
// Options parsing result.
struct Options {
double timeout_secs;
double kill_delay_secs;
std::string stdout_path;
std::string stderr_path;
std::vector<char *> args;
};
static struct Options opt;
// Print out a usage error and exit with EXIT_FAILURE.
static void Usage(char *program_name) {
fprintf(stderr,
"Usage: %s <timeout-secs> <kill-delay-secs> <stdout-redirect> "
"<stderr-redirect> <command> [args] ...\n",
program_name);
exit(EXIT_FAILURE);
}
// Parse the command line flags and return the result in an Options structure
// passed as argument.
static void ParseCommandLine(std::vector<char *> args) {
if (args.size() < 5) {
Usage(args.front());
}
int optind = 1;
if (sscanf(args[optind++], "%lf", &opt.timeout_secs) != 1) {
DIE("timeout_secs is not a real number.\n");
}
if (sscanf(args[optind++], "%lf", &opt.kill_delay_secs) != 1) {
DIE("kill_delay_secs is not a real number.\n");
}
opt.stdout_path.assign(args[optind++]);
opt.stderr_path.assign(args[optind++]);
opt.args.assign(args.begin() + optind, args.end());
// argv[] passed to execve() must be a null-terminated array.
opt.args.push_back(nullptr);
}
// Called when timeout or signal occurs.
void OnSignal(int sig) {
global_signal = sig;
// Nothing to do if we received a signal before spawning the child.
if (global_child_pid == -1) {
return;
}
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_child_pid, true, global_kill_delay);
} else {
// Signals should kill the process quickly, as it's typically blocking
// the return of the prompt after a user hits "Ctrl-C".
KillEverything(global_child_pid, false, global_kill_delay);
}
}
// Run the command specified by the argv array and kill it after timeout
// seconds.
static void SpawnCommand(const std::vector<char *> &args, double timeout_secs) {
global_child_pid = fork();
if (global_child_pid < 0) {
DIE("fork");
} else if (global_child_pid == 0) {
// In child.
if (setsid() < 0) {
DIE("setsid");
}
ClearSignalMask();
// Force umask to include read and execute for everyone, to make
// output permissions predictable.
umask(022);
// Does not return unless something went wrong.
if (execvp(args[0], args.data()) < 0) {
DIE("execvp(%s, ...)", args[0]);
}
} else {
// In parent.
// Set up a signal handler which kills all subprocesses when the given
// signal is triggered.
InstallSignalHandler(SIGALRM, OnSignal);
InstallSignalHandler(SIGTERM, OnSignal);
InstallSignalHandler(SIGINT, OnSignal);
if (timeout_secs > 0) {
SetTimeout(timeout_secs);
}
int status = WaitChild(global_child_pid);
// The child is done for, but may have grandchildren that we still have to
// kill.
kill(-global_child_pid, SIGKILL);
if (global_signal > 0) {
// Don't trust the exit code if we got a timeout or signal.
InstallDefaultSignalHandler(global_signal);
raise(global_signal);
} else if (WIFEXITED(status)) {
exit(WEXITSTATUS(status));
} else {
int sig = WTERMSIG(status);
InstallDefaultSignalHandler(sig);
raise(sig);
}
}
}
int main(int argc, char *argv[]) {
std::vector<char *> args(argv, argv + argc);
ParseCommandLine(args);
global_kill_delay = opt.kill_delay_secs;
SwitchToEuid();
SwitchToEgid();
Redirect(opt.stdout_path, STDOUT_FILENO);
Redirect(opt.stderr_path, STDERR_FILENO);
SpawnCommand(opt.args, opt.timeout_secs);
return 0;
}