builder.rs 48.1 KB
Newer Older
1
use std::any::Any;
2
use std::cell::{Cell, RefCell};
3
use std::collections::BTreeSet;
S
Santiago Pastorino 已提交
4
use std::collections::HashMap;
5
use std::env;
6
use std::fmt::Debug;
7
use std::fs;
8
use std::hash::Hash;
9
use std::ops::Deref;
10 11
use std::path::{Path, PathBuf};
use std::process::Command;
S
Santiago Pastorino 已提交
12
use std::time::{Duration, Instant};
13

14 15
use build_helper::t;

L
ljedrz 已提交
16 17 18 19 20 21 22 23 24 25
use crate::cache::{Cache, Interned, INTERNER};
use crate::check;
use crate::compile;
use crate::dist;
use crate::doc;
use crate::flags::Subcommand;
use crate::install;
use crate::native;
use crate::test;
use crate::tool;
26
use crate::util::{self, add_lib_path, exe, libdir};
L
ljedrz 已提交
27 28 29
use crate::{Build, DocTests, Mode, GitRepo};

pub use crate::Compiler;
30

31
use petgraph::graph::NodeIndex;
S
Santiago Pastorino 已提交
32
use petgraph::Graph;
33

34 35 36 37 38
pub struct Builder<'a> {
    pub build: &'a Build,
    pub top_stage: u32,
    pub kind: Kind,
    cache: Cache,
39
    stack: RefCell<Vec<Box<dyn Any>>>,
40
    time_spent_on_dependencies: Cell<Duration>,
41
    pub paths: Vec<PathBuf>,
42 43 44
    graph_nodes: RefCell<HashMap<String, NodeIndex>>,
    graph: RefCell<Graph<String, bool>>,
    parent: Cell<Option<NodeIndex>>,
45 46 47 48 49 50 51 52 53 54
}

impl<'a> Deref for Builder<'a> {
    type Target = Build;

    fn deref(&self) -> &Self::Target {
        self.build
    }
}

55
pub trait Step: 'static + Clone + Debug + PartialEq + Eq + Hash {
56 57
    /// `PathBuf` when directories are created or to return a `Compiler` once
    /// it's been assembled.
58
    type Output: Clone;
59

60 61
    const DEFAULT: bool = false;

62
    /// If true, then this rule should be skipped if --target was specified, but --host was not
63 64
    const ONLY_HOSTS: bool = false;

A
Alexander Regueiro 已提交
65
    /// Primary function to execute this rule. Can call `builder.ensure()`
66
    /// with other steps to run those.
T
Taiki Endo 已提交
67
    fn run(self, builder: &Builder<'_>) -> Self::Output;
68

69 70
    /// When bootstrap is passed a set of paths, this controls whether this rule
    /// will execute. However, it does not get called in a "default" context
A
Alexander Regueiro 已提交
71
    /// when we are not passed any paths; in that case, `make_run` is called
72
    /// directly.
T
Taiki Endo 已提交
73
    fn should_run(run: ShouldRun<'_>) -> ShouldRun<'_>;
74

A
Alexander Regueiro 已提交
75
    /// Builds up a "root" rule, either as a default rule or from a path passed
76 77 78 79 80
    /// to us.
    ///
    /// When path is `None`, we are executing in a context where no paths were
    /// passed. When `./x.py build` is run, for example, this rule could get
    /// called if it is in the correct list below with a path of `None`.
T
Taiki Endo 已提交
81
    fn make_run(_run: RunConfig<'_>) {
82 83 84 85 86 87
        // It is reasonable to not have an implementation of make_run for rules
        // who do not want to get called from the root context. This means that
        // they are likely dependencies (e.g., sysroot creation) or similar, and
        // as such calling them from ./x.py isn't logical.
        unimplemented!()
    }
88 89
}

90 91 92 93
pub struct RunConfig<'a> {
    pub builder: &'a Builder<'a>,
    pub host: Interned<String>,
    pub target: Interned<String>,
94
    pub path: PathBuf,
95 96
}

