lib.rs 51.7 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
        }

486 487
        if let Subcommand::Format { check, paths } = &self.config.cmd {
            return format::format(self, *check, &paths);
A
Adam Perry 已提交
488 489
        }

T
Tommy Ip 已提交
490 491
        if let Subcommand::Clean { all } = self.config.cmd {
            return clean::clean(self, all);
492 493
        }

A
Antoine Martin 已提交
494 495
        if let Subcommand::Setup { profile } = &self.config.cmd {
            return setup::setup(&self.config.src, *profile);
J
Joshua Nelson 已提交
496 497
        }

498 499 500 501 502 503
        {
            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;
                }
504 505
            }
        }
506 507

        if !self.config.dry_run {
508
            {
509 510
                self.config.dry_run = true;
                let builder = builder::Builder::new(&self);
511 512
                builder.execute_cli();
            }
513 514
            self.config.dry_run = false;
            let builder = builder::Builder::new(&self);
515
            builder.execute_cli();
516 517
        } else {
            let builder = builder::Builder::new(&self);
518
            builder.execute_cli();
519
        }
520 521 522 523 524 525 526 527 528 529

        // 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);
        }
530 531 532 533 534
    }

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

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

555
        match self.config.llvm_libunwind {
556 557 558
            LlvmLibunwind::InTree => features.push_str(" llvm-libunwind"),
            LlvmLibunwind::System => features.push_str(" system-llvm-libunwind"),
            LlvmLibunwind::No => {}
559
        }
560 561 562
        if self.config.backtrace {
            features.push_str(" backtrace");
        }
563
        if self.config.profiler_enabled(target) {
564 565
            features.push_str(" profiler");
        }
M
Mark Simulacrum 已提交
566
        features
567 568
    }

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

        // 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 已提交
585
            features.push_str(" max_level_info");
G
Gus Wynn 已提交
586 587
        }

588
        features
589 590 591 592 593
    }

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

597
    fn tools_dir(&self, compiler: Compiler) -> PathBuf {
598 599 600 601
        let out = self
            .out
            .join(&*compiler.host.triple)
            .join(format!("stage{}-tools-bin", compiler.stage));
602 603 604 605
        t!(fs::create_dir_all(&out));
        out
    }

606 607 608 609
    /// 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.
610
    fn stage_out(&self, compiler: Compiler, mode: Mode) -> PathBuf {
611
        let suffix = match mode {
C
Collins Abitekaniza 已提交
612 613
            Mode::Std => "-std",
            Mode::Rustc => "-rustc",
614
            Mode::Codegen => "-codegen",
615
            Mode::ToolBootstrap => "-bootstrap-tools",
616
            Mode::ToolStd | Mode::ToolRustc => "-tools",
617
        };
618
        self.out.join(&*compiler.host.triple).join(format!("stage{}{}", compiler.stage, suffix))
619 620 621
    }

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

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

636 637
    fn lld_out(&self, target: TargetSelection) -> PathBuf {
        self.out.join(&*target.triple).join("lld")
638 639
    }

640
    /// Output directory for all documentation for a target
641 642
    fn doc_out(&self, target: TargetSelection) -> PathBuf {
        self.out.join(&*target.triple).join("doc")
643 644
    }

G
Guillaume Gomez 已提交
645 646 647 648
    fn test_out(&self, target: TargetSelection) -> PathBuf {
        self.out.join(&*target.triple).join("test")
    }

649
    /// Output directory for all documentation for a target
650 651
    fn compiler_doc_out(&self, target: TargetSelection) -> PathBuf {
        self.out.join(&*target.triple).join("compiler-doc")
652 653
    }

654
    /// Output directory for some generated md crate documentation for a target (temporary)
655 656
    fn md_doc_out(&self, target: TargetSelection) -> Interned<PathBuf> {
        INTERNER.intern_path(self.out.join(&*target.triple).join("md-doc"))
657 658
    }

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

