lib.rs 52.5 KB
Newer Older
1
//! Implementation of rustbuild, the Rust build system.
2
//!
3 4 5
//! This module, and its descendants, are the implementation of the Rust build
//! system. Most of this build system is backed by Cargo but the outer layer
//! here serves as the ability to orchestrate calling Cargo, sequencing Cargo
6
//! builds, building artifacts like LLVM, etc. The goals of rustbuild are:
7
//!
8 9 10 11 12 13 14 15
//! * To be an easily understandable, easily extensible, and maintainable build
//!   system.
//! * Leverage standard tools in the Rust ecosystem to build the compiler, aka
//!   crates.io and Cargo.
//! * A standard interface to build across all platforms, including MSVC
//!
//! ## Architecture
//!
16 17 18 19 20 21
//! The build system defers most of the complicated logic managing invocations
//! of rustc and rustdoc to Cargo itself. However, moving through various stages
//! and copying artifacts is still necessary for it to do. Each time rustbuild
//! is invoked, it will iterate through the list of predefined steps and execute
//! each serially in turn if it matches the paths passed or is a default rule.
//! For each step rustbuild relies on the step internally being incremental and
22 23 24 25 26 27
//! parallel. Note, though, that the `-j` parameter to rustbuild gets forwarded
//! to appropriate test harnesses and such.
//!
//! Most of the "meaty" steps that matter are backed by Cargo, which does indeed
//! have its own parallelism and incremental management. Later steps, like
//! tests, aren't incremental and simply run the entire suite currently.
28 29
//! However, compiletest itself tries to avoid running tests when the artifacts
//! that are involved (mainly the compiler) haven't changed.
30
//!
31
//! When you execute `x.py build`, the steps executed are:
32 33
//!
//! * First, the python script is run. This will automatically download the
34
//!   stage0 rustc and cargo according to `src/stage0.txt`, or use the cached
35 36 37 38
//!   versions if they're available. These are then used to compile rustbuild
//!   itself (using Cargo). Finally, control is then transferred to rustbuild.
//!
//! * Rustbuild takes over, performs sanity checks, probes the environment,
39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71
//!   reads configuration, and starts executing steps as it reads the command
//!   line arguments (paths) or going through the default rules.
//!
//!   The build output will be something like the following:
//!
//!   Building stage0 std artifacts
//!   Copying stage0 std
//!   Building stage0 test artifacts
//!   Copying stage0 test
//!   Building stage0 compiler artifacts
//!   Copying stage0 rustc
//!   Assembling stage1 compiler
//!   Building stage1 std artifacts
//!   Copying stage1 std
//!   Building stage1 test artifacts
//!   Copying stage1 test
//!   Building stage1 compiler artifacts
//!   Copying stage1 rustc
//!   Assembling stage2 compiler
//!   Uplifting stage1 std
//!   Uplifting stage1 test
//!   Uplifting stage1 rustc
//!
//! Let's disect that a little:
//!
//! ## Building stage0 {std,test,compiler} artifacts
//!
//! These steps use the provided (downloaded, usually) compiler to compile the
//! local Rust source into libraries we can use.
//!
//! ## Copying stage0 {std,test,rustc}
//!
//! This copies the build output from Cargo into
A
Alexander Regueiro 已提交
72
//! `build/$HOST/stage0-sysroot/lib/rustlib/$ARCH/lib`. FIXME: this step's
73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88
//! documentation should be expanded -- the information already here may be
//! incorrect.
//!
//! ## Assembling stage1 compiler
//!
//! This copies the libraries we built in "building stage0 ... artifacts" into
//! the stage1 compiler's lib directory. These are the host libraries that the
//! compiler itself uses to run. These aren't actually used by artifacts the new
//! compiler generates. This step also copies the rustc and rustdoc binaries we
//! generated into build/$HOST/stage/bin.
//!
//! The stage1/bin/rustc is a fully functional compiler, but it doesn't yet have
//! any libraries to link built binaries or libraries to. The next 3 steps will
//! provide those libraries for it; they are mostly equivalent to constructing
//! the stage1/bin compiler so we don't go through them individually.
//!
89
//! ## Uplifting stage1 {std,test,rustc}
90 91 92 93 94 95 96
//!
//! This step copies the libraries from the stage1 compiler sysroot into the
//! stage2 compiler. This is done to avoid rebuilding the compiler; libraries
//! we'd build in this step should be identical (in function, if not necessarily
//! identical on disk) so there's no need to recompile the compiler again. Note
//! that if you want to, you can enable the full-bootstrap option to change this
//! behavior.
97 98 99 100 101 102 103 104
//!
//! Each step is driven by a separate Cargo project and rustbuild orchestrates
//! copying files between steps and otherwise preparing for Cargo to run.
//!
//! ## Further information
//!
//! More documentation can be found in each respective module below, and you can
//! also check out the `src/bootstrap/README.md` file for more information.
105

M
Mark Rousskov 已提交
106 107
use std::cell::{Cell, RefCell};
use std::collections::{HashMap, HashSet};
A
Alex Crichton 已提交
108
use std::env;
M
Mark Rousskov 已提交
109 110 111
use std::fs::{self, File, OpenOptions};
use std::io::{Read, Seek, SeekFrom, Write};
use std::path::{Path, PathBuf};
112
use std::process::{self, Command};
M
Mark Simulacrum 已提交
113
use std::slice;
114
use std::str;
115

T
Tom Tromey 已提交
116 117 118 119 120
#[cfg(unix)]
use std::os::unix::fs::symlink as symlink_file;
#[cfg(windows)]
use std::os::windows::fs::symlink_file;

M
Mark Rousskov 已提交
121
use build_helper::{mtime, output, run, run_suppressed, t, try_run, try_run_suppressed};
122
use filetime::FileTime;
123

124
use crate::config::{LlvmLibunwind, TargetSelection};
125
use crate::util::{exe, libdir, CiEnv};
126

M
Mark Rousskov 已提交
127 128
mod builder;
mod cache;
A
Alex Crichton 已提交
129
mod cc_detect;
130 131 132 133 134 135 136 137
mod channel;
mod check;
mod clean;
mod compile;
mod config;
mod dist;
mod doc;
mod flags;
M
Mark Rousskov 已提交
138
mod format;
139
mod install;
M
Mark Rousskov 已提交
140
mod metadata;
141
mod native;
142
mod run;
143
mod sanity;
J
Joshua Nelson 已提交
144
mod setup;
145
mod tarball;
M
Mark Rousskov 已提交
146
mod test;
147
mod tool;
148
mod toolstate;
M
Mark Rousskov 已提交
149
pub mod util;
150 151 152 153

#[cfg(windows)]
mod job;

154
#[cfg(all(unix, not(target_os = "haiku")))]
155
mod job {
L
ljedrz 已提交
156
    pub unsafe fn setup(build: &mut crate::Build) {
157
        if build.config.low_priority {
Q
QuietMisdreavus 已提交
158
            libc::setpriority(libc::PRIO_PGRP as _, 0, 10);
159 160 161 162
        }
    }
}

163
#[cfg(any(target_os = "haiku", target_os = "hermit", not(any(unix, windows))))]
164
mod job {
M
Mark Rousskov 已提交
165
    pub unsafe fn setup(_build: &mut crate::Build) {}
166 167
}

M
Mark Rousskov 已提交
168
use crate::cache::{Interned, INTERNER};
L
ljedrz 已提交
169
pub use crate::config::Config;
J
Joshua Nelson 已提交
170
pub use crate::flags::Subcommand;
171