97 98 99
struct StepDescription {
    default: bool,
    only_hosts: bool,
T
Taiki Endo 已提交
100 101
    should_run: fn(ShouldRun<'_>) -> ShouldRun<'_>,
    make_run: fn(RunConfig<'_>),
102 103 104 105
    name: &'static str,
}

#[derive(Debug, Clone, PartialOrd, Ord, PartialEq, Eq)]
106
pub enum PathSet {
S
Santiago Pastorino 已提交
107 108
    Set(BTreeSet<PathBuf>),
    Suite(PathBuf),
109 110 111
}

impl PathSet {
112
    fn empty() -> PathSet {
113
        PathSet::Set(BTreeSet::new())
114 115
    }

116 117 118
    fn one<P: Into<PathBuf>>(path: P) -> PathSet {
        let mut set = BTreeSet::new();
        set.insert(path.into());
119
        PathSet::Set(set)
120 121 122
    }

    fn has(&self, needle: &Path) -> bool {
123 124
        match self {
            PathSet::Set(set) => set.iter().any(|p| p.ends_with(needle)),
125
            PathSet::Suite(suite) => suite.ends_with(needle),
126
        }
127 128
    }

T
Taiki Endo 已提交
129
    fn path(&self, builder: &Builder<'_>) -> PathBuf {
130
        match self {
S
Santiago Pastorino 已提交
131 132 133 134 135 136
            PathSet::Set(set) => set
                .iter()
                .next()
                .unwrap_or(&builder.build.src)
                .to_path_buf(),
            PathSet::Suite(path) => PathBuf::from(path),
137
        }
138
    }
139 140 141 142 143 144 145 146 147
}

impl StepDescription {
    fn from<S: Step>() -> StepDescription {
        StepDescription {
            default: S::DEFAULT,
            only_hosts: S::ONLY_HOSTS,
            should_run: S::should_run,
            make_run: S::make_run,
148
            name: unsafe { ::std::intrinsics::type_name::<S>() },
149 150 151
        }
    }

T
Taiki Endo 已提交
152
    fn maybe_run(&self, builder: &Builder<'_>, pathset: &PathSet) {
153 154 155 156
        if builder.config.exclude.iter().any(|e| pathset.has(e)) {
            eprintln!("Skipping {:?} because it is excluded", pathset);
            return;
        } else if !builder.config.exclude.is_empty() {
S
Santiago Pastorino 已提交
157 158 159 160
            eprintln!(
                "{:?} not skipped for {:?} -- not in {:?}",
                pathset, self.name, builder.config.exclude
            );
161
        }
162
        let hosts = &builder.hosts;
163

M
Mark Simulacrum 已提交
164
        // Determine the targets participating in this rule.
165
        let targets = if self.only_hosts {
166
            if builder.config.skip_only_host_steps {
167
                return; // don't run anything
168
            } else {
169
                &builder.hosts
170 171
            }
        } else {
172
            &builder.targets
173 174 175 176
        };

        for host in hosts {
            for target in targets {
177 178
                let run = RunConfig {
                    builder,
179
                    path: pathset.path(builder),
180 181 182 183
                    host: *host,
                    target: *target,
                };
                (self.make_run)(run);
184 185 186 187
            }
        }
    }

T
Taiki Endo 已提交
188
    fn run(v: &[StepDescription], builder: &Builder<'_>, paths: &[PathBuf]) {
S
Santiago Pastorino 已提交
189 190 191 192
        let should_runs = v
            .iter()
            .map(|desc| (desc.should_run)(ShouldRun::new(builder)))
            .collect::<Vec<_>>();
193 194 195

        // sanity checks on rules
        for (desc, should_run) in v.iter().zip(&should_runs) {
S
Santiago Pastorino 已提交
196 197 198 199 200
            assert!(
                !should_run.paths.is_empty(),
                "{:?} should have at least one pathset",
                desc.name
            );
201 202
        }

203
        if paths.is_empty() {
204 205
            for (desc, should_run) in v.iter().zip(should_runs) {
                if desc.default && should_run.is_really_default {
206 207 208
                    for pathset in &should_run.paths {
                        desc.maybe_run(builder, pathset);
                    }
209 210 211 212
                }
            }
        } else {
            for path in paths {
213 214 215 216 217 218
                // strip CurDir prefix if present
                let path = match path.strip_prefix(".") {
                    Ok(p) => p,
                    Err(_) => path,
                };

219
                let mut attempted_run = false;
220
                for (desc, should_run) in v.iter().zip(&should_runs) {
221 222 223 224
                    if let Some(suite) = should_run.is_suite_path(path) {
                        attempted_run = true;
                        desc.maybe_run(builder, suite);
                    } else if let Some(pathset) = should_run.pathset_for_path(path) {
225
                        attempted_run = true;
226
                        desc.maybe_run(builder, pathset);
227 228 229 230
                    }
                }

                if !attempted_run {
231
                    panic!("Error: no rules matched {}.", path.display());
232 233 234 235 236 237
                }
            }
        }
    }
}

238 239
#[derive(Clone)]
pub struct ShouldRun<'a> {
240
    pub builder: &'a Builder<'a>,
241
    // use a BTreeSet to maintain sort order
242
    paths: BTreeSet<PathSet>,
243 244

    // If this is a default rule, this is an additional constraint placed on
245
    // its run. Generally something like compiler docs being enabled.
246
    is_really_default: bool,
247 248 249
}

impl<'a> ShouldRun<'a> {
T
Taiki Endo 已提交
250
    fn new(builder: &'a Builder<'_>) -> ShouldRun<'a> {
251
        ShouldRun {
252
            builder,
253
            paths: BTreeSet::new(),
254
            is_really_default: true, // by default no additional conditions
255 256 257
        }
    }

258 259 260 261 262
    pub fn default_condition(mut self, cond: bool) -> Self {
        self.is_really_default = cond;
        self
    }

263 264 265 266 267 268 269 270
    // Unlike `krate` this will create just one pathset. As such, it probably shouldn't actually
    // ever be used, but as we transition to having all rules properly handle passing krate(...) by
    // actually doing something different for every crate passed.
    pub fn all_krates(mut self, name: &str) -> Self {
        let mut set = BTreeSet::new();
        for krate in self.builder.in_tree_crates(name) {
            set.insert(PathBuf::from(&krate.path));
        }
271
        self.paths.insert(PathSet::Set(set));
272 273 274
        self
    }

275
    pub fn krate(mut self, name: &str) -> Self {
276 277
        for krate in self.builder.in_tree_crates(name) {
            self.paths.insert(PathSet::one(&krate.path));
278 279 280 281
        }
        self
    }

282 283 284 285 286 287 288
    // single, non-aliased path
    pub fn path(self, path: &str) -> Self {
        self.paths(&[path])
    }

    // multiple aliases for the same job
    pub fn paths(mut self, paths: &[&str]) -> Self {
S
Santiago Pastorino 已提交
289 290
        self.paths
            .insert(PathSet::Set(paths.iter().map(PathBuf::from).collect()));
291 292 293 294
        self
    }

    pub fn is_suite_path(&self, path: &Path) -> Option<&PathSet> {
S
Santiago Pastorino 已提交
295 296 297
        self.paths.iter().find(|pathset| match pathset {
            PathSet::Suite(p) => path.starts_with(p),
            PathSet::Set(_) => false,
298 299 300 301 302
        })
    }

    pub fn suite_path(mut self, suite: &str) -> Self {
        self.paths.insert(PathSet::Suite(PathBuf::from(suite)));
303 304 305 306
        self
    }

    // allows being more explicit about why should_run in Step returns the value passed to it
307 308
    pub fn never(mut self) -> ShouldRun<'a> {
        self.paths.insert(PathSet::empty());
309 310 311
        self
    }

312 313
    fn pathset_for_path(&self, path: &Path) -> Option<&PathSet> {
        self.paths.iter().find(|pathset| pathset.has(path))
314 315 316
    }
}

317 318 319
#[derive(Copy, Clone, PartialEq, Eq, Debug)]
pub enum Kind {
    Build,
320
    Check,
L
ljedrz 已提交
321 322
    Clippy,
    Fix,
323 324 325 326 327 328 329
    Test,
    Bench,
    Dist,
    Doc,
    Install,
}