667
        match self.config.target_config.get(&target) {
668
            Some(ref c) => c.llvm_config.is_none(),
M
Mark Rousskov 已提交
669
            None => true,
670 671 672
        }
    }

673
    /// Returns the path to `FileCheck` binary for the specified target
674
    fn llvm_filecheck(&self, target: TargetSelection) -> PathBuf {
675
        let target_config = self.config.target_config.get(&target);
676 677 678
        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 已提交
679
            let llvm_bindir = output(Command::new(s).arg("--bindir"));
680
            let filecheck = Path::new(llvm_bindir.trim()).join(exe("FileCheck", target));
681 682 683 684 685 686
            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 已提交
687
                let lib_filecheck =
688
                    Path::new(llvm_libdir.trim()).join("llvm").join(exe("FileCheck", target));
689 690 691 692 693 694 695 696 697
                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
                }
            }
698
        } else {
699
            let base = self.llvm_out(self.config.build).join("build");
700
            let base = if !self.ninja() && self.config.build.contains("msvc") {
J
John Kåre Alsaker 已提交
701 702 703 704 705 706 707 708 709
                if self.config.llvm_optimize {
                    if self.config.llvm_release_debuginfo {
                        base.join("RelWithDebInfo")
                    } else {
                        base.join("Release")
                    }
                } else {
                    base.join("Debug")
                }
710
            } else {
J
John Kåre Alsaker 已提交
711 712
                base
            };
713
            base.join("bin").join(exe("FileCheck", target))
714 715 716
        }
    }

717
    /// Directory for libraries built from C/C++ code and shared between stages.
718 719
    fn native_dir(&self, target: TargetSelection) -> PathBuf {
        self.out.join(&*target.triple).join("native")
720 721
    }

722 723
    /// Root output directory for rust_test_helpers library compiled for
    /// `target`
724
    fn test_helpers_out(&self, target: TargetSelection) -> PathBuf {
725
        self.native_dir(target).join("rust-test-helpers")
726 727
    }

728 729 730 731 732 733 734
    /// 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());
        }
    }

735 736
    /// Returns the libdir of the snapshot compiler.
    fn rustc_snapshot_libdir(&self) -> PathBuf {
737
        self.rustc_snapshot_sysroot().join(libdir(self.config.build))
738 739 740 741
    }

    /// Returns the sysroot of the snapshot compiler.
    fn rustc_snapshot_sysroot(&self) -> &Path {
742
        self.initial_rustc.parent().unwrap().parent().unwrap()
743 744 745 746
    }

    /// Runs a command, printing out nice contextual information if it fails.
    fn run(&self, cmd: &mut Command) {
M
Mark Rousskov 已提交
747 748 749
        if self.config.dry_run {
            return;
        }
750
        self.verbose(&format!("running: {:?}", cmd));
751
        run(cmd)
752 753
    }

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

763 764 765 766
    /// 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 已提交
767 768 769
        if self.config.dry_run {
            return true;
        }
770
        self.verbose(&format!("running: {:?}", cmd));
771
        try_run(cmd)
772 773 774 775 776 777
    }

    /// 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 已提交
778 779 780
        if self.config.dry_run {
            return true;
        }
781
        self.verbose(&format!("running: {:?}", cmd));
782
        try_run_suppressed(cmd)
783 784
    }

M
Mark Simulacrum 已提交
785 786 787 788
    pub fn is_verbose(&self) -> bool {
        self.verbosity > 0
    }

789 790
    /// Prints a message if this build is configured in verbose mode.
    fn verbose(&self, msg: &str) {
M
Mark Simulacrum 已提交
791
        if self.is_verbose() {
792 793 794 795
            println!("{}", msg);
        }
    }

796 797 798 799 800 801 802 803 804 805 806
    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);
        }
    }

807
    fn info(&self, msg: &str) {
M
Mark Rousskov 已提交
808 809 810
        if self.config.dry_run {
            return;
        }
811 812 813
        println!("{}", msg);
    }