J
Jorge Aparicio 已提交
172
const LLVM_TOOLS: &[&str] = &[
D
dalance 已提交
173 174 175 176
    "llvm-cov",      // used to generate coverage report
    "llvm-nm",       // used to inspect binaries; it shows symbol names, their sizes and visibility
    "llvm-objcopy",  // used to transform ELFs into binary format which flashing tools consume
    "llvm-objdump",  // used to disassemble programs
J
Jorge Aparicio 已提交
177
    "llvm-profdata", // used to inspect and merge files generated by profiles
D
dalance 已提交
178 179 180 181
    "llvm-readobj",  // used to get information from ELFs/objects that the other tools don't provide
    "llvm-size",     // used to prints the size of the linker sections of a program
    "llvm-strip",    // used to discard symbols from binary files to reduce their size
    "llvm-ar",       // used for creating and modifying archive files
Z
Zachary Catlin 已提交
182
    "llvm-as",       // used to convert LLVM assembly to LLVM bitcode
D
dalance 已提交
183 184 185
    "llvm-dis",      // used to disassemble LLVM bitcode
    "llc",           // used to compile LLVM bytecode
    "opt",           // used to optimize LLVM bytecode
J
Jorge Aparicio 已提交
186
];
187

188 189
pub const VERSION: usize = 2;

190 191 192 193 194
/// A structure representing a Rust compiler.
///
/// Each compiler has a `stage` that it is associated with and a `host` that
/// corresponds to the platform the compiler runs on. This structure is used as
/// a parameter to many methods below.
M
Mark Simulacrum 已提交
195
#[derive(Eq, PartialOrd, Ord, PartialEq, Clone, Copy, Hash, Debug)]
196
pub struct Compiler {
197
    stage: u32,
198
    host: TargetSelection,
199 200
}

K
kennytm 已提交
201
#[derive(PartialEq, Eq, Copy, Clone, Debug)]
K
kennytm 已提交
202
pub enum DocTests {
A
Alexander Regueiro 已提交
203
    /// Run normal tests and doc tests (default).
K
kennytm 已提交
204
    Yes,
A
Alexander Regueiro 已提交
205
    /// Do not run any doc tests.
K
kennytm 已提交
206
    No,
A
Alexander Regueiro 已提交
207
    /// Only run doc tests.
K
kennytm 已提交
208 209 210
    Only,
}

211 212 213 214 215
pub enum GitRepo {
    Rustc,
    Llvm,
}

216 217 218 219 220 221 222 223 224 225 226
/// Global configuration for the build system.
///
/// This structure transitively contains all configuration for the build system.
/// All filesystem-encoded configuration is in `config`, all flags are in
/// `flags`, and then parsed or probed information is listed in the keys below.
///
/// This structure is a parameter of almost all methods in the build system,
/// although most functions are implemented as free functions rather than
/// methods specifically on this structure itself (to make it easier to
/// organize).
pub struct Build {
A
Alexander Regueiro 已提交
227
    /// User-specified configuration from `config.toml`.
228 229
    config: Config,

230 231 232
    // Version information
    version: String,

A
Alexander Regueiro 已提交
233
    // Properties derived from the above configuration
234 235
    src: PathBuf,
    out: PathBuf,
236 237
    rust_info: channel::GitInfo,
    cargo_info: channel::GitInfo,
A
Alex Crichton 已提交
238
    rls_info: channel::GitInfo,
A
Aleksey Kladov 已提交
239
    rust_analyzer_info: channel::GitInfo,
240
    clippy_info: channel::GitInfo,
241
    miri_info: channel::GitInfo,
N
Nick Cameron 已提交
242
    rustfmt_info: channel::GitInfo,
243
    in_tree_llvm_info: channel::GitInfo,
244
    local_rebuild: bool,
245
    fail_fast: bool,
K
kennytm 已提交
246
    doc_tests: DocTests,
M
Mark Simulacrum 已提交
247
    verbosity: usize,
248

A
Alexander Regueiro 已提交
249
    // Targets for which to build
250 251 252
    build: TargetSelection,
    hosts: Vec<TargetSelection>,
    targets: Vec<TargetSelection>,
253

254
    // Stage 0 (downloaded) compiler, lld and cargo or their local rust equivalents
255 256
    initial_rustc: PathBuf,
    initial_cargo: PathBuf,
257
    initial_lld: PathBuf,
O
O01eg 已提交
258
    initial_libdir: PathBuf,
259

260
    // Runtime state filled in later on
261
    // C/C++ compilers and archiver for all targets
262 263 264 265
    cc: HashMap<TargetSelection, cc::Tool>,
    cxx: HashMap<TargetSelection, cc::Tool>,
    ar: HashMap<TargetSelection, PathBuf>,
    ranlib: HashMap<TargetSelection, PathBuf>,
A
Alexander Regueiro 已提交
266
    // Miscellaneous
267
    crates: HashMap<Interned<String>, Crate>,
268
    is_sudo: bool,
269
    ci_env: CiEnv,
270
    delayed_failures: RefCell<Vec<String>>,
271
    prerelease_version: Cell<Option<u32>>,
M
Mark Rousskov 已提交
272
    tool_artifacts:
273
        RefCell<HashMap<TargetSelection, HashMap<String, (&'static str, PathBuf, Vec<String>)>>>,
274 275 276 277
}

#[derive(Debug)]
struct Crate {
278
    name: Interned<String>,
279 280
    deps: HashSet<Interned<String>>,
    id: String,
281
    path: PathBuf,
282 283
}

284 285 286 287 288 289
impl Crate {
    fn local_path(&self, build: &Build) -> PathBuf {
        self.path.strip_prefix(&build.config.src).unwrap().into()
    }
}

290 291 292 293 294 295 296 297 298 299 300
/// When building Rust various objects are handled differently.
#[derive(Debug, Clone, Copy, PartialEq, Eq, PartialOrd, Ord)]
pub enum DependencyType {
    /// Libraries originating from proc-macros.
    Host,
    /// Typical Rust libraries.
    Target,
    /// Non Rust libraries and objects shipped to ease usage of certain targets.
    TargetSelfContained,
}

301 302 303 304
/// The various "modes" of invoking Cargo.
///
/// These entries currently correspond to the various output directories of the
/// build system, with each mod generating output in a different directory.
K
kennytm 已提交
305
#[derive(Debug, Hash, Clone, Copy, PartialEq, Eq, PartialOrd, Ord)]
306
pub enum Mode {
M
Mark Simulacrum 已提交
307
    /// Build the standard library, placing output in the "stageN-std" directory.
C
Collins Abitekaniza 已提交
308
    Std,
309

310
    /// Build librustc, and compiler libraries, placing output in the "stageN-rustc" directory.
C
Collins Abitekaniza 已提交
311
    Rustc,
312

313 314 315
    /// Build a codegen backend for rustc, placing the output in the "stageN-codegen" directory.
    Codegen,

316 317 318 319
    /// Build a tool, placing output in the "stage0-bootstrap-tools"
    /// directory. This is for miscellaneous sets of tools that are built
    /// using the bootstrap stage0 compiler in its entirety (target libraries
    /// and all). Typically these tools compile with stable Rust.
320 321
    ToolBootstrap,

322 323 324
    /// Build a tool which uses the locally built std, placing output in the
    /// "stageN-tools" directory. Its usage is quite rare, mainly used by
    /// compiletest which needs libtest.
A
Alex Crichton 已提交
325
    ToolStd,
326 327 328 329 330

    /// Build a tool which uses the locally built rustc and the target std,
    /// placing the output in the "stageN-tools" directory. This is used for
    /// anything that needs a fully functional rustc, such as rustdoc, clippy,
    /// cargo, rls, rustfmt, miri, etc.
C
Collins Abitekaniza 已提交
331
    ToolRustc,
332 333
}

C
Collins Abitekaniza 已提交
334 335
impl Mode {
    pub fn is_tool(&self) -> bool {
336
        matches!(self, Mode::ToolBootstrap | Mode::ToolRustc | Mode::ToolStd)
C
Collins Abitekaniza 已提交
337
    }
338 339 340 341