330 331 332
impl<'a> Builder<'a> {
    fn get_step_descriptions(kind: Kind) -> Vec<StepDescription> {
        macro_rules! describe {
T
Taiki Endo 已提交
333
            ($($rule:ty),+ $(,)?) => {{
334 335
                vec![$(StepDescription::from::<$rule>()),+]
            }};
336
        }
337
        match kind {
S
Santiago Pastorino 已提交
338 339 340 341
            Kind::Build => describe!(
                compile::Std,
                compile::Test,
                compile::Rustc,
342
                compile::CodegenBackend,
S
Santiago Pastorino 已提交
343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363
                compile::StartupObjects,
                tool::BuildManifest,
                tool::Rustbook,
                tool::ErrorIndex,
                tool::UnstableBookGen,
                tool::Tidy,
                tool::Linkchecker,
                tool::CargoTest,
                tool::Compiletest,
                tool::RemoteTestServer,
                tool::RemoteTestClient,
                tool::RustInstaller,
                tool::Cargo,
                tool::Rls,
                tool::Rustdoc,
                tool::Clippy,
                native::Llvm,
                tool::Rustfmt,
                tool::Miri,
                native::Lld
            ),
L
ljedrz 已提交
364
            Kind::Check | Kind::Clippy | Kind::Fix => describe!(
S
Santiago Pastorino 已提交
365 366 367 368 369 370 371 372 373 374 375 376 377 378 379
                check::Std,
                check::Test,
                check::Rustc,
                check::CodegenBackend,
                check::Rustdoc
            ),
            Kind::Test => describe!(
                test::Tidy,
                test::Ui,
                test::CompileFail,
                test::RunFail,
                test::RunPassValgrind,
                test::MirOpt,
                test::Codegen,
                test::CodegenUnits,
380
                test::Assembly,
S
Santiago Pastorino 已提交
381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396
                test::Incremental,
                test::Debuginfo,
                test::UiFullDeps,
                test::Rustdoc,
                test::Pretty,
                test::RunFailPretty,
                test::RunPassValgrindPretty,
                test::Crate,
                test::CrateLibrustc,
                test::CrateRustdoc,
                test::Linkcheck,
                test::Cargotest,
                test::Cargo,
                test::Rls,
                test::ErrorIndex,
                test::Distcheck,
397
                test::RunMakeFullDeps,
S
Santiago Pastorino 已提交
398 399 400 401 402 403 404
                test::Nomicon,
                test::Reference,
                test::RustdocBook,
                test::RustByExample,
                test::TheBook,
                test::UnstableBook,
                test::RustcBook,
405
                test::RustcGuide,
406
                test::EmbeddedBook,
E
Eric Huss 已提交
407
                test::EditionGuide,
S
Santiago Pastorino 已提交
408 409 410
                test::Rustfmt,
                test::Miri,
                test::Clippy,
411
                test::CompiletestTest,
G
Guillaume Gomez 已提交
412
                test::RustdocJSStd,
G
Guillaume Gomez 已提交
413
                test::RustdocJSNotStd,
S
Santiago Pastorino 已提交
414
                test::RustdocTheme,
J
John Kåre Alsaker 已提交
415
                test::RustdocUi,
416 417
                // Run bootstrap close to the end as it's unlikely to fail
                test::Bootstrap,
418
                // Run run-make last, since these won't pass without make on Windows
S
Santiago Pastorino 已提交
419 420
                test::RunMake,
            ),
421
            Kind::Bench => describe!(test::Crate, test::CrateLibrustc),
S
Santiago Pastorino 已提交
422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437
            Kind::Doc => describe!(
                doc::UnstableBook,
                doc::UnstableBookGen,
                doc::TheBook,
                doc::Standalone,
                doc::Std,
                doc::Test,
                doc::WhitelistedRustc,
                doc::Rustc,
                doc::Rustdoc,
                doc::ErrorIndex,
                doc::Nomicon,
                doc::Reference,
                doc::RustdocBook,
                doc::RustByExample,
                doc::RustcBook,
S
Steve Klabnik 已提交
438
                doc::CargoBook,
J
James Munns 已提交
439
                doc::EmbeddedBook,
S
Steve Klabnik 已提交
440
                doc::EditionGuide,
S
Santiago Pastorino 已提交
441 442 443 444 445 446 447 448 449 450 451 452 453 454
            ),
            Kind::Dist => describe!(
                dist::Docs,
                dist::RustcDocs,
                dist::Mingw,
                dist::Rustc,
                dist::DebuggerScripts,
                dist::Std,
                dist::Analysis,
                dist::Src,
                dist::PlainSourceTarball,
                dist::Cargo,
                dist::Rls,
                dist::Rustfmt,
455
                dist::Clippy,
456
                dist::Miri,
457
                dist::LlvmTools,
T
Tom Tromey 已提交
458
                dist::Lldb,
S
Santiago Pastorino 已提交
459 460 461 462 463 464 465 466 467
                dist::Extended,
                dist::HashSign
            ),
            Kind::Install => describe!(
                install::Docs,
                install::Std,
                install::Cargo,
                install::Rls,
                install::Rustfmt,
468
                install::Clippy,
469
                install::Miri,
S
Santiago Pastorino 已提交
470 471 472 473
                install::Analysis,
                install::Src,
                install::Rustc
            ),
474 475
        }
    }
476

477 478 479 480 481 482 483 484 485 486 487 488
    pub fn get_help(build: &Build, subcommand: &str) -> Option<String> {
        let kind = match subcommand {
            "build" => Kind::Build,
            "doc" => Kind::Doc,
            "test" => Kind::Test,
            "bench" => Kind::Bench,
            "dist" => Kind::Dist,
            "install" => Kind::Install,
            _ => return None,
        };

        let builder = Builder {
489
            build,
M
Mark Simulacrum 已提交
490
            top_stage: build.config.stage.unwrap_or(2),
491
            kind,
492 493
            cache: Cache::new(),
            stack: RefCell::new(Vec::new()),
494
            time_spent_on_dependencies: Cell::new(Duration::new(0, 0)),
495
            paths: vec![],
496 497 498
            graph_nodes: RefCell::new(HashMap::new()),
            graph: RefCell::new(Graph::new()),
            parent: Cell::new(None),
499 500 501 502
        };

        let builder = &builder;
        let mut should_run = ShouldRun::new(builder);
503 504
        for desc in Builder::get_step_descriptions(builder.kind) {
            should_run = (desc.should_run)(should_run);
505 506
        }
        let mut help = String::from("Available paths:\n");
507
        for pathset in should_run.paths {
S
Santiago Pastorino 已提交
508 509 510 511 512 513
            if let PathSet::Set(set) = pathset {
                set.iter().for_each(|path| {
                    help.push_str(
                        format!("    ./x.py {} {}\n", subcommand, path.display()).as_str(),
                    )
                })
514
            }
515 516 517 518
        }
        Some(help)
    }

T
Taiki Endo 已提交
519
    pub fn new(build: &Build) -> Builder<'_> {
M
Mark Simulacrum 已提交
520
        let (kind, paths) = match build.config.cmd {
521
            Subcommand::Build { ref paths } => (Kind::Build, &paths[..]),
522
            Subcommand::Check { ref paths } => (Kind::Check, &paths[..]),
L
ljedrz 已提交
523 524
            Subcommand::Clippy { ref paths } => (Kind::Clippy, &paths[..]),
            Subcommand::Fix { ref paths } => (Kind::Fix, &paths[..]),
525 526 527 528 529
            Subcommand::Doc { ref paths } => (Kind::Doc, &paths[..]),
            Subcommand::Test { ref paths, .. } => (Kind::Test, &paths[..]),
            Subcommand::Bench { ref paths, .. } => (Kind::Bench, &paths[..]),
            Subcommand::Dist { ref paths } => (Kind::Dist, &paths[..]),
            Subcommand::Install { ref paths } => (Kind::Install, &paths[..]),
O
Oliver Schneider 已提交
530
            Subcommand::Clean { .. } => panic!(),
531 532 533
        };

        let builder = Builder {
534
            build,
M
Mark Simulacrum 已提交
535
            top_stage: build.config.stage.unwrap_or(2),
536
            kind,
537 538
            cache: Cache::new(),
            stack: RefCell::new(Vec::new()),
539
            time_spent_on_dependencies: Cell::new(Duration::new(0, 0)),
540
            paths: paths.to_owned(),
541 542 543
            graph_nodes: RefCell::new(HashMap::new()),
            graph: RefCell::new(Graph::new()),
            parent: Cell::new(None),
544 545
        };

546
        if kind == Kind::Dist {
S
Santiago Pastorino 已提交
547 548 549
            assert!(
                !builder.config.test_miri,
                "Do not distribute with miri enabled.\n\
550
                The distributed libraries would include all MIR (increasing binary size).
S
Santiago Pastorino 已提交
551 552
                The distributed MIR would include validation statements."
            );
553 554
        }

555 556 557
        builder
    }

558
    pub fn execute_cli(&self) -> Graph<String, bool> {
M
Mark Simulacrum 已提交
559
        self.run_step_descriptions(&Builder::get_step_descriptions(self.kind), &self.paths);
560
        self.graph.borrow().clone()
561 562 563 564
    }

    pub fn default_doc(&self, paths: Option<&[PathBuf]>) {
        let paths = paths.unwrap_or(&[]);
M
Mark Simulacrum 已提交
565 566 567 568 569
        self.run_step_descriptions(&Builder::get_step_descriptions(Kind::Doc), paths);
    }

    fn run_step_descriptions(&self, v: &[StepDescription], paths: &[PathBuf]) {
        StepDescription::run(v, self, paths);
570 571
    }