814 815 816
    /// Returns the number of parallel jobs that have been configured for this
    /// build.
    fn jobs(&self) -> u32 {
M
Mark Simulacrum 已提交
817
        self.config.jobs.unwrap_or_else(|| num_cpus::get() as u32)
818 819
    }

820
    fn debuginfo_map_to(&self, which: GitRepo) -> Option<String> {
821
        if !self.config.rust_remap_debuginfo {
M
Mark Rousskov 已提交
822
            return None;
823 824
        }

825
        match which {
826
            GitRepo::Rustc => {
827
                let sha = self.rust_sha().unwrap_or(&self.version);
828
                Some(format!("/rustc/{}", sha))
829
            }
830 831
            GitRepo::Llvm => Some(String::from("/rustc/llvm")),
        }
832 833
    }

834
    /// Returns the path to the C compiler for the target specified.
835
    fn cc(&self, target: TargetSelection) -> &Path {
836
        self.cc[&target].path()
837 838 839 840
    }

    /// Returns a list of flags to pass to the C compiler for the target
    /// specified.
841
    fn cflags(&self, target: TargetSelection, which: GitRepo) -> Vec<String> {
842
        // Filter out -O and /O (the optimization flags) that we picked up from
A
Alex Crichton 已提交
843
        // cc-rs because the build scripts will determine that for themselves.
M
Mark Rousskov 已提交
844 845 846 847 848 849
        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>>();
850

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

        // Work around an apparently bad MinGW / GCC optimization,
        // See: http://lists.llvm.org/pipermail/cfe-dev/2016-December/051980.html
        // See: https://gcc.gnu.org/bugzilla/show_bug.cgi?id=78936
862
        if &*target.triple == "i686-pc-windows-gnu" {
863 864
            base.push("-fno-omit-frame-pointer".into());
        }
865

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

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

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

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

899
    /// Returns the path to the linker for the given target if it needs to be overridden.
900
    fn linker(&self, target: TargetSelection) -> Option<&Path> {
M
Mark Rousskov 已提交
901 902
        if let Some(linker) = self.config.target_config.get(&target).and_then(|c| c.linker.as_ref())
        {
903
            Some(linker)
P
Pang, Baoshan 已提交
904 905 906 907
        } 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 已提交
908
        } else if target != self.config.build
909
            && util::use_host_linker(target)
M
Mark Rousskov 已提交
910 911
            && !target.contains("msvc")
        {
912
            Some(self.cc(target))
913
        } else if self.config.use_lld && !self.is_fuse_ld_lld(target) && self.build == target {
914
            Some(&self.initial_lld)
915 916
        } else {
            None
917
        }
A
Alex Crichton 已提交
918
    }
919

920 921 922 923 924 925
    // 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")
    }

926
    /// Returns if this target should statically link the C runtime, if specified
927
    fn crt_static(&self, target: TargetSelection) -> Option<bool> {
928 929 930
        if target.contains("pc-windows-msvc") {
            Some(true)
        } else {
M
Mark Rousskov 已提交
931
            self.config.target_config.get(&target).and_then(|t| t.crt_static)
932
        }
933 934
    }

935
    /// Returns the "musl root" for this `target`, if defined
936
    fn musl_root(&self, target: TargetSelection) -> Option<&Path> {
M
Mark Rousskov 已提交
937 938 939
        self.config
            .target_config
            .get(&target)
940
            .and_then(|t| t.musl_root.as_ref())
941
            .or_else(|| self.config.musl_root.as_ref())
942 943
            .map(|p| &**p)
    }
944

945
    /// Returns the "musl libdir" for this `target`.
946
    fn musl_libdir(&self, target: TargetSelection) -> Option<PathBuf> {
947 948 949 950 951 952 953
        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"))
    }

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

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

A
Alexander Regueiro 已提交
964
    /// Returns `true` if the target will be tested using the `remote-test-client`
965
    /// and `remote-test-server` binaries.
966
    fn remote_tested(&self, target: TargetSelection) -> bool {
M
Mark Rousskov 已提交
967 968 969
        self.qemu_rootfs(target).is_some()
            || target.contains("android")
            || env::var_os("TEST_DEVICE_ADDR").is_some()
970 971
    }