    pub fn must_support_dlopen(&self) -> bool {
        matches!(self, Mode::Std | Mode::Codegen)
    }
C
Collins Abitekaniza 已提交
342 343
}

344 345 346 347 348
impl Build {
    /// Creates a new set of build configuration from the `flags` on the command
    /// line and the filesystem `config`.
    ///
    /// By default all build output will be placed in the current directory.
M
Mark Simulacrum 已提交
349 350
    pub fn new(config: Config) -> Build {
        let src = config.src.clone();
351
        let out = config.out.clone();
352

353
        let is_sudo = match env::var_os("SUDO_USER") {
M
Mark Rousskov 已提交
354 355 356 357
            Some(sudo_user) => match env::var_os("USER") {
                Some(user) => user != sudo_user,
                None => false,
            },
358 359
            None => false,
        };
J
Josh Stone 已提交
360 361 362 363 364

        let ignore_git = config.ignore_git;
        let rust_info = channel::GitInfo::new(ignore_git, &src);
        let cargo_info = channel::GitInfo::new(ignore_git, &src.join("src/tools/cargo"));
        let rls_info = channel::GitInfo::new(ignore_git, &src.join("src/tools/rls"));
A
Aleksey Kladov 已提交
365 366
        let rust_analyzer_info =
            channel::GitInfo::new(ignore_git, &src.join("src/tools/rust-analyzer"));
J
Josh Stone 已提交
367 368 369 370 371 372
        let clippy_info = channel::GitInfo::new(ignore_git, &src.join("src/tools/clippy"));
        let miri_info = channel::GitInfo::new(ignore_git, &src.join("src/tools/miri"));
        let rustfmt_info = channel::GitInfo::new(ignore_git, &src.join("src/tools/rustfmt"));

        // we always try to use git for LLVM builds
        let in_tree_llvm_info = channel::GitInfo::new(false, &src.join("src/llvm-project"));
373

O
O01eg 已提交
374 375
        let initial_target_libdir_str = if config.dry_run {
            "/dummy/lib/path/to/lib/".to_string()
376 377 378 379
        } else {
            output(
                Command::new(&config.initial_rustc)
                    .arg("--target")
380
                    .arg(config.build.rustc_target_arg())
381 382 383 384
                    .arg("--print")
                    .arg("target-libdir"),
            )
        };
O
O01eg 已提交
385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400
        let initial_target_dir = Path::new(&initial_target_libdir_str).parent().unwrap();
        let initial_lld = initial_target_dir.join("bin").join("rust-lld");

        let initial_sysroot = if config.dry_run {
            "/dummy".to_string()
        } else {
            output(Command::new(&config.initial_rustc).arg("--print").arg("sysroot"))
        };
        let initial_libdir = initial_target_dir
            .parent()
            .unwrap()
            .parent()
            .unwrap()
            .strip_prefix(initial_sysroot.trim())
            .unwrap()
            .to_path_buf();
401

402 403 404 405
        let version = std::fs::read_to_string(src.join("src").join("version"))
            .expect("failed to read src/version");
        let version = version.trim();

406
        let mut build = Build {
407 408
            initial_rustc: config.initial_rustc.clone(),
            initial_cargo: config.initial_cargo.clone(),
409
            initial_lld,
O
O01eg 已提交
410
            initial_libdir,
411
            local_rebuild: config.local_rebuild,
M
Mark Simulacrum 已提交
412
            fail_fast: config.cmd.fail_fast(),
413
            doc_tests: config.cmd.doc_tests(),
M
Mark Simulacrum 已提交
414
            verbosity: config.verbose,
415

M
Mark Simulacrum 已提交
416 417 418
            build: config.build,
            hosts: config.hosts.clone(),
            targets: config.targets.clone(),
419

420
            config,
421
            version: version.to_string(),
422 423
            src,
            out,
424

425 426 427
            rust_info,
            cargo_info,
            rls_info,
A
Aleksey Kladov 已提交
428
            rust_analyzer_info,
429
            clippy_info,
430
            miri_info,
N
Nick Cameron 已提交
431
            rustfmt_info,
432
            in_tree_llvm_info,
433 434
            cc: HashMap::new(),
            cxx: HashMap::new(),
435
            ar: HashMap::new(),
436
            ranlib: HashMap::new(),
437
            crates: HashMap::new(),
438
            is_sudo,
439
            ci_env: CiEnv::current(),
440
            delayed_failures: RefCell::new(Vec::new()),
441
            prerelease_version: Cell::new(None),
442
            tool_artifacts: Default::default(),
443 444 445 446
        };

        build.verbose("finding compilers");
        cc_detect::find(&mut build);
447 448 449 450 451 452 453
        // When running `setup`, the profile is about to change, so any requirements we have now may
        // be different on the next invocation. Don't check for them until the next time x.py is
        // run. This is ok because `setup` never runs any build commands, so it won't fail if commands are missing.
        if !matches!(build.config.cmd, Subcommand::Setup { .. }) {
            build.verbose("running sanity check");
            sanity::check(&mut build);
        }
M
Mark Simulacrum 已提交
454 455 456

        // If local-rust is the same major.minor as the current version, then force a
        // local-rebuild
M
Mark Rousskov 已提交
457 458
        let local_version_verbose =
            output(Command::new(&build.initial_rustc).arg("--version").arg("--verbose"));
M
Mark Simulacrum 已提交
459
        let local_release = local_version_verbose
M
Mark Rousskov 已提交
460
            .lines()
L
Lzu Tao 已提交
461
            .filter_map(|x| x.strip_prefix("release:"))
M
Mark Rousskov 已提交
462 463 464
            .next()
            .unwrap()
            .trim();
465
        if local_release.split('.').take(2).eq(version.split('.').take(2)) {
M
Mark Simulacrum 已提交
466 467
            build.verbose(&format!("auto-detected local-rebuild {}", local_release));
            build.local_rebuild = true;
468
        }
M
Mark Simulacrum 已提交
469

470 471 472 473
        build.verbose("learning about cargo");
        metadata::build(&mut build);

        build
474 475
    }

M
Mark Simulacrum 已提交
476
    pub fn build_triple(&self) -> &[Interned<String>] {
477
        slice::from_ref(&self.build.triple)
M
Mark Simulacrum 已提交
478 479
    }

480 481 482
    /// Executes the entire build, as configured by the flags and configuration.
    pub fn build(&mut self) {
        unsafe {
483
            job::setup(self);
484 485
        }

J
Joshua Nelson 已提交
486 487 488 489 490
        // If the LLVM submodule has been initialized already, sync it unconditionally. This avoids
        // contributors checking in a submodule change by accident.
        if self.in_tree_llvm_info.is_git() {
            native::update_llvm_submodule(self);
        }
491

492 493
        if let Subcommand::Format { check, paths } = &self.config.cmd {
            return format::format(self, *check, &paths);
A
Adam Perry 已提交
494 495
        }

T
Tommy Ip 已提交
496 497
        if let Subcommand::Clean { all } = self.config.cmd {
            return clean::clean(self, all);
498 499
        }

A
Antoine Martin 已提交
500 501
        if let Subcommand::Setup { profile } = &self.config.cmd {
            return setup::setup(&self.config.src, *profile);
J
Joshua Nelson 已提交
502 503
        }

504 505 506 507 508 509
        {
            let builder = builder::Builder::new(&self);
            if let Some(path) = builder.paths.get(0) {
                if path == Path::new("nonexistent/path/to/trigger/cargo/metadata") {
                    return;
                }
510 511
            }
        }
512 513

        if !self.config.dry_run {
514
            {
515 516
                self.config.dry_run = true;
                let builder = builder::Builder::new(&self);
517 518
                builder.execute_cli();
            }
519 520
            self.config.dry_run = false;
            let builder = builder::Builder::new(&self);
521
            builder.execute_cli();
522 523
        } else {
            let builder = builder::Builder::new(&self);
524
            builder.execute_cli();
525
        }
526 527 528 529 530 531 532 533 534 535

        // Check for postponed failures from `test --no-fail-fast`.
        let failures = self.delayed_failures.borrow();
        if failures.len() > 0 {
            println!("\n{} command(s) did not execute successfully:\n", failures.len());
            for failure in failures.iter() {
                println!("  - {}\n", failure);
            }
            process::exit(1);
        }
536 537 538 539 540
    }