B
Bastien Orivel 已提交
572
    /// Obtain a compiler at a given stage and for a given host. Explicitly does
573 574 575
    /// not take `Compiler` since all `Compiler` instances are meant to be
    /// obtained through this function, since it ensures that they are valid
    /// (i.e., built and assembled).
576
    pub fn compiler(&self, stage: u32, host: Interned<String>) -> Compiler {
S
Santiago Pastorino 已提交
577 578 579
        self.ensure(compile::Assemble {
            target_compiler: Compiler { stage, host },
        })
580 581
    }

582 583 584 585 586 587 588 589 590 591 592
    /// Similar to `compiler`, except handles the full-bootstrap option to
    /// silently use the stage1 compiler instead of a stage2 compiler if one is
    /// requested.
    ///
    /// Note that this does *not* have the side effect of creating
    /// `compiler(stage, host)`, unlike `compiler` above which does have such
    /// a side effect. The returned compiler here can only be used to compile
    /// new artifacts, it can't be used to rely on the presence of a particular
    /// sysroot.
    ///
    /// See `force_use_stage1` for documentation on what each argument is.
A
Alex Crichton 已提交
593 594 595 596 597 598
    pub fn compiler_for(
        &self,
        stage: u32,
        host: Interned<String>,
        target: Interned<String>,
    ) -> Compiler {
599 600 601 602 603 604 605
        if self.build.force_use_stage1(Compiler { stage, host }, target) {
            self.compiler(1, self.config.build)
        } else {
            self.compiler(stage, host)
        }
    }

606
    pub fn sysroot(&self, compiler: Compiler) -> Interned<PathBuf> {
607 608 609 610 611
        self.ensure(compile::Sysroot { compiler })
    }

    /// Returns the libdir where the standard library and other artifacts are
    /// found for a compiler's sysroot.