972 973 974 975 976
    /// 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.
977
    fn qemu_rootfs(&self, target: TargetSelection) -> Option<&Path> {
M
Mark Rousskov 已提交
978
        self.config.target_config.get(&target).and_then(|t| t.qemu_rootfs.as_ref()).map(|p| &**p)
979 980
    }

981 982 983 984
    /// Path to the python interpreter to use
    fn python(&self) -> &Path {
        self.config.python.as_ref().unwrap()
    }
985

986 987 988 989 990
    /// Temporary directory that extended error information is emitted to.
    fn extended_error_dir(&self) -> PathBuf {
        self.out.join("tmp/extended-error-metadata")
    }

991 992 993 994 995 996 997 998 999 1000 1001 1002 1003 1004 1005 1006 1007 1008
    /// 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.
1009
    fn force_use_stage1(&self, compiler: Compiler, target: TargetSelection) -> bool {
M
Mark Rousskov 已提交
1010 1011 1012
        !self.config.full_bootstrap
            && compiler.stage >= 2
            && (self.hosts.iter().any(|h| *h == target) || target == self.build)
1013
    }
1014 1015 1016 1017 1018 1019 1020 1021 1022

    /// 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 已提交
1023 1024 1025 1026 1027 1028 1029
            "beta" => {
                if self.rust_info.is_git() {
                    format!("{}-beta.{}", num, self.beta_prerelease_version())
                } else {
                    format!("{}-beta", num)
                }
            }
1030 1031 1032 1033 1034
            "nightly" => format!("{}-nightly", num),
            _ => format!("{}-dev", num),
        }
    }

1035 1036
    fn beta_prerelease_version(&self) -> u32 {
        if let Some(s) = self.prerelease_version.get() {
M
Mark Rousskov 已提交
1037
            return s;
1038 1039
        }

1040 1041 1042
        // 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.)
1043 1044 1045 1046 1047
        let count = output(
            Command::new("git")
                .arg("rev-list")
                .arg("--count")
                .arg("--merges")
1048
                .arg("refs/remotes/origin/master..HEAD")
1049 1050 1051 1052 1053 1054 1055
                .current_dir(&self.src),
        );
        let n = count.trim().parse().unwrap();
        self.prerelease_version.set(Some(n));
        n
    }

1056 1057
    /// Returns the value of `release` above for Rust itself.
    fn rust_release(&self) -> String {
1058
        self.release(&self.version)
1059 1060 1061 1062 1063 1064 1065 1066 1067 1068 1069 1070 1071 1072 1073 1074 1075 1076 1077
    }

    /// 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 {
1078
        self.package_vers(&self.version)
1079 1080
    }

1081
    fn llvm_link_tools_dynamically(&self, target: TargetSelection) -> bool {
1082
        target.contains("linux-gnu") || target.contains("apple-darwin")
1083 1084
    }

1085 1086 1087 1088 1089 1090
    /// 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 {
1091 1092 1093 1094
        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 已提交
1095
            version.push(')');
1096 1097
        }
        version
1098 1099
    }

A
Alexander Regueiro 已提交
1100
    /// Returns the full commit hash.
1101 1102 1103 1104
    fn rust_sha(&self) -> Option<&str> {
        self.rust_info.sha()
    }