    /// Clear out `dir` if `input` is newer.
    ///
    /// After this executes, it will also ensure that `dir` exists.
541
    fn clear_if_dirty(&self, dir: &Path, input: &Path) -> bool {
542
        let stamp = dir.join(".stamp");
543
        let mut cleared = false;
544 545 546
        if mtime(&stamp) < mtime(input) {
            self.verbose(&format!("Dirty - {}", dir.display()));
            let _ = fs::remove_dir_all(dir);
547
            cleared = true;
548
        } else if stamp.exists() {
549
            return cleared;
550 551 552
        }
        t!(fs::create_dir_all(dir));
        t!(File::create(stamp));
553
        cleared
554 555
    }

A
Alexander Regueiro 已提交
556
    /// Gets the space-separated set of activated features for the standard
557
    /// library.
558
    fn std_features(&self, target: TargetSelection) -> String {
559
        let mut features = "panic-unwind".to_string();
J
Jorge Aparicio 已提交
560

561
        match self.config.llvm_libunwind {
562 563 564
            LlvmLibunwind::InTree => features.push_str(" llvm-libunwind"),
            LlvmLibunwind::System => features.push_str(" system-llvm-libunwind"),
            LlvmLibunwind::No => {}
565
        }
566 567 568
        if self.config.backtrace {
            features.push_str(" backtrace");
        }
569
        if self.config.profiler_enabled(target) {
570 571
            features.push_str(" profiler");
        }
M
Mark Simulacrum 已提交
572
        features
573 574
    }

A
Alexander Regueiro 已提交
575
    /// Gets the space-separated set of activated features for the compiler.
576
    fn rustc_features(&self) -> String {
577 578 579 580
        let mut features = String::new();
        if self.config.jemalloc {
            features.push_str("jemalloc");
        }
581 582 583
        if self.config.llvm_enabled() {
            features.push_str(" llvm");
        }
G
Gus Wynn 已提交
584 585 586 587 588 589 590

        // If debug logging is on, then we want the default for tracing:
        // https://github.com/tokio-rs/tracing/blob/3dd5c03d907afdf2c39444a29931833335171554/tracing/src/level_filters.rs#L26
        // which is everything (including debug/trace/etc.)
        // if its unset, if debug_assertions is on, then debug_logging will also be on
        // as well as tracing *ignoring* this feature when debug_assertions is on
        if !self.config.rust_debug_logging {
G
Gus Wynn 已提交
591
            features.push_str(" max_level_info");
G
Gus Wynn 已提交
592 593
        }

594
        features
595 596 597 598 599
    }

    /// Component directory that Cargo will produce output into (e.g.
    /// release/debug)
    fn cargo_dir(&self) -> &'static str {
M
Mark Rousskov 已提交
600
        if self.config.rust_optimize { "release" } else { "debug" }
601 602
    }

603
    fn tools_dir(&self, compiler: Compiler) -> PathBuf {
604 605 606 607
        let out = self
            .out
            .join(&*compiler.host.triple)
            .join(format!("stage{}-tools-bin", compiler.stage));
608 609 610 611
        t!(fs::create_dir_all(&out));
        out
    }

612 613 614 615
    /// Returns the root directory for all output generated in a particular
    /// stage when running with a particular host compiler.
    ///
    /// The mode indicates what the root directory is for.
616
    fn stage_out(&self, compiler: Compiler, mode: Mode) -> PathBuf {
617
        let suffix = match mode {
C
Collins Abitekaniza 已提交
618 619
            Mode::Std => "-std",
            Mode::Rustc => "-rustc",
620
            Mode::Codegen => "-codegen",
621
            Mode::ToolBootstrap => "-bootstrap-tools",
622
            Mode::ToolStd | Mode::ToolRustc => "-tools",
623
        };
624
        self.out.join(&*compiler.host.triple).join(format!("stage{}{}", compiler.stage, suffix))
625 626 627
    }

    /// Returns the root output directory for all Cargo output in a given stage,
F
Fourchaux 已提交
628
    /// running a particular compiler, whether or not we're building the
629
    /// standard library, and targeting the specified architecture.
630 631
    fn cargo_out(&self, compiler: Compiler, mode: Mode, target: TargetSelection) -> PathBuf {
        self.stage_out(compiler, mode).join(&*target.triple).join(self.cargo_dir())
632 633 634 635 636 637
    }

    /// Root output directory for LLVM compiled for `target`
    ///
    /// Note that if LLVM is configured externally then the directory returned
    /// will likely be empty.
638 639
    fn llvm_out(&self, target: TargetSelection) -> PathBuf {
        self.out.join(&*target.triple).join("llvm")
640 641
    }

642 643
    fn lld_out(&self, target: TargetSelection) -> PathBuf {
        self.out.join(&*target.triple).join("lld")
644 645
    }

646
    /// Output directory for all documentation for a target
647 648
    fn doc_out(&self, target: TargetSelection) -> PathBuf {
        self.out.join(&*target.triple).join("doc")
649 650
    }

G
Guillaume Gomez 已提交
651 652 653 654
    fn test_out(&self, target: TargetSelection) -> PathBuf {
        self.out.join(&*target.triple).join("test")
    }

655
    /// Output directory for all documentation for a target
656 657
    fn compiler_doc_out(&self, target: TargetSelection) -> PathBuf {
        self.out.join(&*target.triple).join("compiler-doc")
658 659
    }

660
    /// Output directory for some generated md crate documentation for a target (temporary)
661 662
    fn md_doc_out(&self, target: TargetSelection) -> Interned<PathBuf> {
        INTERNER.intern_path(self.out.join(&*target.triple).join("md-doc"))
663 664
    }

A
Alexander Regueiro 已提交
665
    /// Returns `true` if no custom `llvm-config` is set for the specified target.
666 667
    ///
    /// If no custom `llvm-config` was specified then Rust's llvm will be used.
668
    fn is_rust_llvm(&self, target: TargetSelection) -> bool {
669 670 671 672
        if self.config.llvm_from_ci && target == self.config.build {
            return true;
        }

673
        match self.config.target_config.get(&target) {
674
            Some(ref c) => c.llvm_config.is_none(),
M
Mark Rousskov 已提交
675
            None => true,
676 677 678
        }
    }

679
    /// Returns the path to `FileCheck` binary for the specified target
680
    fn llvm_filecheck(&self, target: TargetSelection) -> PathBuf {
681
        let target_config = self.config.target_config.get(&target);
682 683 684
        if let Some(s) = target_config.and_then(|c| c.llvm_filecheck.as_ref()) {
            s.to_path_buf()
        } else if let Some(s) = target_config.and_then(|c| c.llvm_config.as_ref()) {
S
Seo Sanghyeon 已提交
685
            let llvm_bindir = output(Command::new(s).arg("--bindir"));
686
            let filecheck = Path::new(llvm_bindir.trim()).join(exe("FileCheck", target));
687 688 689 690 691 692
            if filecheck.exists() {
                filecheck
            } else {
                // On Fedora the system LLVM installs FileCheck in the
                // llvm subdirectory of the libdir.
                let llvm_libdir = output(Command::new(s).arg("--libdir"));
M
Mark Rousskov 已提交
693
                let lib_filecheck =
694
                    Path::new(llvm_libdir.trim()).join("llvm").join(exe("FileCheck", target));
695 696 697 698 699 700 701 702 703
                if lib_filecheck.exists() {
                    lib_filecheck
                } else {
                    // Return the most normal file name, even though
                    // it doesn't exist, so that any error message
                    // refers to that.
                    filecheck
                }
            }
704
        } else {
705
            let base = self.llvm_out(self.config.build).join("build");
706
            let base = if !self.ninja() && self.config.build.contains("msvc") {
J
John Kåre Alsaker 已提交
707 708 709 710 711 712 713 714 715
                if self.config.llvm_optimize {
                    if self.config.llvm_release_debuginfo {
                        base.join("RelWithDebInfo")
                    } else {
                        base.join("Release")
                    }
                } else {
                    base.join("Debug")
                }
716
            } else {
J
John Kåre Alsaker 已提交
717 718
                base
            };
719
            base.join("bin").join(exe("FileCheck", target))
720 721 722
        }
    }

723
    /// Directory for libraries built from C/C++ code and shared between stages.
724 725
    fn native_dir(&self, target: TargetSelection) -> PathBuf {
        self.out.join(&*target.triple).join("native")
726 727
    }

728 729
    /// Root output directory for rust_test_helpers library compiled for
    /// `target`
730
    fn test_helpers_out(&self, target: TargetSelection) -> PathBuf {
731
        self.native_dir(target).join("rust-test-helpers")
732 733
    }

734 735 736 737 738 739 740
    /// Adds the `RUST_TEST_THREADS` env var if necessary
    fn add_rust_test_threads(&self, cmd: &mut Command) {
        if env::var_os("RUST_TEST_THREADS").is_none() {
            cmd.env("RUST_TEST_THREADS", self.jobs().to_string());
        }
    }

741 742
    /// Returns the libdir of the snapshot compiler.
    fn rustc_snapshot_libdir(&self) -> PathBuf {
743
        self.rustc_snapshot_sysroot().join(libdir(self.config.build))
744 745 746 747
    }