612
    pub fn sysroot_libdir(
S
Santiago Pastorino 已提交
613 614 615
        &self,
        compiler: Compiler,
        target: Interned<String>,
616 617 618 619 620
    ) -> Interned<PathBuf> {
        #[derive(Debug, Copy, Clone, Hash, PartialEq, Eq)]
        struct Libdir {
            compiler: Compiler,
            target: Interned<String>,
621
        }
622 623
        impl Step for Libdir {
            type Output = Interned<PathBuf>;
624

T
Taiki Endo 已提交
625
            fn should_run(run: ShouldRun<'_>) -> ShouldRun<'_> {
626
                run.never()
627 628
            }

T
Taiki Endo 已提交
629
            fn run(self, builder: &Builder<'_>) -> Interned<PathBuf> {
630
                let compiler = self.compiler;
631 632 633
                let config = &builder.build.config;
                let lib = if compiler.stage >= 1 && config.libdir_relative().is_some() {
                    builder.build.config.libdir_relative().unwrap()
634
                } else {
635
                    Path::new("lib")
636
                };
S
Santiago Pastorino 已提交
637 638 639 640 641 642
                let sysroot = builder
                    .sysroot(self.compiler)
                    .join(lib)
                    .join("rustlib")
                    .join(self.target)
                    .join("lib");
643 644
                let _ = fs::remove_dir_all(&sysroot);
                t!(fs::create_dir_all(&sysroot));
645
                INTERNER.intern_path(sysroot)
646 647 648 649 650
            }
        }
        self.ensure(Libdir { compiler, target })
    }

651 652
    pub fn sysroot_codegen_backends(&self, compiler: Compiler) -> PathBuf {
        self.sysroot_libdir(compiler, compiler.host)
653
            .with_file_name(self.config.rust_codegen_backends_dir.clone())
654 655
    }

656 657 658 659 660 661 662
    /// Returns the compiler's libdir where it stores the dynamic libraries that
    /// it itself links against.
    ///
    /// For example this returns `<sysroot>/lib` on Unix and `<sysroot>/bin` on
    /// Windows.
    pub fn rustc_libdir(&self, compiler: Compiler) -> PathBuf {
        if compiler.is_snapshot(self) {
663
            self.rustc_snapshot_libdir()
664
        } else {
O
O01eg 已提交
665 666 667 668 669 670 671 672 673 674 675 676 677 678 679 680 681 682 683 684 685 686
            match self.config.libdir_relative() {
                Some(relative_libdir) if compiler.stage >= 1
                    => self.sysroot(compiler).join(relative_libdir),
                _ => self.sysroot(compiler).join(libdir(&compiler.host))
            }
        }
    }

    /// Returns the compiler's relative libdir where it stores the dynamic libraries that
    /// it itself links against.
    ///
    /// For example this returns `lib` on Unix and `bin` on
    /// Windows.
    pub fn libdir_relative(&self, compiler: Compiler) -> &Path {
        if compiler.is_snapshot(self) {
            libdir(&self.config.build).as_ref()
        } else {
            match self.config.libdir_relative() {
                Some(relative_libdir) if compiler.stage >= 1
                    => relative_libdir,
                _ => libdir(&compiler.host).as_ref()
            }
687 688 689 690 691 692 693 694 695 696
        }
    }

    /// Adds the compiler's directory of dynamic libraries to `cmd`'s dynamic
    /// library lookup path.
    pub fn add_rustc_lib_path(&self, compiler: Compiler, cmd: &mut Command) {
        // Windows doesn't need dylib path munging because the dlls for the
        // compiler live next to the compiler and the system will find them
        // automatically.
        if cfg!(windows) {
S
Santiago Pastorino 已提交
697
            return;
698 699 700 701 702
        }

        add_lib_path(vec![self.rustc_libdir(compiler)], cmd);
    }

A
Alexander Regueiro 已提交
703
    /// Gets a path to the compiler specified.
704 705 706 707
    pub fn rustc(&self, compiler: Compiler) -> PathBuf {
        if compiler.is_snapshot(self) {
            self.initial_rustc.clone()
        } else {
S
Santiago Pastorino 已提交
708 709 710
            self.sysroot(compiler)
                .join("bin")
                .join(exe("rustc", &compiler.host))
711 712 713
        }
    }

A
Alexander Regueiro 已提交
714
    /// Gets the paths to all of the compiler's codegen backends.
715 716 717 718 719 720 721 722
    fn codegen_backends(&self, compiler: Compiler) -> impl Iterator<Item = PathBuf> {
        fs::read_dir(self.sysroot_codegen_backends(compiler))
            .into_iter()
            .flatten()
            .filter_map(Result::ok)
            .map(|entry| entry.path())
    }

M
Mark Rousskov 已提交
723 724
    pub fn rustdoc(&self, compiler: Compiler) -> PathBuf {
        self.ensure(tool::Rustdoc { compiler })
M
Mark Simulacrum 已提交
725 726
    }

M
Mark Rousskov 已提交
727
    pub fn rustdoc_cmd(&self, compiler: Compiler) -> Command {
M
Mark Simulacrum 已提交
728
        let mut cmd = Command::new(&self.out.join("bootstrap/debug/rustdoc"));
O
Oliver Schneider 已提交
729
        cmd.env("RUSTC_STAGE", compiler.stage.to_string())
S
Santiago Pastorino 已提交
730
            .env("RUSTC_SYSROOT", self.sysroot(compiler))
M
Mark Rousskov 已提交
731 732 733
            // Note that this is *not* the sysroot_libdir because rustdoc must be linked
            // equivalently to rustc.
            .env("RUSTDOC_LIBDIR", self.rustc_libdir(compiler))
S
Santiago Pastorino 已提交
734
            .env("CFG_RELEASE_CHANNEL", &self.config.channel)
M
Mark Rousskov 已提交
735
            .env("RUSTDOC_REAL", self.rustdoc(compiler))
S
Santiago Pastorino 已提交
736 737
            .env("RUSTDOC_CRATE_VERSION", self.rust_version())
            .env("RUSTC_BOOTSTRAP", "1");
738 739 740 741 742

        // Remove make-related flags that can cause jobserver problems.
        cmd.env_remove("MAKEFLAGS");
        cmd.env_remove("MFLAGS");

M
Mark Rousskov 已提交
743
        if let Some(linker) = self.linker(compiler.host) {
O
Oliver Schneider 已提交
744 745
            cmd.env("RUSTC_TARGET_LINKER", linker);
        }
M
Mark Simulacrum 已提交
746
        cmd
747 748
    }

749 750 751 752 753 754 755
    /// Prepares an invocation of `cargo` to be run.
    ///
    /// This will create a `Command` that represents a pending execution of
    /// Cargo. This cargo will be configured to use `compiler` as the actual
    /// rustc compiler, its output will be scoped by `mode`'s output directory,
    /// it will pass the `--target` flag for the specified `target`, and will be
    /// executing the Cargo command `cmd`.
S
Santiago Pastorino 已提交
756 757 758 759 760 761 762
    pub fn cargo(
        &self,
        compiler: Compiler,
        mode: Mode,
        target: Interned<String>,
        cmd: &str,
    ) -> Command {
M
Mark Simulacrum 已提交
763 764
        let mut cargo = Command::new(&self.initial_cargo);
        let out_dir = self.stage_out(compiler, mode);
765

766
        // command specific path, we call clear_if_dirty with this
767 768 769 770
        let mut my_out = match cmd {
            "build" => self.cargo_out(compiler, mode, target),

            // This is the intended out directory for crate documentation.
771
            "doc" | "rustdoc" =>  self.crate_doc_out(target),
772 773 774 775 776 777 778

            _ => self.stage_out(compiler, mode),
        };

        // This is for the original compiler, but if we're forced to use stage 1, then
        // std/test/rustc stamps won't exist in stage 2, so we need to get those from stage 1, since
        // we copy the libs forward.
779
        let cmp = self.compiler_for(compiler.stage, compiler.host, target);
780 781

        let libstd_stamp = match cmd {
L
ljedrz 已提交
782
            "check" | "clippy" | "fix" => check::libstd_stamp(self, cmp, target),
783
            _ => compile::libstd_stamp(self, cmp, target),
784 785 786
        };

        let libtest_stamp = match cmd {
L
ljedrz 已提交
787
            "check" | "clippy" | "fix" => check::libtest_stamp(self, cmp, target),
788
            _ => compile::libtest_stamp(self, cmp, target),
789 790 791
        };

        let librustc_stamp = match cmd {
L
ljedrz 已提交
792
            "check" | "clippy" | "fix" => check::librustc_stamp(self, cmp, target),
793
            _ => compile::librustc_stamp(self, cmp, target),
794 795
        };

796
        if cmd == "doc" || cmd == "rustdoc" {
797
            if mode == Mode::Rustc || mode == Mode::ToolRustc || mode == Mode::Codegen {
798 799 800
                // This is the intended out directory for compiler documentation.
                my_out = self.compiler_doc_out(target);
            }
M
Mark Rousskov 已提交
801
            let rustdoc = self.rustdoc(compiler);
802
            self.clear_if_dirty(&my_out, &rustdoc);
803
        } else if cmd != "test" {
804 805 806
            match mode {
                Mode::Std => {
                    self.clear_if_dirty(&my_out, &self.rustc(compiler));
807 808 809
                    for backend in self.codegen_backends(compiler) {
                        self.clear_if_dirty(&my_out, &backend);
                    }
810
                },
811 812 813
                Mode::Test => {
                    self.clear_if_dirty(&my_out, &libstd_stamp);
                },
814
                Mode::Rustc => {
815
                    self.clear_if_dirty(&my_out, &self.rustc(compiler));
816 817 818
                    self.clear_if_dirty(&my_out, &libstd_stamp);
                    self.clear_if_dirty(&my_out, &libtest_stamp);
                },
C
Collins Abitekaniza 已提交
819
                Mode::Codegen => {
820
                    self.clear_if_dirty(&my_out, &librustc_stamp);
C
Collins Abitekaniza 已提交
821 822 823 824 825 826 827 828 829
                },
                Mode::ToolBootstrap => { },
                Mode::ToolStd => {
                    self.clear_if_dirty(&my_out, &libstd_stamp);
                },
                Mode::ToolTest => {
                    self.clear_if_dirty(&my_out, &libstd_stamp);
                    self.clear_if_dirty(&my_out, &libtest_stamp);
                },
830 831 832 833
                Mode::ToolRustc => {
                    self.clear_if_dirty(&my_out, &libstd_stamp);
                    self.clear_if_dirty(&my_out, &libtest_stamp);
                    self.clear_if_dirty(&my_out, &librustc_stamp);
834
                },
835 836 837
            }
        }

S
Santiago Pastorino 已提交
838 839
        cargo
            .env("CARGO_TARGET_DIR", out_dir)
840 841
            .arg(cmd);

842 843 844 845 846 847 848
        // See comment in librustc_llvm/build.rs for why this is necessary, largely llvm-config
        // needs to not accidentally link to libLLVM in stage0/lib.
        cargo.env("REAL_LIBRARY_PATH_VAR", &util::dylib_path_var());
        if let Some(e) = env::var_os(util::dylib_path_var()) {
            cargo.env("REAL_LIBRARY_PATH", e);
        }

849 850 851 852 853 854
        if cmd != "install" {
            cargo.arg("--target")
                 .arg(target);
        } else {
            assert_eq!(target, compiler.host);
        }
855

L
ljedrz 已提交
856 857 858
        // Set a flag for `check`/`clippy`/`fix`, so that certain build
        // scripts can do less work (e.g. not building/requiring LLVM).
        if cmd == "check" || cmd == "clippy" || cmd == "fix" {
V
varkor 已提交
859 860 861
            cargo.env("RUST_CHECK", "1");
        }

J
John Kåre Alsaker 已提交
862 863 864 865 866 867 868 869 870 871 872
        match mode {
            Mode::Std | Mode::Test | Mode::ToolBootstrap | Mode::ToolStd | Mode::ToolTest=> {},
            Mode::Rustc | Mode::Codegen | Mode::ToolRustc => {
                // Build proc macros both for the host and the target
                if target != compiler.host && cmd != "check" {
                    cargo.arg("-Zdual-proc-macros");
                    cargo.env("RUST_DUAL_PROC_MACROS", "1");
                }
            },
        }

873 874 875 876
        cargo.arg("-j").arg(self.jobs().to_string());
        // Remove make-related flags to ensure Cargo can correctly set things up
        cargo.env_remove("MAKEFLAGS");
        cargo.env_remove("MFLAGS");
877

878 879
        // FIXME: Temporary fix for https://github.com/rust-lang/cargo/issues/3005
        // Force cargo to output binaries with disambiguating hashes in the name
880 881
        let mut metadata = if compiler.stage == 0 {
            // Treat stage0 like a special channel, whether it's a normal prior-
882 883
            // release rustc or a local rebuild with the same version, so we
            // never mix these libraries by accident.
884
            "bootstrap".to_string()
885
        } else {
886
            self.config.channel.to_string()
887
        };
888 889 890 891 892 893 894 895 896 897 898 899 900 901 902 903
        // We want to make sure that none of the dependencies between
        // std/test/rustc unify with one another. This is done for weird linkage
        // reasons but the gist of the problem is that if librustc, libtest, and
        // libstd all depend on libc from crates.io (which they actually do) we
        // want to make sure they all get distinct versions. Things get really
        // weird if we try to unify all these dependencies right now, namely
        // around how many times the library is linked in dynamic libraries and
        // such. If rustc were a static executable or if we didn't ship dylibs
        // this wouldn't be a problem, but we do, so it is. This is in general
        // just here to make sure things build right. If you can remove this and
        // things still build right, please do!
        match mode {
            Mode::Std => metadata.push_str("std"),
            Mode::Test => metadata.push_str("test"),
            _ => {},
        }
904
        cargo.env("__CARGO_DEFAULT_LIB_METADATA", &metadata);
905 906

        let stage;
M
Mark Simulacrum 已提交
907
        if compiler.stage == 0 && self.local_rebuild {
908 909 910 911 912 913
            // Assume the local-rebuild rustc already has stage1 features.
            stage = 1;
        } else {
            stage = compiler.stage;
        }

914 915 916
        let mut extra_args = env::var(&format!("RUSTFLAGS_STAGE_{}", stage)).unwrap_or_default();
        if stage != 0 {
            let s = env::var("RUSTFLAGS_STAGE_NOT_0").unwrap_or_default();
917 918 919
            if !extra_args.is_empty() {
                extra_args.push_str(" ");
            }
920 921 922
            extra_args.push_str(&s);
        }

L
ljedrz 已提交
923 924 925 926 927
        if cmd == "clippy" {
            extra_args.push_str("-Zforce-unstable-if-unmarked -Zunstable-options \
                --json-rendered=termcolor");
        }

928
        if !extra_args.is_empty() {
S
Santiago Pastorino 已提交
929 930 931 932 933 934 935 936
            cargo.env(
                "RUSTFLAGS",
                format!(
                    "{} {}",
                    env::var("RUSTFLAGS").unwrap_or_default(),
                    extra_args
                ),
            );
937 938
        }

K
kennytm 已提交
939
        let want_rustdoc = self.doc_tests != DocTests::No;
K
kennytm 已提交
940

941 942 943 944 945 946
        // We synthetically interpret a stage0 compiler used to build tools as a
        // "raw" compiler in that it's the exact snapshot we download. Normally
        // the stage0 build means it uses libraries build by the stage0
        // compiler, but for tools we just use the precompiled libraries that
        // we've downloaded
        let use_snapshot = mode == Mode::ToolBootstrap;
947
        assert!(!use_snapshot || stage == 0 || self.local_rebuild);
948 949 950 951 952 953 954

        let maybe_sysroot = self.sysroot(compiler);
        let sysroot = if use_snapshot {
            self.rustc_snapshot_sysroot()
        } else {
            &maybe_sysroot
        };
M
Mark Rousskov 已提交
955
        let libdir = self.rustc_libdir(compiler);
956

957 958
        // Customize the compiler we're running. Specify the compiler to cargo
        // as our shim and then pass it some various options used to configure
M
Mark Simulacrum 已提交
959
        // how the actual compiler itself is called.
960 961 962
        //
        // These variables are primarily all read by
        // src/bootstrap/bin/{rustc.rs,rustdoc.rs}
S
Santiago Pastorino 已提交
963 964 965 966 967 968 969 970 971
        cargo
            .env("RUSTBUILD_NATIVE_DIR", self.native_dir(target))
            .env("RUSTC", self.out.join("bootstrap/debug/rustc"))
            .env("RUSTC_REAL", self.rustc(compiler))
            .env("RUSTC_STAGE", stage.to_string())
            .env(
                "RUSTC_DEBUG_ASSERTIONS",
                self.config.rust_debug_assertions.to_string(),
            )
972 973
            .env("RUSTC_SYSROOT", &sysroot)
            .env("RUSTC_LIBDIR", &libdir)
S
Santiago Pastorino 已提交
974 975 976 977
            .env("RUSTC_RPATH", self.config.rust_rpath.to_string())
            .env("RUSTDOC", self.out.join("bootstrap/debug/rustdoc"))
            .env(
                "RUSTDOC_REAL",
978
                if cmd == "doc" || cmd == "rustdoc" || (cmd == "test" && want_rustdoc) {
M
Mark Rousskov 已提交
979
                    self.rustdoc(compiler)
S
Santiago Pastorino 已提交
980 981 982 983 984
                } else {
                    PathBuf::from("/path/to/nowhere/rustdoc/not/required")
                },
            )
            .env("RUSTC_ERROR_METADATA_DST", self.extended_error_dir());
985

986
        if let Some(host_linker) = self.linker(compiler.host) {
O
Oliver Schneider 已提交
987 988
            cargo.env("RUSTC_HOST_LINKER", host_linker);
        }
989
        if let Some(target_linker) = self.linker(target) {
O
Oliver Schneider 已提交
990 991
            cargo.env("RUSTC_TARGET_LINKER", target_linker);
        }
P
penpalperson 已提交
992 993 994
        if let Some(ref error_format) = self.config.rustc_error_format {
            cargo.env("RUSTC_ERROR_FORMAT", error_format);
        }
L
ljedrz 已提交
995
        if !(["build", "check", "clippy", "fix", "rustc"].contains(&cmd)) && want_rustdoc {
M
Mark Rousskov 已提交
996
            cargo.env("RUSTDOC_LIBDIR", self.rustc_libdir(compiler));
997
        }
998

999 1000 1001 1002 1003 1004 1005 1006 1007
        let debuginfo_level = match mode {
            Mode::Rustc | Mode::Codegen => self.config.rust_debuginfo_level_rustc,
            Mode::Std | Mode::Test => self.config.rust_debuginfo_level_std,
            Mode::ToolBootstrap | Mode::ToolStd |
            Mode::ToolTest | Mode::ToolRustc => self.config.rust_debuginfo_level_tools,
        };
        cargo.env("RUSTC_DEBUGINFO_LEVEL", debuginfo_level.to_string());

        if !mode.is_tool() {
O
Oliver Schneider 已提交
1008
            cargo.env("RUSTC_FORCE_UNSTABLE", "1");
1009 1010

            // Currently the compiler depends on crates from crates.io, and
1011
            // then other crates can depend on the compiler (e.g., proc-macro
M
Mark Simulacrum 已提交
1012
            // crates). Let's say, for example that rustc itself depends on the
1013 1014
            // bitflags crate. If an external crate then depends on the
            // bitflags crate as well, we need to make sure they don't
B
Bastien Orivel 已提交
1015
            // conflict, even if they pick the same version of bitflags. We'll
1016
            // want to make sure that e.g., a plugin and rustc each get their
1017 1018 1019 1020 1021 1022 1023 1024 1025 1026 1027 1028
            // own copy of bitflags.

            // Cargo ensures that this works in general through the -C metadata
            // flag. This flag will frob the symbols in the binary to make sure
            // they're different, even though the source code is the exact
            // same. To solve this problem for the compiler we extend Cargo's
            // already-passed -C metadata flag with our own. Our rustc.rs
            // wrapper around the actual rustc will detect -C metadata being
            // passed and frob it with this extra string we're passing in.
            cargo.env("RUSTC_METADATA_SUFFIX", "rustc");
        }

1029 1030 1031 1032
        if let Some(x) = self.crt_static(target) {
            cargo.env("RUSTC_CRT_STATIC", x.to_string());
        }

1033 1034 1035 1036
        if let Some(x) = self.crt_static(compiler.host) {
            cargo.env("RUSTC_HOST_CRT_STATIC", x.to_string());
        }

1037 1038 1039 1040
        if let Some(map) = self.build.debuginfo_map(GitRepo::Rustc) {
            cargo.env("RUSTC_DEBUGINFO_MAP", map);
        }

1041 1042
        // Enable usage of unstable features
        cargo.env("RUSTC_BOOTSTRAP", "1");
M
Mark Simulacrum 已提交
1043
        self.add_rust_test_threads(&mut cargo);
1044 1045 1046

        // Almost all of the crates that we compile as part of the bootstrap may
        // have a build script, including the standard library. To compile a
M
Mark Simulacrum 已提交
1047
        // build script, however, it itself needs a standard library! This
1048
        // introduces a bit of a pickle when we're compiling the standard
M
Mark Simulacrum 已提交
1049
        // library itself.
1050 1051
        //
        // To work around this we actually end up using the snapshot compiler
M
Mark Simulacrum 已提交
1052
        // (stage0) for compiling build scripts of the standard library itself.
1053 1054 1055 1056 1057
        // The stage0 compiler is guaranteed to have a libstd available for use.
        //
        // For other crates, however, we know that we've already got a standard
        // library up and running, so we can use the normal compiler to compile
        // build scripts in that situation.
B
bjorn3 已提交
1058
        if mode == Mode::Std {
S
Santiago Pastorino 已提交
1059 1060 1061
            cargo
                .env("RUSTC_SNAPSHOT", &self.initial_rustc)
                .env("RUSTC_SNAPSHOT_LIBDIR", self.rustc_snapshot_libdir());
1062
        } else {
S
Santiago Pastorino 已提交
1063 1064 1065
            cargo
                .env("RUSTC_SNAPSHOT", self.rustc(compiler))
                .env("RUSTC_SNAPSHOT_LIBDIR", self.rustc_libdir(compiler));
1066 1067
        }

1068
        if self.config.incremental {
1069
            cargo.env("CARGO_INCREMENTAL", "1");
1070 1071 1072
        } else {
            // Don't rely on any default setting for incr. comp. in Cargo
            cargo.env("CARGO_INCREMENTAL", "0");
1073 1074
        }

M
Mark Simulacrum 已提交
1075
        if let Some(ref on_fail) = self.config.on_fail {
1076 1077 1078
            cargo.env("RUSTC_ON_FAIL", on_fail);
        }

1079 1080 1081 1082
        if self.config.print_step_timings {
            cargo.env("RUSTC_PRINT_STEP_TIMINGS", "1");
        }

J
John Kåre Alsaker 已提交
1083 1084 1085 1086
        if self.config.backtrace_on_ice {
            cargo.env("RUSTC_BACKTRACE_ON_ICE", "1");
        }

L
ljedrz 已提交
1087
        cargo.env("RUSTC_VERBOSE", self.verbosity.to_string());
1088

V
varkor 已提交
1089
        if self.config.deny_warnings {
1090 1091 1092
            cargo.env("RUSTC_DENY_WARNINGS", "1");
        }

O
Oliver Schneider 已提交
1093 1094 1095 1096 1097
        // Throughout the build Cargo can execute a number of build scripts
        // compiling C/C++ code and we need to pass compilers, archivers, flags, etc
        // obtained previously to those build scripts.
        // Build scripts use either the `cc` crate or `configure/make` so we pass
        // the options through environment variables that are fetched and understood by both.
1098 1099
        //
        // FIXME: the guard against msvc shouldn't need to be here
1100 1101 1102 1103 1104
        if target.contains("msvc") {
            if let Some(ref cl) = self.config.llvm_clang_cl {
                cargo.env("CC", cl).env("CXX", cl);
            }
        } else {
1105 1106 1107 1108 1109 1110 1111 1112 1113 1114 1115 1116 1117 1118 1119 1120 1121
            let ccache = self.config.ccache.as_ref();
            let ccacheify = |s: &Path| {
                let ccache = match ccache {
                    Some(ref s) => s,
                    None => return s.display().to_string(),
                };
                // FIXME: the cc-rs crate only recognizes the literal strings
                // `ccache` and `sccache` when doing caching compilations, so we
                // mirror that here. It should probably be fixed upstream to
                // accept a new env var or otherwise work with custom ccache
                // vars.
                match &ccache[..] {
                    "ccache" | "sccache" => format!("{} {}", ccache, s.display()),
                    _ => s.display().to_string(),
                }
            };
            let cc = ccacheify(&self.cc(target));
1122
            cargo.env(format!("CC_{}", target), &cc);
O
Oliver Schneider 已提交
1123

1124
            let cflags = self.cflags(target, GitRepo::Rustc).join(" ");
S
Santiago Pastorino 已提交
1125
            cargo
1126
                .env(format!("CFLAGS_{}", target), cflags.clone());
O
Oliver Schneider 已提交
1127 1128 1129

            if let Some(ar) = self.ar(target) {
                let ranlib = format!("{} s", ar.display());
S
Santiago Pastorino 已提交
1130 1131
                cargo
                    .env(format!("AR_{}", target), ar)
1132
                    .env(format!("RANLIB_{}", target), ranlib);
O
Oliver Schneider 已提交
1133
            }
1134

M
Mark Simulacrum 已提交
1135
            if let Ok(cxx) = self.cxx(target) {
1136
                let cxx = ccacheify(&cxx);
S
Santiago Pastorino 已提交
1137 1138
                cargo
                    .env(format!("CXX_{}", target), &cxx)
1139
                    .env(format!("CXXFLAGS_{}", target), cflags);
1140 1141 1142
            }
        }

1143
        if (cmd == "build" || cmd == "rustc")
C
Collins Abitekaniza 已提交
1144
            && mode == Mode::Std
S
Santiago Pastorino 已提交
1145 1146
            && self.config.extended
            && compiler.is_final_stage(self)
1147
        {
1148 1149 1150
            cargo.env("RUSTC_SAVE_ANALYSIS", "api".to_string());
        }

O
Oliver Schneider 已提交
1151
        // For `cargo doc` invocations, make rustdoc print the Rust version into the docs
1152
        cargo.env("RUSTDOC_CRATE_VERSION", self.rust_version());
O
Oliver Schneider 已提交
1153

1154 1155 1156 1157 1158
        // Environment variables *required* throughout the build
        //
        // FIXME: should update code to not require this env var
        cargo.env("CFG_COMPILER_HOST_TRIPLE", target);

1159
        // Set this for all builds to make sure doc builds also get it.
1160
        cargo.env("CFG_RELEASE_CHANNEL", &self.config.channel);
1161

1162 1163 1164 1165 1166 1167 1168 1169 1170 1171 1172 1173 1174 1175 1176 1177 1178 1179
        // This one's a bit tricky. As of the time of this writing the compiler
        // links to the `winapi` crate on crates.io. This crate provides raw
        // bindings to Windows system functions, sort of like libc does for
        // Unix. This crate also, however, provides "import libraries" for the
        // MinGW targets. There's an import library per dll in the windows
        // distribution which is what's linked to. These custom import libraries
        // are used because the winapi crate can reference Windows functions not
        // present in the MinGW import libraries.
        //
        // For example MinGW may ship libdbghelp.a, but it may not have
        // references to all the functions in the dbghelp dll. Instead the
        // custom import library for dbghelp in the winapi crates has all this
        // information.
        //
        // Unfortunately for us though the import libraries are linked by
        // default via `-ldylib=winapi_foo`. That is, they're linked with the
        // `dylib` type with a `winapi_` prefix (so the winapi ones don't
        // conflict with the system MinGW ones). This consequently means that
I
Irina Popa 已提交
1180
        // the binaries we ship of things like rustc_codegen_llvm (aka the rustc_codegen_llvm
1181 1182 1183 1184 1185 1186 1187 1188 1189 1190 1191 1192
        // DLL) when linked against *again*, for example with procedural macros
        // or plugins, will trigger the propagation logic of `-ldylib`, passing
        // `-lwinapi_foo` to the linker again. This isn't actually available in
        // our distribution, however, so the link fails.
        //
        // To solve this problem we tell winapi to not use its bundled import
        // libraries. This means that it will link to the system MinGW import
        // libraries by default, and the `-ldylib=foo` directives will still get
        // passed to the final linker, but they'll look like `-lfoo` which can
        // be resolved because MinGW has the import library. The downside is we
        // don't get newer functions from Windows, but we don't use any of them
        // anyway.
C
Collins Abitekaniza 已提交
1193
        if !mode.is_tool() {
A
Alex Crichton 已提交
1194 1195
            cargo.env("WINAPI_NO_BUNDLED_LIBRARIES", "1");
        }
1196

C
comex 已提交
1197
        for _ in 1..self.verbosity {
1198 1199
            cargo.arg("-v");
        }
1200

1201 1202 1203 1204 1205 1206 1207 1208 1209
        match (mode, self.config.rust_codegen_units_std, self.config.rust_codegen_units) {
            (Mode::Std, Some(n), _) |
            (Mode::Test, Some(n), _) |
            (_, _, Some(n)) => {
                cargo.env("RUSTC_CODEGEN_UNITS", n.to_string());
            }
            _ => {
                // Don't set anything
            }
1210 1211
        }

O
Oliver Schneider 已提交
1212
        if self.config.rust_optimize {
1213 1214
            // FIXME: cargo bench/install do not accept `--release`
            if cmd != "bench" && cmd != "install" {
O
Oliver Schneider 已提交
1215 1216
                cargo.arg("--release");
            }
1217
        }
1218

M
Mark Simulacrum 已提交
1219
        if self.config.locked_deps {
1220 1221
            cargo.arg("--locked");
        }
M
Mark Simulacrum 已提交
1222
        if self.config.vendor || self.is_sudo {
1223 1224 1225
            cargo.arg("--frozen");
        }

M
Mark Simulacrum 已提交
1226
        self.ci_env.force_coloring_in_ci(&mut cargo);
1227 1228 1229 1230

        cargo
    }

V
varkor 已提交
1231
    /// Ensure that a given step is built, returning its output. This will
1232 1233
    /// cache the step, so it is safe (and good!) to call this as often as
    /// needed to ensure that all dependencies are built.
1234
    pub fn ensure<S: Step>(&'a self, step: S) -> S::Output {
1235 1236
        {
            let mut stack = self.stack.borrow_mut();
1237 1238
            for stack_step in stack.iter() {
                // should skip
S
Santiago Pastorino 已提交
1239 1240 1241 1242
                if stack_step
                    .downcast_ref::<S>()
                    .map_or(true, |stack_step| *stack_step != step)
                {
1243
                    continue;
1244
                }
1245
                let mut out = String::new();
1246
                out += &format!("\n\nCycle in build detected when adding {:?}\n", step);
1247 1248 1249 1250 1251
                for el in stack.iter().rev() {
                    out += &format!("\t{:?}\n", el);
                }
                panic!(out);
            }
1252
            if let Some(out) = self.cache.get(&step) {
1253
                self.verbose(&format!("{}c {:?}", "  ".repeat(stack.len()), step));
1254

1255 1256 1257
                {
                    let mut graph = self.graph.borrow_mut();
                    let parent = self.parent.get();
S
Santiago Pastorino 已提交
1258 1259 1260
                    let us = *self
                        .graph_nodes
                        .borrow_mut()
1261 1262 1263 1264 1265 1266 1267
                        .entry(format!("{:?}", step))
                        .or_insert_with(|| graph.add_node(format!("{:?}", step)));
                    if let Some(parent) = parent {
                        graph.add_edge(parent, us, false);
                    }
                }

1268 1269
                return out;
            }
1270
            self.verbose(&format!("{}> {:?}", "  ".repeat(stack.len()), step));
1271
            stack.push(Box::new(step.clone()));
1272
        }
1273

1274 1275 1276 1277 1278
        let prev_parent = self.parent.get();

        {
            let mut graph = self.graph.borrow_mut();
            let parent = self.parent.get();
S
Santiago Pastorino 已提交
1279 1280 1281
            let us = *self
                .graph_nodes
                .borrow_mut()
1282 1283 1284 1285 1286 1287 1288 1289
                .entry(format!("{:?}", step))
                .or_insert_with(|| graph.add_node(format!("{:?}", step)));
            self.parent.set(Some(us));
            if let Some(parent) = parent {
                graph.add_edge(parent, us, true);
            }
        }

1290 1291 1292 1293 1294 1295 1296 1297 1298 1299
        let (out, dur) = {
            let start = Instant::now();
            let zero = Duration::new(0, 0);
            let parent = self.time_spent_on_dependencies.replace(zero);
            let out = step.clone().run(self);
            let dur = start.elapsed();
            let deps = self.time_spent_on_dependencies.replace(parent + dur);
            (out, dur - deps)
        };

1300 1301
        self.parent.set(prev_parent);

1302
        if self.config.print_step_timings && dur > Duration::from_millis(100) {
S
Santiago Pastorino 已提交
1303 1304 1305 1306 1307 1308
            println!(
                "[TIMING] {:?} -- {}.{:03}",
                step,
                dur.as_secs(),
                dur.subsec_nanos() / 1_000_000
            );
1309 1310
        }

1311 1312
        {
            let mut stack = self.stack.borrow_mut();
1313 1314
            let cur_step = stack.pop().expect("step stack empty");
            assert_eq!(cur_step.downcast_ref(), Some(&step));
1315
        }
S
Santiago Pastorino 已提交
1316 1317 1318 1319 1320
        self.verbose(&format!(
            "{}< {:?}",
            "  ".repeat(self.stack.borrow().len()),
            step
        ));
1321 1322
        self.cache.put(step, out.clone());
        out
1323 1324
    }
}
M
Mark Simulacrum 已提交
1325 1326

#[cfg(test)]
C
chansuke 已提交
1327
mod tests;