N
Nick Cameron 已提交
1105 1106
    /// Returns the `a.b.c` version that the given package is at.
    fn release_num(&self, package: &str) -> String {
A
Alex Crichton 已提交
1107
        let toml_file_name = self.src.join(&format!("src/tools/{}/Cargo.toml", package));
1108
        let toml = t!(fs::read_to_string(&toml_file_name));
1109
        for line in toml.lines() {
L
Lzu Tao 已提交
1110 1111 1112 1113
            if let Some(stripped) =
                line.strip_prefix("version = \"").and_then(|s| s.strip_suffix("\""))
            {
                return stripped.to_owned();
1114 1115 1116
            }
        }

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

A
Alexander Regueiro 已提交
1120
    /// Returns `true` if unstable features should be enabled for the compiler
1121 1122 1123 1124 1125 1126 1127
    /// we're building.
    fn unstable_features(&self) -> bool {
        match &self.config.channel[..] {
            "stable" | "beta" => false,
            "nightly" | _ => true,
        }
    }
1128

1129 1130 1131
    /// 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).
1132
    fn in_tree_crates(&self, root: &str, target: Option<TargetSelection>) -> Vec<&Crate> {
1133
        let mut ret = Vec::new();
1134
        let mut list = vec![INTERNER.intern_str(root)];
1135 1136
        let mut visited = HashSet::new();
        while let Some(krate) = list.pop() {
1137
            let krate = &self.crates[&krate];
1138
            ret.push(krate);
1139
            for dep in &krate.deps {
1140 1141 1142 1143
                if !self.crates.contains_key(dep) {
                    // Ignore non-workspace members.
                    continue;
                }
1144 1145 1146 1147 1148 1149 1150 1151 1152
                // 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"
1153 1154 1155
                    && (dep != "profiler_builtins"
                        || target
                            .map(|t| self.config.profiler_enabled(t))
M
Matthias Krüger 已提交
1156
                            .unwrap_or_else(|| self.config.any_profiler_enabled()))
1157 1158
                    && (dep != "rustc_codegen_llvm" || self.config.llvm_enabled())
                {
1159
                    list.push(*dep);
1160 1161 1162 1163 1164
                }
            }
        }
        ret
    }
1165

1166
    fn read_stamp_file(&self, stamp: &Path) -> Vec<(PathBuf, DependencyType)> {
1167 1168 1169 1170 1171
        if self.config.dry_run {
            return Vec::new();
        }

        let mut paths = Vec::new();
1172
        let contents = t!(fs::read(stamp), &stamp);
1173 1174 1175 1176
        // 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 已提交
1177
                continue;
1178
            }
1179 1180 1181 1182 1183 1184
            let dependency_type = match part[0] as char {
                'h' => DependencyType::Host,
                's' => DependencyType::TargetSelfContained,
                't' => DependencyType::Target,
                _ => unreachable!(),
            };
J
John Kåre Alsaker 已提交
1185
            let path = PathBuf::from(t!(str::from_utf8(&part[1..])));
1186
            paths.push((path, dependency_type));
1187 1188 1189 1190 1191 1192
        }
        paths
    }

    /// Copies a file from `src` to `dst`
    pub fn copy(&self, src: &Path, dst: &Path) {
M
Mark Rousskov 已提交
1193 1194 1195
        if self.config.dry_run {
            return;
        }
1196
        self.verbose_than(1, &format!("Copy {:?} to {:?}", src, dst));
M
Mark Rousskov 已提交
1197 1198 1199
        if src == dst {
            return;
        }
1200
        let _ = fs::remove_file(&dst);
T
Tom Tromey 已提交
1201 1202 1203 1204 1205 1206 1207 1208 1209 1210
        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 已提交
1211
                panic!("failed to copy `{}` to `{}`: {}", src.display(), dst.display(), e)
T
Tom Tromey 已提交
1212 1213 1214 1215 1216
            }
            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));
1217 1218 1219 1220 1221 1222
        }
    }

    /// 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 已提交
1223 1224 1225
        if self.config.dry_run {
            return;
        }
1226 1227 1228 1229 1230 1231 1232 1233 1234 1235 1236 1237 1238 1239
        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 已提交
1240 1241 1242
        if self.config.dry_run {
            return;
        }