    /// Returns the sysroot of the snapshot compiler.
    fn rustc_snapshot_sysroot(&self) -> &Path {
748
        self.initial_rustc.parent().unwrap().parent().unwrap()
749 750 751 752
    }

    /// Runs a command, printing out nice contextual information if it fails.
    fn run(&self, cmd: &mut Command) {
M
Mark Rousskov 已提交
753 754 755
        if self.config.dry_run {
            return;
        }
756
        self.verbose(&format!("running: {:?}", cmd));
757
        run(cmd)
758 759
    }

760 761
    /// Runs a command, printing out nice contextual information if it fails.
    fn run_quiet(&self, cmd: &mut Command) {
M
Mark Rousskov 已提交
762 763 764
        if self.config.dry_run {
            return;
        }
765
        self.verbose(&format!("running: {:?}", cmd));
766
        run_suppressed(cmd)
767 768
    }

769 770 771 772
    /// Runs a command, printing out nice contextual information if it fails.
    /// Exits if the command failed to execute at all, otherwise returns its
    /// `status.success()`.
    fn try_run(&self, cmd: &mut Command) -> bool {
M
Mark Rousskov 已提交
773 774 775
        if self.config.dry_run {
            return true;
        }
776
        self.verbose(&format!("running: {:?}", cmd));
777
        try_run(cmd)
778 779 780 781 782 783
    }

    /// Runs a command, printing out nice contextual information if it fails.
    /// Exits if the command failed to execute at all, otherwise returns its
    /// `status.success()`.
    fn try_run_quiet(&self, cmd: &mut Command) -> bool {
M
Mark Rousskov 已提交
784 785 786
        if self.config.dry_run {
            return true;
        }
787
        self.verbose(&format!("running: {:?}", cmd));
788
        try_run_suppressed(cmd)
789 790
    }

M
Mark Simulacrum 已提交
791 792 793 794
    pub fn is_verbose(&self) -> bool {
        self.verbosity > 0
    }

795 796
    /// Prints a message if this build is configured in verbose mode.
    fn verbose(&self, msg: &str) {
M
Mark Simulacrum 已提交
797
        if self.is_verbose() {
798 799 800 801
            println!("{}", msg);
        }
    }

802 803 804 805 806 807 808 809 810 811 812
    pub fn is_verbose_than(&self, level: usize) -> bool {
        self.verbosity > level
    }

    /// Prints a message if this build is configured in more verbose mode than `level`.
    fn verbose_than(&self, level: usize, msg: &str) {
        if self.is_verbose_than(level) {
            println!("{}", msg);
        }
    }

813
    fn info(&self, msg: &str) {
M
Mark Rousskov 已提交
814 815 816
        if self.config.dry_run {
            return;
        }
817 818 819
        println!("{}", msg);
    }

820 821 822
    /// Returns the number of parallel jobs that have been configured for this
    /// build.
    fn jobs(&self) -> u32 {
M
Mark Simulacrum 已提交
823
        self.config.jobs.unwrap_or_else(|| num_cpus::get() as u32)
824 825
    }

826
    fn debuginfo_map_to(&self, which: GitRepo) -> Option<String> {
827
        if !self.config.rust_remap_debuginfo {
M
Mark Rousskov 已提交
828
            return None;
829 830
        }

831
        match which {
832
            GitRepo::Rustc => {
833
                let sha = self.rust_sha().unwrap_or(&self.version);
834
                Some(format!("/rustc/{}", sha))
835
            }
836 837
            GitRepo::Llvm => Some(String::from("/rustc/llvm")),
        }
838 839
    }

840
    /// Returns the path to the C compiler for the target specified.
841
    fn cc(&self, target: TargetSelection) -> &Path {
842
        self.cc[&target].path()
843 844 845 846
    }

    /// Returns a list of flags to pass to the C compiler for the target
    /// specified.
847
    fn cflags(&self, target: TargetSelection, which: GitRepo) -> Vec<String> {
848
        // Filter out -O and /O (the optimization flags) that we picked up from
A
Alex Crichton 已提交
849
        // cc-rs because the build scripts will determine that for themselves.
M
Mark Rousskov 已提交
850 851 852 853 854 855
        let mut base = self.cc[&target]
            .args()
            .iter()
            .map(|s| s.to_string_lossy().into_owned())
            .filter(|s| !s.starts_with("-O") && !s.starts_with("/O"))
            .collect::<Vec<String>>();
856

857
        // If we're compiling on macOS then we add a few unconditional flags
858 859
        // indicating that we want libc++ (more filled out than libstdc++) and
        // we want to compile for 10.7. This way we can ensure that
860
        // LLVM/etc are all properly compiled.
861 862 863
        if target.contains("apple-darwin") {
            base.push("-stdlib=libc++".into());
        }
864 865

        // Work around an apparently bad MinGW / GCC optimization,
S
Smitty 已提交
866
        // See: https://lists.llvm.org/pipermail/cfe-dev/2016-December/051980.html
867
        // See: https://gcc.gnu.org/bugzilla/show_bug.cgi?id=78936
868
        if &*target.triple == "i686-pc-windows-gnu" {
869 870
            base.push("-fno-omit-frame-pointer".into());
        }
871

872 873
        if let Some(map_to) = self.debuginfo_map_to(which) {
            let map = format!("{}={}", self.src.display(), map_to);
M
Mark Rousskov 已提交
874
            let cc = self.cc(target);
875
            if cc.ends_with("clang") || cc.ends_with("gcc") {
876
                base.push(format!("-fdebug-prefix-map={}", map));
877 878
            } else if cc.ends_with("clang-cl.exe") {
                base.push("-Xclang".into());
879
                base.push(format!("-fdebug-prefix-map={}", map));
880 881
            }
        }
M
Mark Simulacrum 已提交
882
        base
883 884 885
    }

