blob: 305ad0e442ae6662e9086bbae7d15d45ae76205e [file] [log] [blame]
philwo9d9ac152017-06-21 15:25:10 +02001// 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>
jmmvc0926372020-05-28 06:42:55 -070022
23#include <cstring>
philwo9d9ac152017-06-21 15:25:10 +020024#include <memory>
25#include <string>
26#include <vector>
27
28#include "src/main/tools/logging.h"
29
30struct 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.
34static 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"
jmmv91eca5a2020-06-05 10:37:28 -070044 " -g/--graceful_sigterm propagate SIGTERM to the subprocess and delay "
45 "the corresponding SIGKILL until --kill_delay has passed\n"
philwo9d9ac152017-06-21 15:25:10 +020046 " -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"
rupertsb394da42017-11-28 21:17:48 -080052 " -s/--stats <file> if set, write stats in protobuf format to a file\n"
philwo9d9ac152017-06-21 15:25:10 +020053 " -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
philwo9d9ac152017-06-21 15:25:10 +020058// Parses command line flags from an argv array and puts the results into the
59// global `opt` struct.
60static void ParseCommandLine(const std::vector<char *> &args) {
61 static struct option long_options[] = {
jmmv91eca5a2020-06-05 10:37:28 -070062 {"graceful_sigterm", no_argument, 0, 'g'},
philwo9d9ac152017-06-21 15:25:10 +020063 {"timeout", required_argument, 0, 't'},
64 {"kill_delay", required_argument, 0, 'k'},
65 {"stdout", required_argument, 0, 'o'},
66 {"stderr", required_argument, 0, 'e'},
rupertsb394da42017-11-28 21:17:48 -080067 {"stats", required_argument, 0, 's'},
philwo9d9ac152017-06-21 15:25:10 +020068 {"debug", no_argument, 0, 'd'},
69 {0, 0, 0, 0}};
70 extern char *optarg;
71 extern int optind, optopt;
72 int c;
73
jmmv6230d422020-08-03 08:00:34 -070074 while ((c = getopt_long(args.size(), args.data(), "+:gt:k:o:e:s:d",
rupertsb394da42017-11-28 21:17:48 -080075 long_options, nullptr)) != -1) {
philwo9d9ac152017-06-21 15:25:10 +020076 switch (c) {
jmmv91eca5a2020-06-05 10:37:28 -070077 case 'g':
78 opt.graceful_sigterm = true;
79 break;
philwo9d9ac152017-06-21 15:25:10 +020080 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;
rupertsb394da42017-11-28 21:17:48 -0800106 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;
philwo9d9ac152017-06-21 15:25:10 +0200114 case 'd':
115 opt.debug = true;
116 break;
philwo9d9ac152017-06-21 15:25:10 +0200117 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
131void 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}