]> git.lizzy.rs Git - rust.git/blob - src/bootstrap/flags.rs
Rollup merge of #98654 - nnethercote:pest-2.1.3-opt, r=pnkfelix
[rust.git] / src / bootstrap / flags.rs
1 //! Command-line interface of the rustbuild build system.
2 //!
3 //! This module implements the command-line parsing of the build system which
4 //! has various flags to configure how it's run.
5
6 use std::path::PathBuf;
7
8 use getopts::Options;
9
10 use crate::builder::{Builder, Kind};
11 use crate::config::{Config, TargetSelection};
12 use crate::setup::Profile;
13 use crate::util::t;
14 use crate::{Build, DocTests};
15
16 #[derive(Copy, Clone)]
17 pub enum Color {
18     Always,
19     Never,
20     Auto,
21 }
22
23 impl Default for Color {
24     fn default() -> Self {
25         Self::Auto
26     }
27 }
28
29 impl std::str::FromStr for Color {
30     type Err = ();
31
32     fn from_str(s: &str) -> Result<Self, Self::Err> {
33         match s.to_lowercase().as_str() {
34             "always" => Ok(Self::Always),
35             "never" => Ok(Self::Never),
36             "auto" => Ok(Self::Auto),
37             _ => Err(()),
38         }
39     }
40 }
41
42 /// Deserialized version of all flags for this compile.
43 pub struct Flags {
44     pub verbose: usize, // number of -v args; each extra -v after the first is passed to Cargo
45     pub on_fail: Option<String>,
46     pub stage: Option<u32>,
47     pub keep_stage: Vec<u32>,
48     pub keep_stage_std: Vec<u32>,
49
50     pub host: Option<Vec<TargetSelection>>,
51     pub target: Option<Vec<TargetSelection>>,
52     pub config: Option<PathBuf>,
53     pub build_dir: Option<PathBuf>,
54     pub jobs: Option<u32>,
55     pub cmd: Subcommand,
56     pub incremental: bool,
57     pub exclude: Vec<PathBuf>,
58     pub include_default_paths: bool,
59     pub rustc_error_format: Option<String>,
60     pub json_output: bool,
61     pub dry_run: bool,
62     pub color: Color,
63
64     // This overrides the deny-warnings configuration option,
65     // which passes -Dwarnings to the compiler invocations.
66     //
67     // true => deny, false => warn
68     pub deny_warnings: Option<bool>,
69
70     pub llvm_skip_rebuild: Option<bool>,
71
72     pub rust_profile_use: Option<String>,
73     pub rust_profile_generate: Option<String>,
74
75     pub llvm_profile_use: Option<String>,
76     // LLVM doesn't support a custom location for generating profile
77     // information.
78     //
79     // llvm_out/build/profiles/ is the location this writes to.
80     pub llvm_profile_generate: bool,
81 }
82
83 #[cfg_attr(test, derive(Clone))]
84 pub enum Subcommand {
85     Build {
86         paths: Vec<PathBuf>,
87     },
88     Check {
89         paths: Vec<PathBuf>,
90     },
91     Clippy {
92         fix: bool,
93         paths: Vec<PathBuf>,
94     },
95     Fix {
96         paths: Vec<PathBuf>,
97     },
98     Format {
99         paths: Vec<PathBuf>,
100         check: bool,
101     },
102     Doc {
103         paths: Vec<PathBuf>,
104         open: bool,
105     },
106     Test {
107         paths: Vec<PathBuf>,
108         /// Whether to automatically update stderr/stdout files
109         bless: bool,
110         force_rerun: bool,
111         compare_mode: Option<String>,
112         pass: Option<String>,
113         run: Option<String>,
114         skip: Vec<String>,
115         test_args: Vec<String>,
116         rustc_args: Vec<String>,
117         fail_fast: bool,
118         doc_tests: DocTests,
119         rustfix_coverage: bool,
120     },
121     Bench {
122         paths: Vec<PathBuf>,
123         test_args: Vec<String>,
124     },
125     Clean {
126         all: bool,
127     },
128     Dist {
129         paths: Vec<PathBuf>,
130     },
131     Install {
132         paths: Vec<PathBuf>,
133     },
134     Run {
135         paths: Vec<PathBuf>,
136     },
137     Setup {
138         profile: Profile,
139     },
140 }
141
142 impl Default for Subcommand {
143     fn default() -> Subcommand {
144         Subcommand::Build { paths: vec![PathBuf::from("nowhere")] }
145     }
146 }
147
148 impl Flags {
149     pub fn parse(args: &[String]) -> Flags {
150         let mut subcommand_help = String::from(
151             "\
152 Usage: x.py <subcommand> [options] [<paths>...]
153
154 Subcommands:
155     build, b    Compile either the compiler or libraries
156     check, c    Compile either the compiler or libraries, using cargo check
157     clippy      Run clippy (uses rustup/cargo-installed clippy binary)
158     fix         Run cargo fix
159     fmt         Run rustfmt
160     test, t     Build and run some test suites
161     bench       Build and run some benchmarks
162     doc, d      Build documentation
163     clean       Clean out build directories
164     dist        Build distribution artifacts
165     install     Install distribution artifacts
166     run, r      Run tools contained in this repository
167     setup       Create a config.toml (making it easier to use `x.py` itself)
168
169 To learn more about a subcommand, run `./x.py <subcommand> -h`",
170         );
171
172         let mut opts = Options::new();
173         // Options common to all subcommands
174         opts.optflagmulti("v", "verbose", "use verbose output (-vv for very verbose)");
175         opts.optflag("i", "incremental", "use incremental compilation");
176         opts.optopt("", "config", "TOML configuration file for build", "FILE");
177         opts.optopt(
178             "",
179             "build-dir",
180             "Build directory, overrides `build.build-dir` in `config.toml`",
181             "DIR",
182         );
183         opts.optopt("", "build", "build target of the stage0 compiler", "BUILD");
184         opts.optmulti("", "host", "host targets to build", "HOST");
185         opts.optmulti("", "target", "target targets to build", "TARGET");
186         opts.optmulti("", "exclude", "build paths to exclude", "PATH");
187         opts.optflag(
188             "",
189             "include-default-paths",
190             "include default paths in addition to the provided ones",
191         );
192         opts.optopt("", "on-fail", "command to run on failure", "CMD");
193         opts.optflag("", "dry-run", "dry run; don't build anything");
194         opts.optopt(
195             "",
196             "stage",
197             "stage to build (indicates compiler to use/test, e.g., stage 0 uses the \
198              bootstrap compiler, stage 1 the stage 0 rustc artifacts, etc.)",
199             "N",
200         );
201         opts.optmulti(
202             "",
203             "keep-stage",
204             "stage(s) to keep without recompiling \
205             (pass multiple times to keep e.g., both stages 0 and 1)",
206             "N",
207         );
208         opts.optmulti(
209             "",
210             "keep-stage-std",
211             "stage(s) of the standard library to keep without recompiling \
212             (pass multiple times to keep e.g., both stages 0 and 1)",
213             "N",
214         );
215         opts.optopt("", "src", "path to the root of the rust checkout", "DIR");
216         let j_msg = format!(
217             "number of jobs to run in parallel; \
218              defaults to {} (this host's logical CPU count)",
219             num_cpus::get()
220         );
221         opts.optopt("j", "jobs", &j_msg, "JOBS");
222         opts.optflag("h", "help", "print this help message");
223         opts.optopt(
224             "",
225             "warnings",
226             "if value is deny, will deny warnings, otherwise use default",
227             "VALUE",
228         );
229         opts.optopt("", "error-format", "rustc error format", "FORMAT");
230         opts.optflag("", "json-output", "use message-format=json");
231         opts.optopt("", "color", "whether to use color in cargo and rustc output", "STYLE");
232         opts.optopt(
233             "",
234             "llvm-skip-rebuild",
235             "whether rebuilding llvm should be skipped \
236              a VALUE of TRUE indicates that llvm will not be rebuilt \
237              VALUE overrides the skip-rebuild option in config.toml.",
238             "VALUE",
239         );
240         opts.optopt(
241             "",
242             "rust-profile-generate",
243             "generate PGO profile with rustc build",
244             "PROFILE",
245         );
246         opts.optopt("", "rust-profile-use", "use PGO profile for rustc build", "PROFILE");
247         opts.optflag("", "llvm-profile-generate", "generate PGO profile with llvm built for rustc");
248         opts.optopt("", "llvm-profile-use", "use PGO profile for llvm build", "PROFILE");
249
250         // We can't use getopt to parse the options until we have completed specifying which
251         // options are valid, but under the current implementation, some options are conditional on
252         // the subcommand. Therefore we must manually identify the subcommand first, so that we can
253         // complete the definition of the options.  Then we can use the getopt::Matches object from
254         // there on out.
255         let subcommand = match args.iter().find_map(|s| Kind::parse(&s)) {
256             Some(s) => s,
257             None => {
258                 // No or an invalid subcommand -- show the general usage and subcommand help
259                 // An exit code will be 0 when no subcommand is given, and 1 in case of an invalid
260                 // subcommand.
261                 println!("{}\n", subcommand_help);
262                 let exit_code = if args.is_empty() { 0 } else { 1 };
263                 crate::detail_exit(exit_code);
264             }
265         };
266
267         // Some subcommands get extra options
268         match subcommand {
269             Kind::Test => {
270                 opts.optflag("", "no-fail-fast", "Run all tests regardless of failure");
271                 opts.optmulti("", "skip", "skips tests matching SUBSTRING, if supported by test tool. May be passed multiple times", "SUBSTRING");
272                 opts.optmulti(
273                     "",
274                     "test-args",
275                     "extra arguments to be passed for the test tool being used \
276                         (e.g. libtest, compiletest or rustdoc)",
277                     "ARGS",
278                 );
279                 opts.optmulti(
280                     "",
281                     "rustc-args",
282                     "extra options to pass the compiler when running tests",
283                     "ARGS",
284                 );
285                 opts.optflag("", "no-doc", "do not run doc tests");
286                 opts.optflag("", "doc", "only run doc tests");
287                 opts.optflag("", "bless", "update all stderr/stdout files of failing ui tests");
288                 opts.optflag("", "force-rerun", "rerun tests even if the inputs are unchanged");
289                 opts.optopt(
290                     "",
291                     "compare-mode",
292                     "mode describing what file the actual ui output will be compared to",
293                     "COMPARE MODE",
294                 );
295                 opts.optopt(
296                     "",
297                     "pass",
298                     "force {check,build,run}-pass tests to this mode.",
299                     "check | build | run",
300                 );
301                 opts.optopt("", "run", "whether to execute run-* tests", "auto | always | never");
302                 opts.optflag(
303                     "",
304                     "rustfix-coverage",
305                     "enable this to generate a Rustfix coverage file, which is saved in \
306                         `/<build_base>/rustfix_missing_coverage.txt`",
307                 );
308             }
309             Kind::Check => {
310                 opts.optflag("", "all-targets", "Check all targets");
311             }
312             Kind::Bench => {
313                 opts.optmulti("", "test-args", "extra arguments", "ARGS");
314             }
315             Kind::Clippy => {
316                 opts.optflag("", "fix", "automatically apply lint suggestions");
317             }
318             Kind::Doc => {
319                 opts.optflag("", "open", "open the docs in a browser");
320             }
321             Kind::Clean => {
322                 opts.optflag("", "all", "clean all build artifacts");
323             }
324             Kind::Format => {
325                 opts.optflag("", "check", "check formatting instead of applying.");
326             }
327             _ => {}
328         };
329
330         // fn usage()
331         let usage = |exit_code: i32, opts: &Options, verbose: bool, subcommand_help: &str| -> ! {
332             let config = Config::parse(&["build".to_string()]);
333             let build = Build::new(config);
334             let paths = Builder::get_help(&build, subcommand);
335
336             println!("{}", opts.usage(subcommand_help));
337             if let Some(s) = paths {
338                 if verbose {
339                     println!("{}", s);
340                 } else {
341                     println!(
342                         "Run `./x.py {} -h -v` to see a list of available paths.",
343                         subcommand.as_str()
344                     );
345                 }
346             } else if verbose {
347                 panic!("No paths available for subcommand `{}`", subcommand.as_str());
348             }
349             crate::detail_exit(exit_code);
350         };
351
352         // Done specifying what options are possible, so do the getopts parsing
353         let matches = opts.parse(args).unwrap_or_else(|e| {
354             // Invalid argument/option format
355             println!("\n{}\n", e);
356             usage(1, &opts, false, &subcommand_help);
357         });
358
359         // Extra sanity check to make sure we didn't hit this crazy corner case:
360         //
361         //     ./x.py --frobulate clean build
362         //            ^-- option  ^     ^- actual subcommand
363         //                        \_ arg to option could be mistaken as subcommand
364         let mut pass_sanity_check = true;
365         match matches.free.get(0).and_then(|s| Kind::parse(&s)) {
366             Some(check_subcommand) => {
367                 if check_subcommand != subcommand {
368                     pass_sanity_check = false;
369                 }
370             }
371             None => {
372                 pass_sanity_check = false;
373             }
374         }
375         if !pass_sanity_check {
376             eprintln!("{}\n", subcommand_help);
377             eprintln!(
378                 "Sorry, I couldn't figure out which subcommand you were trying to specify.\n\
379                  You may need to move some options to after the subcommand.\n"
380             );
381             crate::detail_exit(1);
382         }
383         // Extra help text for some commands
384         match subcommand {
385             Kind::Build => {
386                 subcommand_help.push_str(
387                     "\n
388 Arguments:
389     This subcommand accepts a number of paths to directories to the crates
390     and/or artifacts to compile. For example, for a quick build of a usable
391     compiler:
392
393         ./x.py build --stage 1 library/std
394
395     This will build a compiler and standard library from the local source code.
396     Once this is done, build/$ARCH/stage1 contains a usable compiler.
397
398     If no arguments are passed then the default artifacts for that stage are
399     compiled. For example:
400
401         ./x.py build --stage 0
402         ./x.py build ",
403                 );
404             }
405             Kind::Check => {
406                 subcommand_help.push_str(
407                     "\n
408 Arguments:
409     This subcommand accepts a number of paths to directories to the crates
410     and/or artifacts to compile. For example:
411
412         ./x.py check library/std
413
414     If no arguments are passed then many artifacts are checked.",
415                 );
416             }
417             Kind::Clippy => {
418                 subcommand_help.push_str(
419                     "\n
420 Arguments:
421     This subcommand accepts a number of paths to directories to the crates
422     and/or artifacts to run clippy against. For example:
423
424         ./x.py clippy library/core
425         ./x.py clippy library/core library/proc_macro",
426                 );
427             }
428             Kind::Fix => {
429                 subcommand_help.push_str(
430                     "\n
431 Arguments:
432     This subcommand accepts a number of paths to directories to the crates
433     and/or artifacts to run `cargo fix` against. For example:
434
435         ./x.py fix library/core
436         ./x.py fix library/core library/proc_macro",
437                 );
438             }
439             Kind::Format => {
440                 subcommand_help.push_str(
441                     "\n
442 Arguments:
443     This subcommand optionally accepts a `--check` flag which succeeds if formatting is correct and
444     fails if it is not. For example:
445
446         ./x.py fmt
447         ./x.py fmt --check",
448                 );
449             }
450             Kind::Test => {
451                 subcommand_help.push_str(
452                     "\n
453 Arguments:
454     This subcommand accepts a number of paths to test directories that
455     should be compiled and run. For example:
456
457         ./x.py test src/test/ui
458         ./x.py test library/std --test-args hash_map
459         ./x.py test library/std --stage 0 --no-doc
460         ./x.py test src/test/ui --bless
461         ./x.py test src/test/ui --compare-mode chalk
462
463     Note that `test src/test/* --stage N` does NOT depend on `build compiler/rustc --stage N`;
464     just like `build library/std --stage N` it tests the compiler produced by the previous
465     stage.
466
467     Execute tool tests with a tool name argument:
468
469         ./x.py test tidy
470
471     If no arguments are passed then the complete artifacts for that stage are
472     compiled and tested.
473
474         ./x.py test
475         ./x.py test --stage 1",
476                 );
477             }
478             Kind::Doc => {
479                 subcommand_help.push_str(
480                     "\n
481 Arguments:
482     This subcommand accepts a number of paths to directories of documentation
483     to build. For example:
484
485         ./x.py doc src/doc/book
486         ./x.py doc src/doc/nomicon
487         ./x.py doc src/doc/book library/std
488         ./x.py doc library/std --open
489
490     If no arguments are passed then everything is documented:
491
492         ./x.py doc
493         ./x.py doc --stage 1",
494                 );
495             }
496             Kind::Run => {
497                 subcommand_help.push_str(
498                     "\n
499 Arguments:
500     This subcommand accepts a number of paths to tools to build and run. For
501     example:
502
503         ./x.py run src/tools/expand-yaml-anchors
504
505     At least a tool needs to be called.",
506                 );
507             }
508             Kind::Setup => {
509                 subcommand_help.push_str(&format!(
510                     "\n
511 x.py setup creates a `config.toml` which changes the defaults for x.py itself.
512
513 Arguments:
514     This subcommand accepts a 'profile' to use for builds. For example:
515
516         ./x.py setup library
517
518     The profile is optional and you will be prompted interactively if it is not given.
519     The following profiles are available:
520
521 {}",
522                     Profile::all_for_help("        ").trim_end()
523                 ));
524             }
525             Kind::Bench | Kind::Clean | Kind::Dist | Kind::Install => {}
526         };
527         // Get any optional paths which occur after the subcommand
528         let mut paths = matches.free[1..].iter().map(|p| p.into()).collect::<Vec<PathBuf>>();
529
530         let verbose = matches.opt_present("verbose");
531
532         // User passed in -h/--help?
533         if matches.opt_present("help") {
534             usage(0, &opts, verbose, &subcommand_help);
535         }
536
537         let cmd = match subcommand {
538             Kind::Build => Subcommand::Build { paths },
539             Kind::Check => {
540                 if matches.opt_present("all-targets") {
541                     println!(
542                         "Warning: --all-targets is now on by default and does not need to be passed explicitly."
543                     );
544                 }
545                 Subcommand::Check { paths }
546             }
547             Kind::Clippy => Subcommand::Clippy { paths, fix: matches.opt_present("fix") },
548             Kind::Fix => Subcommand::Fix { paths },
549             Kind::Test => Subcommand::Test {
550                 paths,
551                 bless: matches.opt_present("bless"),
552                 force_rerun: matches.opt_present("force-rerun"),
553                 compare_mode: matches.opt_str("compare-mode"),
554                 pass: matches.opt_str("pass"),
555                 run: matches.opt_str("run"),
556                 skip: matches.opt_strs("skip"),
557                 test_args: matches.opt_strs("test-args"),
558                 rustc_args: matches.opt_strs("rustc-args"),
559                 fail_fast: !matches.opt_present("no-fail-fast"),
560                 rustfix_coverage: matches.opt_present("rustfix-coverage"),
561                 doc_tests: if matches.opt_present("doc") {
562                     DocTests::Only
563                 } else if matches.opt_present("no-doc") {
564                     DocTests::No
565                 } else {
566                     DocTests::Yes
567                 },
568             },
569             Kind::Bench => Subcommand::Bench { paths, test_args: matches.opt_strs("test-args") },
570             Kind::Doc => Subcommand::Doc { paths, open: matches.opt_present("open") },
571             Kind::Clean => {
572                 if !paths.is_empty() {
573                     println!("\nclean does not take a path argument\n");
574                     usage(1, &opts, verbose, &subcommand_help);
575                 }
576
577                 Subcommand::Clean { all: matches.opt_present("all") }
578             }
579             Kind::Format => Subcommand::Format { check: matches.opt_present("check"), paths },
580             Kind::Dist => Subcommand::Dist { paths },
581             Kind::Install => Subcommand::Install { paths },
582             Kind::Run => {
583                 if paths.is_empty() {
584                     println!("\nrun requires at least a path!\n");
585                     usage(1, &opts, verbose, &subcommand_help);
586                 }
587                 Subcommand::Run { paths }
588             }
589             Kind::Setup => {
590                 let profile = if paths.len() > 1 {
591                     println!("\nat most one profile can be passed to setup\n");
592                     usage(1, &opts, verbose, &subcommand_help)
593                 } else if let Some(path) = paths.pop() {
594                     let profile_string = t!(path.into_os_string().into_string().map_err(
595                         |path| format!("{} is not a valid UTF8 string", path.to_string_lossy())
596                     ));
597
598                     profile_string.parse().unwrap_or_else(|err| {
599                         eprintln!("error: {}", err);
600                         eprintln!("help: the available profiles are:");
601                         eprint!("{}", Profile::all_for_help("- "));
602                         crate::detail_exit(1);
603                     })
604                 } else {
605                     t!(crate::setup::interactive_path())
606                 };
607                 Subcommand::Setup { profile }
608             }
609         };
610
611         if let Subcommand::Check { .. } = &cmd {
612             if matches.opt_str("keep-stage").is_some()
613                 || matches.opt_str("keep-stage-std").is_some()
614             {
615                 eprintln!("--keep-stage not yet supported for x.py check");
616                 crate::detail_exit(1);
617             }
618         }
619
620         Flags {
621             verbose: matches.opt_count("verbose"),
622             stage: matches.opt_str("stage").map(|j| j.parse().expect("`stage` should be a number")),
623             dry_run: matches.opt_present("dry-run"),
624             on_fail: matches.opt_str("on-fail"),
625             rustc_error_format: matches.opt_str("error-format"),
626             json_output: matches.opt_present("json-output"),
627             keep_stage: matches
628                 .opt_strs("keep-stage")
629                 .into_iter()
630                 .map(|j| j.parse().expect("`keep-stage` should be a number"))
631                 .collect(),
632             keep_stage_std: matches
633                 .opt_strs("keep-stage-std")
634                 .into_iter()
635                 .map(|j| j.parse().expect("`keep-stage-std` should be a number"))
636                 .collect(),
637             host: if matches.opt_present("host") {
638                 Some(
639                     split(&matches.opt_strs("host"))
640                         .into_iter()
641                         .map(|x| TargetSelection::from_user(&x))
642                         .collect::<Vec<_>>(),
643                 )
644             } else {
645                 None
646             },
647             target: if matches.opt_present("target") {
648                 Some(
649                     split(&matches.opt_strs("target"))
650                         .into_iter()
651                         .map(|x| TargetSelection::from_user(&x))
652                         .collect::<Vec<_>>(),
653                 )
654             } else {
655                 None
656             },
657             config: matches.opt_str("config").map(PathBuf::from),
658             build_dir: matches.opt_str("build-dir").map(PathBuf::from),
659             jobs: matches.opt_str("jobs").map(|j| j.parse().expect("`jobs` should be a number")),
660             cmd,
661             incremental: matches.opt_present("incremental"),
662             exclude: split(&matches.opt_strs("exclude"))
663                 .into_iter()
664                 .map(|p| p.into())
665                 .collect::<Vec<_>>(),
666             include_default_paths: matches.opt_present("include-default-paths"),
667             deny_warnings: parse_deny_warnings(&matches),
668             llvm_skip_rebuild: matches.opt_str("llvm-skip-rebuild").map(|s| s.to_lowercase()).map(
669                 |s| s.parse::<bool>().expect("`llvm-skip-rebuild` should be either true or false"),
670             ),
671             color: matches
672                 .opt_get_default("color", Color::Auto)
673                 .expect("`color` should be `always`, `never`, or `auto`"),
674             rust_profile_use: matches.opt_str("rust-profile-use"),
675             rust_profile_generate: matches.opt_str("rust-profile-generate"),
676             llvm_profile_use: matches.opt_str("llvm-profile-use"),
677             llvm_profile_generate: matches.opt_present("llvm-profile-generate"),
678         }
679     }
680 }
681
682 impl Subcommand {
683     pub fn kind(&self) -> Kind {
684         match self {
685             Subcommand::Bench { .. } => Kind::Bench,
686             Subcommand::Build { .. } => Kind::Build,
687             Subcommand::Check { .. } => Kind::Check,
688             Subcommand::Clippy { .. } => Kind::Clippy,
689             Subcommand::Doc { .. } => Kind::Doc,
690             Subcommand::Fix { .. } => Kind::Fix,
691             Subcommand::Format { .. } => Kind::Format,
692             Subcommand::Test { .. } => Kind::Test,
693             Subcommand::Clean { .. } => Kind::Clean,
694             Subcommand::Dist { .. } => Kind::Dist,
695             Subcommand::Install { .. } => Kind::Install,
696             Subcommand::Run { .. } => Kind::Run,
697             Subcommand::Setup { .. } => Kind::Setup,
698         }
699     }
700
701     pub fn test_args(&self) -> Vec<&str> {
702         let mut args = vec![];
703
704         match *self {
705             Subcommand::Test { ref skip, .. } => {
706                 for s in skip {
707                     args.push("--skip");
708                     args.push(s.as_str());
709                 }
710             }
711             _ => (),
712         };
713
714         match *self {
715             Subcommand::Test { ref test_args, .. } | Subcommand::Bench { ref test_args, .. } => {
716                 args.extend(test_args.iter().flat_map(|s| s.split_whitespace()))
717             }
718             _ => (),
719         }
720
721         args
722     }
723
724     pub fn rustc_args(&self) -> Vec<&str> {
725         match *self {
726             Subcommand::Test { ref rustc_args, .. } => {
727                 rustc_args.iter().flat_map(|s| s.split_whitespace()).collect()
728             }
729             _ => Vec::new(),
730         }
731     }
732
733     pub fn fail_fast(&self) -> bool {
734         match *self {
735             Subcommand::Test { fail_fast, .. } => fail_fast,
736             _ => false,
737         }
738     }
739
740     pub fn doc_tests(&self) -> DocTests {
741         match *self {
742             Subcommand::Test { doc_tests, .. } => doc_tests,
743             _ => DocTests::Yes,
744         }
745     }
746
747     pub fn bless(&self) -> bool {
748         match *self {
749             Subcommand::Test { bless, .. } => bless,
750             _ => false,
751         }
752     }
753
754     pub fn force_rerun(&self) -> bool {
755         match *self {
756             Subcommand::Test { force_rerun, .. } => force_rerun,
757             _ => false,
758         }
759     }
760
761     pub fn rustfix_coverage(&self) -> bool {
762         match *self {
763             Subcommand::Test { rustfix_coverage, .. } => rustfix_coverage,
764             _ => false,
765         }
766     }
767
768     pub fn compare_mode(&self) -> Option<&str> {
769         match *self {
770             Subcommand::Test { ref compare_mode, .. } => compare_mode.as_ref().map(|s| &s[..]),
771             _ => None,
772         }
773     }
774
775     pub fn pass(&self) -> Option<&str> {
776         match *self {
777             Subcommand::Test { ref pass, .. } => pass.as_ref().map(|s| &s[..]),
778             _ => None,
779         }
780     }
781
782     pub fn run(&self) -> Option<&str> {
783         match *self {
784             Subcommand::Test { ref run, .. } => run.as_ref().map(|s| &s[..]),
785             _ => None,
786         }
787     }
788
789     pub fn open(&self) -> bool {
790         match *self {
791             Subcommand::Doc { open, .. } => open,
792             _ => false,
793         }
794     }
795 }
796
797 fn split(s: &[String]) -> Vec<String> {
798     s.iter().flat_map(|s| s.split(',')).filter(|s| !s.is_empty()).map(|s| s.to_string()).collect()
799 }
800
801 fn parse_deny_warnings(matches: &getopts::Matches) -> Option<bool> {
802     match matches.opt_str("warnings").as_deref() {
803         Some("deny") => Some(true),
804         Some("warn") => Some(false),
805         Some(value) => {
806             eprintln!(r#"invalid value for --warnings: {:?}, expected "warn" or "deny""#, value,);
807             crate::detail_exit(1);
808         }
809         None => None,
810     }
811 }