    /// Returns the path to the `ar` archive utility for the target specified.
886
    fn ar(&self, target: TargetSelection) -> Option<&Path> {
887 888 889
        self.ar.get(&target).map(|p| &**p)
    }

890
    /// Returns the path to the `ranlib` utility for the target specified.
891
    fn ranlib(&self, target: TargetSelection) -> Option<&Path> {
892 893 894
        self.ranlib.get(&target).map(|p| &**p)
    }

895
    /// Returns the path to the C++ compiler for the target specified.
896
    fn cxx(&self, target: TargetSelection) -> Result<&Path, String> {
897
        match self.cxx.get(&target) {
898
            Some(p) => Ok(p.path()),
M
Mark Rousskov 已提交
899 900 901
            None => {
                Err(format!("target `{}` is not configured as a host, only as a target", target))
            }
902
        }
903 904
    }

905
    /// Returns the path to the linker for the given target if it needs to be overridden.
906
    fn linker(&self, target: TargetSelection) -> Option<&Path> {
M
Mark Rousskov 已提交
907 908
        if let Some(linker) = self.config.target_config.get(&target).and_then(|c| c.linker.as_ref())
        {
909
            Some(linker)
P
Pang, Baoshan 已提交
910 911 912 913
        } else if target.contains("vxworks") {
            // need to use CXX compiler as linker to resolve the exception functions
            // that are only existed in CXX libraries
            Some(self.cxx[&target].path())
M
Mark Rousskov 已提交
914
        } else if target != self.config.build
915
            && util::use_host_linker(target)
M
Mark Rousskov 已提交
916 917
            && !target.contains("msvc")
        {
918
            Some(self.cc(target))
919
        } else if self.config.use_lld && !self.is_fuse_ld_lld(target) && self.build == target {
920
            Some(&self.initial_lld)
921 922
        } else {
            None
923
        }
A
Alex Crichton 已提交
924
    }
925

926 927 928 929 930 931
    // LLD is used through `-fuse-ld=lld` rather than directly.
    // Only MSVC targets use LLD directly at the moment.
    fn is_fuse_ld_lld(&self, target: TargetSelection) -> bool {
        self.config.use_lld && !target.contains("msvc")
    }

932 933 934 935 936 937 938 939 940 941 942 943 944 945 946
    fn lld_flags(&self, target: TargetSelection) -> impl Iterator<Item = String> {
        let mut options = [None, None];

        if self.config.use_lld {
            if self.is_fuse_ld_lld(target) {
                options[0] = Some("-Clink-arg=-fuse-ld=lld".to_string());
            }

            let threads = if target.contains("windows") { "/threads:1" } else { "--threads=1" };
            options[1] = Some(format!("-Clink-arg=-Wl,{}", threads));
        }

        std::array::IntoIter::new(options).flatten()
    }

947
    /// Returns if this target should statically link the C runtime, if specified
948
    fn crt_static(&self, target: TargetSelection) -> Option<bool> {
949 950 951
        if target.contains("pc-windows-msvc") {
            Some(true)
        } else {
M
Mark Rousskov 已提交
952
            self.config.target_config.get(&target).and_then(|t| t.crt_static)
953
        }
954 955
    }

956
    /// Returns the "musl root" for this `target`, if defined
957
    fn musl_root(&self, target: TargetSelection) -> Option<&Path> {
M
Mark Rousskov 已提交
958 959 960
        self.config
            .target_config
            .get(&target)
961
            .and_then(|t| t.musl_root.as_ref())
962
            .or_else(|| self.config.musl_root.as_ref())
963 964
            .map(|p| &**p)
    }
965

966
    /// Returns the "musl libdir" for this `target`.
967
    fn musl_libdir(&self, target: TargetSelection) -> Option<PathBuf> {
968 969 970 971 972 973 974
        let t = self.config.target_config.get(&target)?;
        if let libdir @ Some(_) = &t.musl_libdir {
            return libdir.clone();
        }
        self.musl_root(target).map(|root| root.join("lib"))
    }

975
    /// Returns the sysroot for the wasi target, if defined
976
    fn wasi_root(&self, target: TargetSelection) -> Option<&Path> {
M
Mark Rousskov 已提交
977
        self.config.target_config.get(&target).and_then(|t| t.wasi_root.as_ref()).map(|p| &**p)
978 979
    }

A
Alexander Regueiro 已提交
980
    /// Returns `true` if this is a no-std `target`, if defined
981
    fn no_std(&self, target: TargetSelection) -> Option<bool> {
M
Mark Rousskov 已提交
982
        self.config.target_config.get(&target).map(|t| t.no_std)
983 984
    }

A
Alexander Regueiro 已提交
985
    /// Returns `true` if the target will be tested using the `remote-test-client`
986
    /// and `remote-test-server` binaries.
987
    fn remote_tested(&self, target: TargetSelection) -> bool {
M
Mark Rousskov 已提交
988 989 990
        self.qemu_rootfs(target).is_some()
            || target.contains("android")
            || env::var_os("TEST_DEVICE_ADDR").is_some()
991 992
    }

993 994 995 996 997
    /// Returns the root of the "rootfs" image that this target will be using,
    /// if one was configured.
    ///
    /// If `Some` is returned then that means that tests for this target are
    /// emulated with QEMU and binaries will need to be shipped to the emulator.
998
    fn qemu_rootfs(&self, target: TargetSelection) -> Option<&Path> {
M
Mark Rousskov 已提交
999
        self.config.target_config.get(&target).and_then(|t| t.qemu_rootfs.as_ref()).map(|p| &**p)
1000 1001
    }

1002 1003 1004 1005
    /// Path to the python interpreter to use
    fn python(&self) -> &Path {
        self.config.python.as_ref().unwrap()
    }
1006

1007 1008 1009 1010 1011
    /// Temporary directory that extended error information is emitted to.
    fn extended_error_dir(&self) -> PathBuf {
        self.out.join("tmp/extended-error-metadata")
    }

1012 1013 1014 1015 1016 1017 1018 1019 1020 1021 1022 1023 1024 1025 1026 1027 1028 1029
    /// Tests whether the `compiler` compiling for `target` should be forced to
    /// use a stage1 compiler instead.
    ///
    /// Currently, by default, the build system does not perform a "full
    /// bootstrap" by default where we compile the compiler three times.
    /// Instead, we compile the compiler two times. The final stage (stage2)
    /// just copies the libraries from the previous stage, which is what this
    /// method detects.
    ///
    /// Here we return `true` if:
    ///
    /// * The build isn't performing a full bootstrap
    /// * The `compiler` is in the final stage, 2
    /// * We're not cross-compiling, so the artifacts are already available in
    ///   stage1
    ///
    /// When all of these conditions are met the build will lift artifacts from
    /// the previous stage forward.
1030
    fn force_use_stage1(&self, compiler: Compiler, target: TargetSelection) -> bool {
M
Mark Rousskov 已提交
1031 1032 1033
        !self.config.full_bootstrap
            && compiler.stage >= 2
            && (self.hosts.iter().any(|h| *h == target) || target == self.build)
1034
    }
1035 1036 1037 1038 1039 1040 1041 1042 1043

    /// Given `num` in the form "a.b.c" return a "release string" which
    /// describes the release version number.
    ///
    /// For example on nightly this returns "a.b.c-nightly", on beta it returns
    /// "a.b.c-beta.1" and on stable it just returns "a.b.c".
    fn release(&self, num: &str) -> String {
        match &self.config.channel[..] {
            "stable" => num.to_string(),
M
Mark Rousskov 已提交
1044 1045 1046 1047 1048 1049 1050
            "beta" => {
                if self.rust_info.is_git() {
                    format!("{}-beta.{}", num, self.beta_prerelease_version())
                } else {
                    format!("{}-beta", num)
                }
            }
1051 1052 1053 1054 1055
            "nightly" => format!("{}-nightly", num),
            _ => format!("{}-dev", num),
        }
    }

1056 1057
    fn beta_prerelease_version(&self) -> u32 {
        if let Some(s) = self.prerelease_version.get() {
M
Mark Rousskov 已提交
1058
            return s;
1059 1060
        }

1061 1062 1063
        // Figure out how many merge commits happened since we branched off master.
        // That's our beta number!
        // (Note that we use a `..` range, not the `...` symmetric difference.)
1064 1065 1066 1067 1068
        let count = output(
            Command::new("git")
                .arg("rev-list")
                .arg("--count")
                .arg("--merges")
1069
                .arg("refs/remotes/origin/master..HEAD")
1070 1071 1072 1073 1074 1075 1076
                .current_dir(&self.src),
        );
        let n = count.trim().parse().unwrap();
        self.prerelease_version.set(Some(n));
        n
    }

1077 1078
    /// Returns the value of `release` above for Rust itself.
    fn rust_release(&self) -> String {
1079
        self.release(&self.version)
1080 1081 1082 1083 1084 1085 1086 1087 1088 1089 1090 1091 1092 1093 1094 1095 1096 1097 1098
    }

    /// Returns the "package version" for a component given the `num` release
    /// number.
    ///
    /// The package version is typically what shows up in the names of tarballs.
    /// For channels like beta/nightly it's just the channel name, otherwise
    /// it's the `num` provided.
    fn package_vers(&self, num: &str) -> String {
        match &self.config.channel[..] {
            "stable" => num.to_string(),
            "beta" => "beta".to_string(),
            "nightly" => "nightly".to_string(),
            _ => format!("{}-dev", num),
        }
    }