1243
        for f in self.read_dir(src) {
1244 1245 1246 1247 1248 1249 1250 1251 1252 1253 1254 1255 1256 1257 1258 1259
            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.
1260
    pub fn cp_filtered(&self, src: &Path, dst: &Path, filter: &dyn Fn(&Path) -> bool) {
1261 1262 1263 1264 1265
        // Immediately recurse with an empty relative path
        self.recurse_(src, dst, Path::new(""), filter)
    }

    // Inner function does the actual work
1266
    fn recurse_(&self, src: &Path, dst: &Path, relative: &Path, filter: &dyn Fn(&Path) -> bool) {
1267 1268 1269 1270 1271 1272 1273 1274 1275 1276 1277 1278 1279 1280 1281 1282 1283 1284 1285 1286 1287 1288 1289 1290 1291 1292
        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 已提交
1293 1294 1295
        if self.config.dry_run {
            return;
        }
1296
        let dst = dstdir.join(src.file_name().unwrap());
O
O01eg 已提交
1297
        self.verbose_than(1, &format!("Install {:?} to {:?}", src, dst));
1298 1299 1300
        t!(fs::create_dir_all(dstdir));
        drop(fs::remove_file(&dst));
        {
1301 1302 1303
            if !src.exists() {
                panic!("Error: File \"{}\" not found!", src.display());
            }
1304 1305
            let metadata = t!(src.symlink_metadata());
            if let Err(e) = fs::copy(&src, &dst) {
M
Mark Rousskov 已提交
1306
                panic!("failed to copy `{}` to `{}`: {}", src.display(), dst.display(), e)
1307 1308 1309 1310 1311
            }
            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));
1312 1313 1314 1315 1316
        }
        chmod(&dst, perms);
    }

    fn create(&self, path: &Path, s: &str) {
M
Mark Rousskov 已提交
1317 1318 1319
        if self.config.dry_run {
            return;
        }
1320 1321 1322 1323
        t!(fs::write(path, s));
    }

    fn read(&self, path: &Path) -> String {
M
Mark Rousskov 已提交
1324 1325 1326
        if self.config.dry_run {
            return String::new();
        }
1327
        t!(fs::read_to_string(path))
1328 1329 1330
    }

    fn create_dir(&self, dir: &Path) {
M
Mark Rousskov 已提交
1331 1332 1333
        if self.config.dry_run {
            return;
        }
1334 1335 1336 1337
        t!(fs::create_dir_all(dir))
    }

    fn remove_dir(&self, dir: &Path) {
M
Mark Rousskov 已提交
1338 1339 1340
        if self.config.dry_run {
            return;
        }
1341 1342 1343
        t!(fs::remove_dir_all(dir))
    }

M
Mark Rousskov 已提交
1344
    fn read_dir(&self, dir: &Path) -> impl Iterator<Item = fs::DirEntry> {
1345 1346 1347 1348 1349 1350 1351
        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()
    }
1352 1353

    fn remove(&self, f: &Path) {
M
Mark Rousskov 已提交
1354 1355 1356
        if self.config.dry_run {
            return;
        }
1357 1358
        fs::remove_file(f).unwrap_or_else(|_| panic!("failed to remove {:?}", f));
    }
1359 1360 1361 1362 1363 1364 1365 1366 1367 1368 1369 1370 1371 1372 1373

    /// 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 已提交
1374
You should install ninja, or set `ninja=false` in config.toml in the `[llvm]` section.
1375 1376 1377 1378 1379 1380 1381 1382 1383 1384 1385 1386 1387 1388 1389 1390 1391 1392 1393 1394 1395
"
                );
                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
    }
1396 1397 1398 1399 1400 1401
}

#[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 已提交
1402
}
1403 1404 1405
#[cfg(windows)]
fn chmod(_path: &Path, _perms: u32) {}

1406
impl Compiler {
1407
    pub fn with_stage(mut self, stage: u32) -> Compiler {
1408 1409
        self.stage = stage;
        self
1410 1411
    }

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

1417 1418 1419 1420
    /// 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`!
1421
    pub fn is_final_stage(&self, build: &Build) -> bool {
1422 1423 1424
        let final_stage = if build.config.full_bootstrap { 2 } else { 1 };
        self.stage >= final_stage
    }
A
Alex Crichton 已提交
1425
}
1426 1427 1428 1429 1430 1431 1432 1433 1434 1435

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