philwo | 9d9ac15 | 2017-06-21 15:25:10 +0200 | [diff] [blame] | 1 | // Copyright 2017 The Bazel Authors. All rights reserved. |
| 2 | // |
| 3 | // Licensed under the Apache License, Version 2.0 (the "License"); |
| 4 | // you may not use this file except in compliance with the License. |
| 5 | // You may obtain a copy of the License at |
| 6 | // |
| 7 | // http://www.apache.org/licenses/LICENSE-2.0 |
| 8 | // |
| 9 | // Unless required by applicable law or agreed to in writing, software |
| 10 | // distributed under the License is distributed on an "AS IS" BASIS, |
| 11 | // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| 12 | // See the License for the specific language governing permissions and |
| 13 | // limitations under the License. |
| 14 | |
| 15 | #include "src/main/tools/process-wrapper-options.h" |
| 16 | |
| 17 | #include <getopt.h> |
| 18 | #include <stdarg.h> |
| 19 | #include <stdio.h> |
| 20 | #include <stdlib.h> |
| 21 | #include <unistd.h> |
jmmv | c092637 | 2020-05-28 06:42:55 -0700 | [diff] [blame] | 22 | |
| 23 | #include <cstring> |
philwo | 9d9ac15 | 2017-06-21 15:25:10 +0200 | [diff] [blame] | 24 | #include <memory> |
| 25 | #include <string> |
| 26 | #include <vector> |
| 27 | |
| 28 | #include "src/main/tools/logging.h" |
| 29 | |
| 30 | struct Options opt; |
| 31 | |
| 32 | // Print out a usage error. argc and argv are the argument counter and vector, |
| 33 | // fmt is a format, string for the error message to print. |
| 34 | static void Usage(char *program_name, const char *fmt, ...) { |
| 35 | va_list ap; |
| 36 | va_start(ap, fmt); |
| 37 | vfprintf(stderr, fmt, ap); |
| 38 | va_end(ap); |
| 39 | |
| 40 | fprintf(stderr, "\nUsage: %s -- command arg1 @args\n", program_name); |
| 41 | fprintf( |
| 42 | stderr, |
| 43 | "\nPossible arguments:\n" |
jmmv | 91eca5a | 2020-06-05 10:37:28 -0700 | [diff] [blame] | 44 | " -g/--graceful_sigterm propagate SIGTERM to the subprocess and delay " |
| 45 | "the corresponding SIGKILL until --kill_delay has passed\n" |
philwo | 9d9ac15 | 2017-06-21 15:25:10 +0200 | [diff] [blame] | 46 | " -t/--timeout <timeout> timeout after which the child process will be " |
| 47 | "terminated with SIGTERM\n" |
| 48 | " -k/--kill_delay <timeout> in case timeout occurs, how long to wait " |
| 49 | "before killing the child with SIGKILL\n" |
| 50 | " -o/--stdout <file> redirect stdout to a file\n" |
| 51 | " -e/--stderr <file> redirect stderr to a file\n" |
ruperts | b394da4 | 2017-11-28 21:17:48 -0800 | [diff] [blame] | 52 | " -s/--stats <file> if set, write stats in protobuf format to a file\n" |
philwo | 9d9ac15 | 2017-06-21 15:25:10 +0200 | [diff] [blame] | 53 | " -d/--debug if set, debug info will be printed\n" |
| 54 | " -- command to run inside sandbox, followed by arguments\n"); |
| 55 | exit(EXIT_FAILURE); |
| 56 | } |
| 57 | |
philwo | 9d9ac15 | 2017-06-21 15:25:10 +0200 | [diff] [blame] | 58 | // Parses command line flags from an argv array and puts the results into the |
| 59 | // global `opt` struct. |
| 60 | static void ParseCommandLine(const std::vector<char *> &args) { |
| 61 | static struct option long_options[] = { |
jmmv | 91eca5a | 2020-06-05 10:37:28 -0700 | [diff] [blame] | 62 | {"graceful_sigterm", no_argument, 0, 'g'}, |
philwo | 9d9ac15 | 2017-06-21 15:25:10 +0200 | [diff] [blame] | 63 | {"timeout", required_argument, 0, 't'}, |
| 64 | {"kill_delay", required_argument, 0, 'k'}, |
| 65 | {"stdout", required_argument, 0, 'o'}, |
| 66 | {"stderr", required_argument, 0, 'e'}, |
ruperts | b394da4 | 2017-11-28 21:17:48 -0800 | [diff] [blame] | 67 | {"stats", required_argument, 0, 's'}, |
philwo | 9d9ac15 | 2017-06-21 15:25:10 +0200 | [diff] [blame] | 68 | {"debug", no_argument, 0, 'd'}, |
| 69 | {0, 0, 0, 0}}; |
| 70 | extern char *optarg; |
| 71 | extern int optind, optopt; |
| 72 | int c; |
| 73 | |
jmmv | 6230d42 | 2020-08-03 08:00:34 -0700 | [diff] [blame] | 74 | while ((c = getopt_long(args.size(), args.data(), "+:gt:k:o:e:s:d", |
ruperts | b394da4 | 2017-11-28 21:17:48 -0800 | [diff] [blame] | 75 | long_options, nullptr)) != -1) { |
philwo | 9d9ac15 | 2017-06-21 15:25:10 +0200 | [diff] [blame] | 76 | switch (c) { |
jmmv | 91eca5a | 2020-06-05 10:37:28 -0700 | [diff] [blame] | 77 | case 'g': |
| 78 | opt.graceful_sigterm = true; |
| 79 | break; |
philwo | 9d9ac15 | 2017-06-21 15:25:10 +0200 | [diff] [blame] | 80 | case 't': |
| 81 | if (sscanf(optarg, "%lf", &opt.timeout_secs) != 1) { |
| 82 | Usage(args.front(), "Invalid timeout (-t) value: %s", optarg); |
| 83 | } |
| 84 | break; |
| 85 | case 'k': |
| 86 | if (sscanf(optarg, "%lf", &opt.kill_delay_secs) != 1) { |
| 87 | Usage(args.front(), "Invalid kill delay (-k) value: %s", optarg); |
| 88 | } |
| 89 | break; |
| 90 | case 'o': |
| 91 | if (opt.stdout_path.empty()) { |
| 92 | opt.stdout_path.assign(optarg); |
| 93 | } else { |
| 94 | Usage(args.front(), |
| 95 | "Cannot redirect stdout (-o) to more than one destination."); |
| 96 | } |
| 97 | break; |
| 98 | case 'e': |
| 99 | if (opt.stderr_path.empty()) { |
| 100 | opt.stderr_path.assign(optarg); |
| 101 | } else { |
| 102 | Usage(args.front(), |
| 103 | "Cannot redirect stderr (-e) to more than one destination."); |
| 104 | } |
| 105 | break; |
ruperts | b394da4 | 2017-11-28 21:17:48 -0800 | [diff] [blame] | 106 | case 's': |
| 107 | if (opt.stats_path.empty()) { |
| 108 | opt.stats_path.assign(optarg); |
| 109 | } else { |
| 110 | Usage(args.front(), |
| 111 | "Cannot write stats (-s) to more than one destination."); |
| 112 | } |
| 113 | break; |
philwo | 9d9ac15 | 2017-06-21 15:25:10 +0200 | [diff] [blame] | 114 | case 'd': |
| 115 | opt.debug = true; |
| 116 | break; |
philwo | 9d9ac15 | 2017-06-21 15:25:10 +0200 | [diff] [blame] | 117 | case '?': |
| 118 | Usage(args.front(), "Unrecognized argument: -%c (%d)", optopt, optind); |
| 119 | break; |
| 120 | case ':': |
| 121 | Usage(args.front(), "Flag -%c requires an argument", optopt); |
| 122 | break; |
| 123 | } |
| 124 | } |
| 125 | |
| 126 | if (optind < static_cast<int>(args.size())) { |
| 127 | opt.args.assign(args.begin() + optind, args.end()); |
| 128 | } |
| 129 | } |
| 130 | |
| 131 | void ParseOptions(int argc, char *argv[]) { |
| 132 | std::vector<char *> args(argv, argv + argc); |
| 133 | |
| 134 | ParseCommandLine(args); |
| 135 | |
| 136 | if (opt.args.empty()) { |
| 137 | Usage(args.front(), "No command specified."); |
| 138 | } |
| 139 | |
| 140 | // argv[] passed to execve() must be a null-terminated array. |
| 141 | opt.args.push_back(nullptr); |
| 142 | } |