    /// Returns the value of `package_vers` above for Rust itself.
    fn rust_package_vers(&self) -> String {
1099
        self.package_vers(&self.version)
1100 1101
    }

1102
    fn llvm_link_tools_dynamically(&self, target: TargetSelection) -> bool {
1103
        target.contains("linux-gnu") || target.contains("apple-darwin")
1104 1105
    }

1106 1107 1108 1109 1110 1111
    /// Returns the `version` string associated with this compiler for Rust
    /// itself.
    ///
    /// Note that this is a descriptive string which includes the commit date,
    /// sha, version, etc.
    fn rust_version(&self) -> String {
1112 1113 1114 1115
        let mut version = self.rust_info.version(self, &self.version);
        if let Some(ref s) = self.config.description {
            version.push_str(" (");
            version.push_str(s);
M
Matthias Krüger 已提交
1116
            version.push(')');
1117 1118
        }
        version
1119 1120
    }

A
Alexander Regueiro 已提交
1121
    /// Returns the full commit hash.
1122 1123 1124 1125
    fn rust_sha(&self) -> Option<&str> {
        self.rust_info.sha()
    }

N
Nick Cameron 已提交
1126 1127
    /// Returns the `a.b.c` version that the given package is at.
    fn release_num(&self, package: &str) -> String {
A
Alex Crichton 已提交
1128
        let toml_file_name = self.src.join(&format!("src/tools/{}/Cargo.toml", package));
1129
        let toml = t!(fs::read_to_string(&toml_file_name));
1130
        for line in toml.lines() {
L
Lzu Tao 已提交
1131 1132 1133 1134
            if let Some(stripped) =
                line.strip_prefix("version = \"").and_then(|s| s.strip_suffix("\""))
            {
                return stripped.to_owned();
1135 1136 1137
            }
        }

N
Nick Cameron 已提交
1138
        panic!("failed to find version in {}'s Cargo.toml", package)
1139 1140
    }

A
Alexander Regueiro 已提交
1141
    /// Returns `true` if unstable features should be enabled for the compiler
1142 1143 1144 1145 1146 1147 1148
    /// we're building.
    fn unstable_features(&self) -> bool {
        match &self.config.channel[..] {
            "stable" | "beta" => false,
            "nightly" | _ => true,
        }
    }
1149

1150 1151 1152
    /// Returns a Vec of all the dependencies of the given root crate,
    /// including transitive dependencies and the root itself. Only includes
    /// "local" crates (those in the local source tree, not from a registry).
1153
    fn in_tree_crates(&self, root: &str, target: Option<TargetSelection>) -> Vec<&Crate> {
1154
        let mut ret = Vec::new();
1155
        let mut list = vec![INTERNER.intern_str(root)];
1156 1157
        let mut visited = HashSet::new();
        while let Some(krate) = list.pop() {
1158
            let krate = &self.crates[&krate];
1159
            ret.push(krate);
1160
            for dep in &krate.deps {
1161 1162 1163 1164
                if !self.crates.contains_key(dep) {
                    // Ignore non-workspace members.
                    continue;
                }
1165 1166 1167 1168 1169 1170 1171 1172 1173
                // Don't include optional deps if their features are not
                // enabled. Ideally this would be computed from `cargo
                // metadata --features …`, but that is somewhat slow. Just
                // skip `build_helper` since there aren't any operations we
                // want to perform on it. In the future, we may want to
                // consider just filtering all build and dev dependencies in
                // metadata::build.
                if visited.insert(dep)
                    && dep != "build_helper"
1174 1175 1176
                    && (dep != "profiler_builtins"
                        || target
                            .map(|t| self.config.profiler_enabled(t))
M
Matthias Krüger 已提交
1177
                            .unwrap_or_else(|| self.config.any_profiler_enabled()))
1178 1179
                    && (dep != "rustc_codegen_llvm" || self.config.llvm_enabled())
                {
1180
                    list.push(*dep);
1181 1182 1183 1184 1185
                }
            }
        }
        ret
    }
1186

1187
    fn read_stamp_file(&self, stamp: &Path) -> Vec<(PathBuf, DependencyType)> {
1188 1189 1190 1191 1192
        if self.config.dry_run {
            return Vec::new();
        }

        let mut paths = Vec::new();
1193
        let contents = t!(fs::read(stamp), &stamp);
1194 1195 1196 1197
        // This is the method we use for extracting paths from the stamp file passed to us. See
        // run_cargo for more information (in compile.rs).
        for part in contents.split(|b| *b == 0) {
            if part.is_empty() {
M
Mark Rousskov 已提交
1198
                continue;
1199
            }
1200 1201 1202 1203 1204 1205
            let dependency_type = match part[0] as char {
                'h' => DependencyType::Host,
                's' => DependencyType::TargetSelfContained,
                't' => DependencyType::Target,
                _ => unreachable!(),
            };
J
John Kåre Alsaker 已提交
1206
            let path = PathBuf::from(t!(str::from_utf8(&part[1..])));
1207
            paths.push((path, dependency_type));
1208 1209 1210 1211 1212 1213
        }
        paths
    }

    /// Copies a file from `src` to `dst`
    pub fn copy(&self, src: &Path, dst: &Path) {
M
Mark Rousskov 已提交
1214 1215 1216
        if self.config.dry_run {
            return;
        }
1217
        self.verbose_than(1, &format!("Copy {:?} to {:?}", src, dst));
M
Mark Rousskov 已提交
1218 1219 1220
        if src == dst {
            return;
        }
1221
        let _ = fs::remove_file(&dst);
T
Tom Tromey 已提交
1222 1223 1224 1225 1226 1227 1228 1229 1230 1231
        let metadata = t!(src.symlink_metadata());
        if metadata.file_type().is_symlink() {
            let link = t!(fs::read_link(src));
            t!(symlink_file(link, dst));
        } else if let Ok(()) = fs::hard_link(src, dst) {
            // Attempt to "easy copy" by creating a hard link
            // (symlinks don't work on windows), but if that fails
            // just fall back to a slow `copy` operation.
        } else {
            if let Err(e) = fs::copy(src, dst) {
M
Mark Rousskov 已提交
1232
                panic!("failed to copy `{}` to `{}`: {}", src.display(), dst.display(), e)
T
Tom Tromey 已提交
1233 1234 1235 1236 1237
            }
            t!(fs::set_permissions(dst, metadata.permissions()));
            let atime = FileTime::from_last_access_time(&metadata);
            let mtime = FileTime::from_last_modification_time(&metadata);
            t!(filetime::set_file_times(dst, atime, mtime));
1238 1239 1240 1241 1242 1243
        }
    }

    /// Search-and-replaces within a file. (Not maximally efficiently: allocates a
    /// new string for each replacement.)
    pub fn replace_in_file(&self, path: &Path, replacements: &[(&str, &str)]) {
M
Mark Rousskov 已提交
1244 1245 1246
        if self.config.dry_run {
            return;
        }
1247 1248 1249 1250 1251 1252 1253 1254 1255 1256 1257 1258 1259 1260
        let mut contents = String::new();
        let mut file = t!(OpenOptions::new().read(true).write(true).open(path));
        t!(file.read_to_string(&mut contents));
        for &(target, replacement) in replacements {
            contents = contents.replace(target, replacement);
        }
        t!(file.seek(SeekFrom::Start(0)));
        t!(file.set_len(0));
        t!(file.write_all(contents.as_bytes()));
    }

    /// Copies the `src` directory recursively to `dst`. Both are assumed to exist
    /// when this function is called.
    pub fn cp_r(&self, src: &Path, dst: &Path) {
M
Mark Rousskov 已提交
1261 1262 1263
        if self.config.dry_run {
            return;
        }
1264
        for f in self.read_dir(src) {
1265 1266 1267 1268 1269 1270 1271 1272 1273 1274 1275 1276 1277 1278 1279 1280
            let path = f.path();
            let name = path.file_name().unwrap();
            let dst = dst.join(name);
            if t!(f.file_type()).is_dir() {
                t!(fs::create_dir_all(&dst));
                self.cp_r(&path, &dst);
            } else {
                let _ = fs::remove_file(&dst);
                self.copy(&path, &dst);
            }
        }
    }

    /// Copies the `src` directory recursively to `dst`. Both are assumed to exist
    /// when this function is called. Unwanted files or directories can be skipped
    /// by returning `false` from the filter function.
1281
    pub fn cp_filtered(&self, src: &Path, dst: &Path, filter: &dyn Fn(&Path) -> bool) {
1282 1283 1284 1285 1286
        // Immediately recurse with an empty relative path
        self.recurse_(src, dst, Path::new(""), filter)
    }

    // Inner function does the actual work
1287
    fn recurse_(&self, src: &Path, dst: &Path, relative: &Path, filter: &dyn Fn(&Path) -> bool) {
1288 1289 1290 1291 1292 1293 1294 1295 1296 1297 1298 1299 1300 1301 1302 1303 1304 1305 1306 1307 1308 1309 1310 1311 1312 1313
        for f in self.read_dir(src) {
            let path = f.path();
            let name = path.file_name().unwrap();
            let dst = dst.join(name);
            let relative = relative.join(name);
            // Only copy file or directory if the filter function returns true
            if filter(&relative) {
                if t!(f.file_type()).is_dir() {
                    let _ = fs::remove_dir_all(&dst);
                    self.create_dir(&dst);
                    self.recurse_(&path, &dst, &relative, filter);
                } else {
                    let _ = fs::remove_file(&dst);
                    self.copy(&path, &dst);
                }
            }
        }
    }

    fn copy_to_folder(&self, src: &Path, dest_folder: &Path) {
        let file_name = src.file_name().unwrap();
        let dest = dest_folder.join(file_name);
        self.copy(src, &dest);
    }

    fn install(&self, src: &Path, dstdir: &Path, perms: u32) {
M
Mark Rousskov 已提交
1314 1315 1316
        if self.config.dry_run {
            return;
        }
1317
        let dst = dstdir.join(src.file_name().unwrap());
O
O01eg 已提交
1318
        self.verbose_than(1, &format!("Install {:?} to {:?}", src, dst));
1319 1320 1321
        t!(fs::create_dir_all(dstdir));
        drop(fs::remove_file(&dst));
        {
1322 1323 1324
            if !src.exists() {
                panic!("Error: File \"{}\" not found!", src.display());
            }
1325 1326
            let metadata = t!(src.symlink_metadata());
            if let Err(e) = fs::copy(&src, &dst) {
M
Mark Rousskov 已提交
1327
                panic!("failed to copy `{}` to `{}`: {}", src.display(), dst.display(), e)
1328 1329 1330 1331 1332
            }
            t!(fs::set_permissions(&dst, metadata.permissions()));
            let atime = FileTime::from_last_access_time(&metadata);
            let mtime = FileTime::from_last_modification_time(&metadata);
            t!(filetime::set_file_times(&dst, atime, mtime));
1333 1334 1335 1336 1337
        }
        chmod(&dst, perms);
    }

    fn create(&self, path: &Path, s: &str) {
M
Mark Rousskov 已提交
1338 1339 1340
        if self.config.dry_run {
            return;
        }
1341 1342 1343 1344
        t!(fs::write(path, s));
    }

    fn read(&self, path: &Path) -> String {
M
Mark Rousskov 已提交
1345 1346 1347
        if self.config.dry_run {
            return String::new();
        }
1348
        t!(fs::read_to_string(path))
1349 1350 1351
    }

    fn create_dir(&self, dir: &Path) {
M
Mark Rousskov 已提交
1352 1353 1354
        if self.config.dry_run {
            return;
        }
1355 1356 1357 1358
        t!(fs::create_dir_all(dir))
    }

    fn remove_dir(&self, dir: &Path) {
M
Mark Rousskov 已提交
1359 1360 1361
        if self.config.dry_run {
            return;
        }
1362 1363 1364
        t!(fs::remove_dir_all(dir))
    }

M
Mark Rousskov 已提交
1365
    fn read_dir(&self, dir: &Path) -> impl Iterator<Item = fs::DirEntry> {
1366 1367 1368 1369 1370 1371 1372
        let iter = match fs::read_dir(dir) {
            Ok(v) => v,
            Err(_) if self.config.dry_run => return vec![].into_iter(),
            Err(err) => panic!("could not read dir {:?}: {:?}", dir, err),
        };
        iter.map(|e| t!(e)).collect::<Vec<_>>().into_iter()
    }
1373 1374

    fn remove(&self, f: &Path) {
M
Mark Rousskov 已提交
1375 1376 1377
        if self.config.dry_run {
            return;
        }
1378 1379
        fs::remove_file(f).unwrap_or_else(|_| panic!("failed to remove {:?}", f));
    }
1380 1381 1382 1383 1384 1385 1386 1387 1388 1389 1390 1391 1392 1393 1394

    /// Returns if config.ninja is enabled, and checks for ninja existence,
    /// exiting with a nicer error message if not.
    fn ninja(&self) -> bool {
        let mut cmd_finder = crate::sanity::Finder::new();

        if self.config.ninja_in_file {
            // Some Linux distros rename `ninja` to `ninja-build`.
            // CMake can work with either binary name.
            if cmd_finder.maybe_have("ninja-build").is_none()
                && cmd_finder.maybe_have("ninja").is_none()
            {
                eprintln!(
                    "
Couldn't find required command: ninja
J
Joshua Nelson 已提交
1395
You should install ninja, or set `ninja=false` in config.toml in the `[llvm]` section.
1396 1397 1398 1399 1400 1401 1402 1403 1404 1405 1406 1407 1408 1409 1410 1411 1412 1413 1414 1415 1416
"
                );
                std::process::exit(1);
            }
        }

        // If ninja isn't enabled but we're building for MSVC then we try
        // doubly hard to enable it. It was realized in #43767 that the msbuild
        // CMake generator for MSVC doesn't respect configuration options like
        // disabling LLVM assertions, which can often be quite important!
        //
        // In these cases we automatically enable Ninja if we find it in the
        // environment.
        if !self.config.ninja_in_file && self.config.build.contains("msvc") {
            if cmd_finder.maybe_have("ninja").is_some() {
                return true;
            }
        }

        self.config.ninja_in_file
    }
1417 1418 1419 1420 1421 1422
}

#[cfg(unix)]
fn chmod(path: &Path, perms: u32) {
    use std::os::unix::fs::*;
    t!(fs::set_permissions(path, fs::Permissions::from_mode(perms)));
A
Alex Crichton 已提交
1423
}
1424 1425 1426
#[cfg(windows)]
fn chmod(_path: &Path, _perms: u32) {}

1427
impl Compiler {
1428
    pub fn with_stage(mut self, stage: u32) -> Compiler {
1429 1430
        self.stage = stage;
        self
1431 1432
    }

A
Alexander Regueiro 已提交
1433
    /// Returns `true` if this is a snapshot compiler for `build`'s configuration
1434
    pub fn is_snapshot(&self, build: &Build) -> bool {
1435
        self.stage == 0 && self.host == build.build
1436
    }
1437

1438 1439 1440 1441
    /// Returns if this compiler should be treated as a final stage one in the
    /// current build session.
    /// This takes into account whether we're performing a full bootstrap or
    /// not; don't directly compare the stage with `2`!
1442
    pub fn is_final_stage(&self, build: &Build) -> bool {
1443 1444 1445
        let final_stage = if build.config.full_bootstrap { 2 } else { 1 };
        self.stage >= final_stage
    }
A
Alex Crichton 已提交
1446
}
1447 1448 1449 1450 1451 1452 1453 1454 1455 1456

fn envify(s: &str) -> String {
    s.chars()
        .map(|c| match c {
            '-' => '_',
            c => c,
        })
        .flat_map(|c| c.to_uppercase())
        .collect()
}