2015-11-19 15:20:12 -08:00
|
|
|
// Copyright 2015 The Rust Project Developers. See the COPYRIGHT
|
|
|
|
// file at the top-level directory of this distribution and at
|
|
|
|
// http://rust-lang.org/COPYRIGHT.
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
|
|
|
|
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
|
|
|
|
// option. This file may not be copied, modified, or distributed
|
|
|
|
// except according to those terms.
|
|
|
|
|
2016-07-05 21:58:20 -07:00
|
|
|
//! Implementation of rustbuild, the Rust build system.
|
2016-05-02 15:16:15 -07:00
|
|
|
//!
|
2016-07-05 21:58:20 -07:00
|
|
|
//! 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
|
|
|
|
//! builds, building artifacts like LLVM, etc.
|
|
|
|
//!
|
|
|
|
//! More documentation can be found in each respective module below.
|
|
|
|
|
|
|
|
extern crate build_helper;
|
|
|
|
extern crate cmake;
|
|
|
|
extern crate filetime;
|
|
|
|
extern crate gcc;
|
|
|
|
extern crate getopts;
|
|
|
|
extern crate md5;
|
|
|
|
extern crate num_cpus;
|
|
|
|
extern crate rustc_serialize;
|
|
|
|
extern crate toml;
|
2016-07-09 13:33:03 +01:00
|
|
|
extern crate regex;
|
2016-05-02 15:16:15 -07:00
|
|
|
|
2016-07-05 21:58:20 -07:00
|
|
|
use std::collections::HashMap;
|
2015-11-19 15:20:12 -08:00
|
|
|
use std::env;
|
2016-07-05 21:58:20 -07:00
|
|
|
use std::fs::{self, File};
|
2016-08-27 21:33:38 -05:00
|
|
|
use std::path::{Component, PathBuf, Path};
|
2016-07-05 21:58:20 -07:00
|
|
|
use std::process::Command;
|
|
|
|
|
|
|
|
use build_helper::{run_silent, output};
|
|
|
|
|
|
|
|
use util::{exe, mtime, libdir, add_lib_path};
|
|
|
|
|
|
|
|
/// A helper macro to `unwrap` a result except also print out details like:
|
|
|
|
///
|
|
|
|
/// * The file/line of the panic
|
|
|
|
/// * The expression that failed
|
|
|
|
/// * The error itself
|
|
|
|
///
|
|
|
|
/// This is currently used judiciously throughout the build system rather than
|
2016-08-27 02:10:09 -07:00
|
|
|
/// using a `Result` with `try!`, but this may change one day...
|
2016-07-05 21:58:20 -07:00
|
|
|
macro_rules! t {
|
|
|
|
($e:expr) => (match $e {
|
|
|
|
Ok(e) => e,
|
|
|
|
Err(e) => panic!("{} failed with {}", stringify!($e), e),
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
mod cc;
|
|
|
|
mod channel;
|
|
|
|
mod check;
|
|
|
|
mod clean;
|
|
|
|
mod compile;
|
|
|
|
mod config;
|
|
|
|
mod dist;
|
|
|
|
mod doc;
|
|
|
|
mod flags;
|
|
|
|
mod native;
|
|
|
|
mod sanity;
|
|
|
|
mod step;
|
|
|
|
pub mod util;
|
|
|
|
|
|
|
|
#[cfg(windows)]
|
|
|
|
mod job;
|
|
|
|
|
|
|
|
#[cfg(not(windows))]
|
|
|
|
mod job {
|
|
|
|
pub unsafe fn setup() {}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub use config::Config;
|
|
|
|
pub use flags::Flags;
|
|
|
|
|
|
|
|
/// 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.
|
|
|
|
#[derive(Eq, PartialEq, Clone, Copy, Hash, Debug)]
|
|
|
|
pub struct Compiler<'a> {
|
|
|
|
stage: u32,
|
|
|
|
host: &'a str,
|
|
|
|
}
|
|
|
|
|
|
|
|
/// 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 {
|
|
|
|
// User-specified configuration via config.toml
|
|
|
|
config: Config,
|
|
|
|
|
|
|
|
// User-specified configuration via CLI flags
|
|
|
|
flags: Flags,
|
|
|
|
|
|
|
|
// Derived properties from the above two configurations
|
|
|
|
cargo: PathBuf,
|
|
|
|
rustc: PathBuf,
|
|
|
|
src: PathBuf,
|
|
|
|
out: PathBuf,
|
|
|
|
release: String,
|
|
|
|
unstable_features: bool,
|
|
|
|
ver_hash: Option<String>,
|
|
|
|
short_ver_hash: Option<String>,
|
|
|
|
ver_date: Option<String>,
|
|
|
|
version: String,
|
|
|
|
package_vers: String,
|
2016-07-14 19:39:55 +02:00
|
|
|
local_rebuild: bool,
|
2016-07-05 21:58:20 -07:00
|
|
|
bootstrap_key: String,
|
|
|
|
bootstrap_key_stage0: String,
|
|
|
|
|
|
|
|
// Probed tools at runtime
|
|
|
|
gdb_version: Option<String>,
|
|
|
|
lldb_version: Option<String>,
|
|
|
|
lldb_python_dir: Option<String>,
|
|
|
|
|
|
|
|
// Runtime state filled in later on
|
|
|
|
cc: HashMap<String, (gcc::Tool, Option<PathBuf>)>,
|
|
|
|
cxx: HashMap<String, gcc::Tool>,
|
|
|
|
}
|
|
|
|
|
|
|
|
/// 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.
|
|
|
|
#[derive(Clone, Copy)]
|
|
|
|
pub enum Mode {
|
|
|
|
/// This cargo is going to build the standard library, placing output in the
|
|
|
|
/// "stageN-std" directory.
|
|
|
|
Libstd,
|
|
|
|
|
|
|
|
/// This cargo is going to build libtest, placing output in the
|
|
|
|
/// "stageN-test" directory.
|
|
|
|
Libtest,
|
|
|
|
|
|
|
|
/// This cargo is going to build librustc and compiler libraries, placing
|
|
|
|
/// output in the "stageN-rustc" directory.
|
|
|
|
Librustc,
|
|
|
|
|
|
|
|
/// This cargo is going to some build tool, placing output in the
|
|
|
|
/// "stageN-tools" directory.
|
|
|
|
Tool,
|
|
|
|
}
|
|
|
|
|
|
|
|
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.
|
|
|
|
pub fn new(flags: Flags, config: Config) -> Build {
|
|
|
|
let cwd = t!(env::current_dir());
|
|
|
|
let src = flags.src.clone().unwrap_or(cwd.clone());
|
|
|
|
let out = cwd.join("build");
|
|
|
|
|
|
|
|
let stage0_root = out.join(&config.build).join("stage0/bin");
|
|
|
|
let rustc = match config.rustc {
|
|
|
|
Some(ref s) => PathBuf::from(s),
|
|
|
|
None => stage0_root.join(exe("rustc", &config.build)),
|
|
|
|
};
|
|
|
|
let cargo = match config.cargo {
|
|
|
|
Some(ref s) => PathBuf::from(s),
|
|
|
|
None => stage0_root.join(exe("cargo", &config.build)),
|
|
|
|
};
|
2016-07-14 19:39:55 +02:00
|
|
|
let local_rebuild = config.local_rebuild;
|
2016-07-05 21:58:20 -07:00
|
|
|
|
|
|
|
Build {
|
|
|
|
flags: flags,
|
|
|
|
config: config,
|
|
|
|
cargo: cargo,
|
|
|
|
rustc: rustc,
|
|
|
|
src: src,
|
|
|
|
out: out,
|
|
|
|
|
|
|
|
release: String::new(),
|
|
|
|
unstable_features: false,
|
|
|
|
ver_hash: None,
|
|
|
|
short_ver_hash: None,
|
|
|
|
ver_date: None,
|
|
|
|
version: String::new(),
|
2016-07-14 19:39:55 +02:00
|
|
|
local_rebuild: local_rebuild,
|
2016-07-05 21:58:20 -07:00
|
|
|
bootstrap_key: String::new(),
|
|
|
|
bootstrap_key_stage0: String::new(),
|
|
|
|
package_vers: String::new(),
|
|
|
|
cc: HashMap::new(),
|
|
|
|
cxx: HashMap::new(),
|
|
|
|
gdb_version: None,
|
|
|
|
lldb_version: None,
|
|
|
|
lldb_python_dir: None,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Executes the entire build, as configured by the flags and configuration.
|
|
|
|
pub fn build(&mut self) {
|
|
|
|
use step::Source::*;
|
|
|
|
|
|
|
|
unsafe {
|
|
|
|
job::setup();
|
|
|
|
}
|
|
|
|
|
|
|
|
if self.flags.clean {
|
|
|
|
return clean::clean(self);
|
|
|
|
}
|
|
|
|
|
|
|
|
self.verbose("finding compilers");
|
|
|
|
cc::find(self);
|
|
|
|
self.verbose("running sanity check");
|
|
|
|
sanity::check(self);
|
|
|
|
self.verbose("collecting channel variables");
|
|
|
|
channel::collect(self);
|
2016-07-14 19:39:55 +02:00
|
|
|
// If local-rust is the same as the current version, then force a local-rebuild
|
|
|
|
let local_version_verbose = output(
|
|
|
|
Command::new(&self.rustc).arg("--version").arg("--verbose"));
|
|
|
|
let local_release = local_version_verbose
|
|
|
|
.lines().filter(|x| x.starts_with("release:"))
|
|
|
|
.next().unwrap().trim_left_matches("release:").trim();
|
|
|
|
if local_release == self.release {
|
|
|
|
self.verbose(&format!("auto-detected local-rebuild {}", self.release));
|
|
|
|
self.local_rebuild = true;
|
|
|
|
}
|
2016-07-05 21:58:20 -07:00
|
|
|
self.verbose("updating submodules");
|
|
|
|
self.update_submodules();
|
|
|
|
|
|
|
|
// The main loop of the build system.
|
|
|
|
//
|
|
|
|
// The `step::all` function returns a topographically sorted list of all
|
|
|
|
// steps that need to be executed as part of this build. Each step has a
|
|
|
|
// corresponding entry in `step.rs` and indicates some unit of work that
|
|
|
|
// needs to be done as part of the build.
|
|
|
|
//
|
|
|
|
// Almost all of these are simple one-liners that shell out to the
|
|
|
|
// corresponding functionality in the extra modules, where more
|
|
|
|
// documentation can be found.
|
2016-09-23 01:36:01 +00:00
|
|
|
let steps = step::all(self);
|
|
|
|
|
|
|
|
self.verbose("bootstrap build plan:");
|
|
|
|
for step in &steps {
|
|
|
|
self.verbose(&format!("{:?}", step));
|
|
|
|
}
|
|
|
|
|
|
|
|
for target in steps {
|
2016-07-05 21:58:20 -07:00
|
|
|
let doc_out = self.out.join(&target.target).join("doc");
|
|
|
|
match target.src {
|
|
|
|
Llvm { _dummy } => {
|
|
|
|
native::llvm(self, target.target);
|
|
|
|
}
|
|
|
|
TestHelpers { _dummy } => {
|
|
|
|
native::test_helpers(self, target.target);
|
|
|
|
}
|
|
|
|
Libstd { compiler } => {
|
|
|
|
compile::std(self, target.target, &compiler);
|
|
|
|
}
|
|
|
|
Libtest { compiler } => {
|
|
|
|
compile::test(self, target.target, &compiler);
|
|
|
|
}
|
|
|
|
Librustc { compiler } => {
|
|
|
|
compile::rustc(self, target.target, &compiler);
|
|
|
|
}
|
|
|
|
LibstdLink { compiler, host } => {
|
|
|
|
compile::std_link(self, target.target, &compiler, host);
|
|
|
|
}
|
|
|
|
LibtestLink { compiler, host } => {
|
|
|
|
compile::test_link(self, target.target, &compiler, host);
|
|
|
|
}
|
|
|
|
LibrustcLink { compiler, host } => {
|
|
|
|
compile::rustc_link(self, target.target, &compiler, host);
|
|
|
|
}
|
|
|
|
Rustc { stage: 0 } => {
|
|
|
|
// nothing to do...
|
|
|
|
}
|
|
|
|
Rustc { stage } => {
|
|
|
|
compile::assemble_rustc(self, stage, target.target);
|
|
|
|
}
|
|
|
|
ToolLinkchecker { stage } => {
|
|
|
|
compile::tool(self, stage, target.target, "linkchecker");
|
|
|
|
}
|
|
|
|
ToolRustbook { stage } => {
|
|
|
|
compile::tool(self, stage, target.target, "rustbook");
|
|
|
|
}
|
|
|
|
ToolErrorIndex { stage } => {
|
|
|
|
compile::tool(self, stage, target.target,
|
|
|
|
"error_index_generator");
|
|
|
|
}
|
|
|
|
ToolCargoTest { stage } => {
|
|
|
|
compile::tool(self, stage, target.target, "cargotest");
|
|
|
|
}
|
|
|
|
ToolTidy { stage } => {
|
|
|
|
compile::tool(self, stage, target.target, "tidy");
|
|
|
|
}
|
|
|
|
ToolCompiletest { stage } => {
|
|
|
|
compile::tool(self, stage, target.target, "compiletest");
|
|
|
|
}
|
|
|
|
DocBook { stage } => {
|
|
|
|
doc::rustbook(self, stage, target.target, "book", &doc_out);
|
|
|
|
}
|
|
|
|
DocNomicon { stage } => {
|
|
|
|
doc::rustbook(self, stage, target.target, "nomicon",
|
|
|
|
&doc_out);
|
|
|
|
}
|
|
|
|
DocStandalone { stage } => {
|
|
|
|
doc::standalone(self, stage, target.target, &doc_out);
|
|
|
|
}
|
|
|
|
DocStd { stage } => {
|
|
|
|
doc::std(self, stage, target.target, &doc_out);
|
|
|
|
}
|
|
|
|
DocTest { stage } => {
|
|
|
|
doc::test(self, stage, target.target, &doc_out);
|
|
|
|
}
|
|
|
|
DocRustc { stage } => {
|
|
|
|
doc::rustc(self, stage, target.target, &doc_out);
|
|
|
|
}
|
|
|
|
DocErrorIndex { stage } => {
|
|
|
|
doc::error_index(self, stage, target.target, &doc_out);
|
|
|
|
}
|
|
|
|
|
|
|
|
CheckLinkcheck { stage } => {
|
|
|
|
check::linkcheck(self, stage, target.target);
|
|
|
|
}
|
|
|
|
CheckCargoTest { stage } => {
|
|
|
|
check::cargotest(self, stage, target.target);
|
|
|
|
}
|
|
|
|
CheckTidy { stage } => {
|
|
|
|
check::tidy(self, stage, target.target);
|
|
|
|
}
|
|
|
|
CheckRPass { compiler } => {
|
|
|
|
check::compiletest(self, &compiler, target.target,
|
|
|
|
"run-pass", "run-pass");
|
|
|
|
}
|
|
|
|
CheckRPassFull { compiler } => {
|
|
|
|
check::compiletest(self, &compiler, target.target,
|
|
|
|
"run-pass", "run-pass-fulldeps");
|
|
|
|
}
|
|
|
|
CheckCFail { compiler } => {
|
|
|
|
check::compiletest(self, &compiler, target.target,
|
|
|
|
"compile-fail", "compile-fail");
|
|
|
|
}
|
|
|
|
CheckCFailFull { compiler } => {
|
|
|
|
check::compiletest(self, &compiler, target.target,
|
|
|
|
"compile-fail", "compile-fail-fulldeps")
|
|
|
|
}
|
|
|
|
CheckPFail { compiler } => {
|
|
|
|
check::compiletest(self, &compiler, target.target,
|
|
|
|
"parse-fail", "parse-fail");
|
|
|
|
}
|
|
|
|
CheckRFail { compiler } => {
|
|
|
|
check::compiletest(self, &compiler, target.target,
|
|
|
|
"run-fail", "run-fail");
|
|
|
|
}
|
|
|
|
CheckRFailFull { compiler } => {
|
|
|
|
check::compiletest(self, &compiler, target.target,
|
|
|
|
"run-fail", "run-fail-fulldeps");
|
|
|
|
}
|
|
|
|
CheckPretty { compiler } => {
|
|
|
|
check::compiletest(self, &compiler, target.target,
|
|
|
|
"pretty", "pretty");
|
|
|
|
}
|
|
|
|
CheckPrettyRPass { compiler } => {
|
|
|
|
check::compiletest(self, &compiler, target.target,
|
|
|
|
"pretty", "run-pass");
|
|
|
|
}
|
|
|
|
CheckPrettyRPassFull { compiler } => {
|
|
|
|
check::compiletest(self, &compiler, target.target,
|
|
|
|
"pretty", "run-pass-fulldeps");
|
|
|
|
}
|
|
|
|
CheckPrettyRFail { compiler } => {
|
|
|
|
check::compiletest(self, &compiler, target.target,
|
|
|
|
"pretty", "run-fail");
|
|
|
|
}
|
|
|
|
CheckPrettyRFailFull { compiler } => {
|
|
|
|
check::compiletest(self, &compiler, target.target,
|
|
|
|
"pretty", "run-fail-fulldeps");
|
|
|
|
}
|
|
|
|
CheckPrettyRPassValgrind { compiler } => {
|
|
|
|
check::compiletest(self, &compiler, target.target,
|
|
|
|
"pretty", "run-pass-valgrind");
|
|
|
|
}
|
2016-07-07 16:40:01 -07:00
|
|
|
CheckMirOpt { compiler } => {
|
|
|
|
check::compiletest(self, &compiler, target.target,
|
|
|
|
"mir-opt", "mir-opt");
|
|
|
|
}
|
2016-07-05 21:58:20 -07:00
|
|
|
CheckCodegen { compiler } => {
|
2016-08-27 22:14:29 -05:00
|
|
|
if self.config.codegen_tests {
|
|
|
|
check::compiletest(self, &compiler, target.target,
|
|
|
|
"codegen", "codegen");
|
|
|
|
}
|
2016-07-05 21:58:20 -07:00
|
|
|
}
|
|
|
|
CheckCodegenUnits { compiler } => {
|
|
|
|
check::compiletest(self, &compiler, target.target,
|
|
|
|
"codegen-units", "codegen-units");
|
|
|
|
}
|
|
|
|
CheckIncremental { compiler } => {
|
|
|
|
check::compiletest(self, &compiler, target.target,
|
|
|
|
"incremental", "incremental");
|
|
|
|
}
|
|
|
|
CheckUi { compiler } => {
|
|
|
|
check::compiletest(self, &compiler, target.target,
|
|
|
|
"ui", "ui");
|
|
|
|
}
|
|
|
|
CheckDebuginfo { compiler } => {
|
|
|
|
if target.target.contains("msvc") {
|
|
|
|
// nothing to do
|
|
|
|
} else if target.target.contains("apple") {
|
|
|
|
check::compiletest(self, &compiler, target.target,
|
|
|
|
"debuginfo-lldb", "debuginfo");
|
|
|
|
} else {
|
|
|
|
check::compiletest(self, &compiler, target.target,
|
|
|
|
"debuginfo-gdb", "debuginfo");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
CheckRustdoc { compiler } => {
|
|
|
|
check::compiletest(self, &compiler, target.target,
|
|
|
|
"rustdoc", "rustdoc");
|
|
|
|
}
|
|
|
|
CheckRPassValgrind { compiler } => {
|
|
|
|
check::compiletest(self, &compiler, target.target,
|
|
|
|
"run-pass-valgrind", "run-pass-valgrind");
|
|
|
|
}
|
|
|
|
CheckDocs { compiler } => {
|
|
|
|
check::docs(self, &compiler);
|
|
|
|
}
|
|
|
|
CheckErrorIndex { compiler } => {
|
|
|
|
check::error_index(self, &compiler);
|
|
|
|
}
|
|
|
|
CheckRMake { compiler } => {
|
|
|
|
check::compiletest(self, &compiler, target.target,
|
|
|
|
"run-make", "run-make")
|
|
|
|
}
|
|
|
|
CheckCrateStd { compiler } => {
|
|
|
|
check::krate(self, &compiler, target.target, Mode::Libstd)
|
|
|
|
}
|
|
|
|
CheckCrateTest { compiler } => {
|
|
|
|
check::krate(self, &compiler, target.target, Mode::Libtest)
|
|
|
|
}
|
|
|
|
CheckCrateRustc { compiler } => {
|
|
|
|
check::krate(self, &compiler, target.target, Mode::Librustc)
|
|
|
|
}
|
|
|
|
|
|
|
|
DistDocs { stage } => dist::docs(self, stage, target.target),
|
|
|
|
DistMingw { _dummy } => dist::mingw(self, target.target),
|
|
|
|
DistRustc { stage } => dist::rustc(self, stage, target.target),
|
|
|
|
DistStd { compiler } => dist::std(self, &compiler, target.target),
|
2016-07-09 13:33:03 +01:00
|
|
|
DistSrc { _dummy } => dist::rust_src(self),
|
2016-07-05 21:58:20 -07:00
|
|
|
|
|
|
|
DebuggerScripts { stage } => {
|
|
|
|
let compiler = Compiler::new(stage, target.target);
|
|
|
|
dist::debugger_scripts(self,
|
|
|
|
&self.sysroot(&compiler),
|
|
|
|
target.target);
|
|
|
|
}
|
|
|
|
|
|
|
|
AndroidCopyLibs { compiler } => {
|
|
|
|
check::android_copy_libs(self, &compiler, target.target);
|
|
|
|
}
|
|
|
|
|
|
|
|
// pseudo-steps
|
|
|
|
Dist { .. } |
|
|
|
|
Doc { .. } |
|
|
|
|
CheckTarget { .. } |
|
|
|
|
Check { .. } => {}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Updates all git submodules that we have.
|
|
|
|
///
|
|
|
|
/// This will detect if any submodules are out of date an run the necessary
|
|
|
|
/// commands to sync them all with upstream.
|
|
|
|
fn update_submodules(&self) {
|
2016-08-27 21:33:38 -05:00
|
|
|
struct Submodule<'a> {
|
|
|
|
path: &'a Path,
|
|
|
|
state: State,
|
|
|
|
}
|
|
|
|
|
|
|
|
enum State {
|
|
|
|
// The submodule may have staged/unstaged changes
|
|
|
|
MaybeDirty,
|
|
|
|
// Or could be initialized but never updated
|
|
|
|
NotInitialized,
|
|
|
|
// The submodule, itself, has extra commits but those changes haven't been commited to
|
|
|
|
// the (outer) git repository
|
|
|
|
OutOfSync,
|
|
|
|
}
|
|
|
|
|
2016-07-05 21:58:20 -07:00
|
|
|
if !self.config.submodules {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if fs::metadata(self.src.join(".git")).is_err() {
|
|
|
|
return
|
|
|
|
}
|
2016-08-27 21:33:38 -05:00
|
|
|
let git = || {
|
|
|
|
let mut cmd = Command::new("git");
|
|
|
|
cmd.current_dir(&self.src);
|
|
|
|
return cmd
|
|
|
|
};
|
2016-07-05 21:58:20 -07:00
|
|
|
let git_submodule = || {
|
|
|
|
let mut cmd = Command::new("git");
|
|
|
|
cmd.current_dir(&self.src).arg("submodule");
|
|
|
|
return cmd
|
|
|
|
};
|
|
|
|
|
|
|
|
// FIXME: this takes a seriously long time to execute on Windows and a
|
|
|
|
// nontrivial amount of time on Unix, we should have a better way
|
|
|
|
// of detecting whether we need to run all the submodule commands
|
|
|
|
// below.
|
|
|
|
let out = output(git_submodule().arg("status"));
|
2016-08-27 21:33:38 -05:00
|
|
|
let mut submodules = vec![];
|
|
|
|
for line in out.lines() {
|
|
|
|
// NOTE `git submodule status` output looks like this:
|
|
|
|
//
|
|
|
|
// -5066b7dcab7e700844b0e2ba71b8af9dc627a59b src/liblibc
|
2016-08-28 22:13:35 -05:00
|
|
|
// +b37ef24aa82d2be3a3cc0fe89bf82292f4ca181c src/compiler-rt (remotes/origin/..)
|
2016-08-27 21:33:38 -05:00
|
|
|
// e058ca661692a8d01f8cf9d35939dfe3105ce968 src/jemalloc (3.6.0-533-ge058ca6)
|
|
|
|
//
|
|
|
|
// The first character can be '-', '+' or ' ' and denotes the `State` of the submodule
|
|
|
|
// Right next to this character is the SHA-1 of the submodule HEAD
|
|
|
|
// And after that comes the path to the submodule
|
|
|
|
let path = Path::new(line[1..].split(' ').skip(1).next().unwrap());
|
|
|
|
let state = if line.starts_with('-') {
|
|
|
|
State::NotInitialized
|
2016-08-29 16:53:38 +08:00
|
|
|
} else if line.starts_with('+') {
|
2016-08-27 21:33:38 -05:00
|
|
|
State::OutOfSync
|
|
|
|
} else if line.starts_with(' ') {
|
|
|
|
State::MaybeDirty
|
|
|
|
} else {
|
|
|
|
panic!("unexpected git submodule state: {:?}", line.chars().next());
|
|
|
|
};
|
|
|
|
|
|
|
|
submodules.push(Submodule { path: path, state: state })
|
2016-07-05 21:58:20 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
self.run(git_submodule().arg("sync"));
|
2016-08-27 21:33:38 -05:00
|
|
|
|
|
|
|
for submodule in submodules {
|
|
|
|
// If using llvm-root then don't touch the llvm submodule.
|
|
|
|
if submodule.path.components().any(|c| c == Component::Normal("llvm".as_ref())) &&
|
2016-08-27 22:53:19 -05:00
|
|
|
self.config.target_config.get(&self.config.build)
|
|
|
|
.and_then(|c| c.llvm_config.as_ref()).is_some()
|
|
|
|
{
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
if submodule.path.components().any(|c| c == Component::Normal("jemalloc".as_ref())) &&
|
|
|
|
!self.config.use_jemalloc
|
2016-08-27 21:33:38 -05:00
|
|
|
{
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
2016-09-30 11:15:37 +02:00
|
|
|
// `submodule.path` is the relative path to a submodule (from the repository root)
|
|
|
|
// `submodule_path` is the path to a submodule from the cwd
|
|
|
|
|
|
|
|
// use `submodule.path` when e.g. executing a submodule specific command from the
|
|
|
|
// repository root
|
|
|
|
// use `submodule_path` when e.g. executing a normal git command for the submodule
|
|
|
|
// (set via `current_dir`)
|
|
|
|
let submodule_path = self.src.join(submodule.path);
|
2016-09-13 12:30:17 -07:00
|
|
|
|
2016-08-27 21:33:38 -05:00
|
|
|
match submodule.state {
|
|
|
|
State::MaybeDirty => {
|
|
|
|
// drop staged changes
|
2016-09-30 11:15:37 +02:00
|
|
|
self.run(git().current_dir(&submodule_path)
|
2016-09-13 12:30:17 -07:00
|
|
|
.args(&["reset", "--hard"]));
|
2016-08-27 21:33:38 -05:00
|
|
|
// drops unstaged changes
|
2016-09-30 11:15:37 +02:00
|
|
|
self.run(git().current_dir(&submodule_path)
|
2016-09-13 12:30:17 -07:00
|
|
|
.args(&["clean", "-fdx"]));
|
2016-08-27 21:33:38 -05:00
|
|
|
},
|
|
|
|
State::NotInitialized => {
|
|
|
|
self.run(git_submodule().arg("init").arg(submodule.path));
|
|
|
|
self.run(git_submodule().arg("update").arg(submodule.path));
|
|
|
|
},
|
|
|
|
State::OutOfSync => {
|
|
|
|
// drops submodule commits that weren't reported to the (outer) git repository
|
|
|
|
self.run(git_submodule().arg("update").arg(submodule.path));
|
2016-09-30 11:15:37 +02:00
|
|
|
self.run(git().current_dir(&submodule_path)
|
2016-09-13 12:30:17 -07:00
|
|
|
.args(&["reset", "--hard"]));
|
2016-09-30 11:15:37 +02:00
|
|
|
self.run(git().current_dir(&submodule_path)
|
2016-09-13 12:30:17 -07:00
|
|
|
.args(&["clean", "-fdx"]));
|
2016-08-27 21:33:38 -05:00
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
2016-07-05 21:58:20 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Clear out `dir` if `input` is newer.
|
|
|
|
///
|
|
|
|
/// After this executes, it will also ensure that `dir` exists.
|
|
|
|
fn clear_if_dirty(&self, dir: &Path, input: &Path) {
|
|
|
|
let stamp = dir.join(".stamp");
|
|
|
|
if mtime(&stamp) < mtime(input) {
|
|
|
|
self.verbose(&format!("Dirty - {}", dir.display()));
|
|
|
|
let _ = fs::remove_dir_all(dir);
|
2016-09-12 21:46:35 -07:00
|
|
|
} else if stamp.exists() {
|
|
|
|
return
|
2016-07-05 21:58:20 -07:00
|
|
|
}
|
|
|
|
t!(fs::create_dir_all(dir));
|
|
|
|
t!(File::create(stamp));
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Prepares an invocation of `cargo` to be run.
|
|
|
|
///
|
|
|
|
/// This will create a `Command` that represents a pending execution of
|
|
|
|
/// Cargo. This cargo will be configured to use `compiler` as the actual
|
|
|
|
/// rustc compiler, its output will be scoped by `mode`'s output directory,
|
|
|
|
/// it will pass the `--target` flag for the specified `target`, and will be
|
|
|
|
/// executing the Cargo command `cmd`.
|
|
|
|
fn cargo(&self,
|
|
|
|
compiler: &Compiler,
|
|
|
|
mode: Mode,
|
|
|
|
target: &str,
|
|
|
|
cmd: &str) -> Command {
|
|
|
|
let mut cargo = Command::new(&self.cargo);
|
|
|
|
let out_dir = self.stage_out(compiler, mode);
|
|
|
|
cargo.env("CARGO_TARGET_DIR", out_dir)
|
|
|
|
.arg(cmd)
|
|
|
|
.arg("-j").arg(self.jobs().to_string())
|
|
|
|
.arg("--target").arg(target);
|
|
|
|
|
2016-08-22 19:32:54 +00:00
|
|
|
// FIXME: Temporary fix for https://github.com/rust-lang/cargo/issues/3005
|
|
|
|
// Force cargo to output binaries with disambiguating hashes in the name
|
|
|
|
cargo.env("__CARGO_DEFAULT_LIB_METADATA", "1");
|
|
|
|
|
2016-07-05 21:58:20 -07:00
|
|
|
let stage;
|
2016-07-14 19:39:55 +02:00
|
|
|
if compiler.stage == 0 && self.local_rebuild {
|
2016-07-05 21:58:20 -07:00
|
|
|
// Assume the local-rebuild rustc already has stage1 features.
|
|
|
|
stage = 1;
|
|
|
|
} else {
|
|
|
|
stage = compiler.stage;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Customize the compiler we're running. Specify the compiler to cargo
|
|
|
|
// as our shim and then pass it some various options used to configure
|
|
|
|
// how the actual compiler itself is called.
|
|
|
|
//
|
|
|
|
// These variables are primarily all read by
|
|
|
|
// src/bootstrap/{rustc,rustdoc.rs}
|
|
|
|
cargo.env("RUSTC", self.out.join("bootstrap/debug/rustc"))
|
|
|
|
.env("RUSTC_REAL", self.compiler_path(compiler))
|
|
|
|
.env("RUSTC_STAGE", stage.to_string())
|
|
|
|
.env("RUSTC_DEBUGINFO", self.config.rust_debuginfo.to_string())
|
|
|
|
.env("RUSTC_CODEGEN_UNITS",
|
|
|
|
self.config.rust_codegen_units.to_string())
|
|
|
|
.env("RUSTC_DEBUG_ASSERTIONS",
|
|
|
|
self.config.rust_debug_assertions.to_string())
|
|
|
|
.env("RUSTC_SNAPSHOT", &self.rustc)
|
|
|
|
.env("RUSTC_SYSROOT", self.sysroot(compiler))
|
|
|
|
.env("RUSTC_LIBDIR", self.rustc_libdir(compiler))
|
|
|
|
.env("RUSTC_SNAPSHOT_LIBDIR", self.rustc_snapshot_libdir())
|
|
|
|
.env("RUSTC_RPATH", self.config.rust_rpath.to_string())
|
|
|
|
.env("RUSTDOC", self.out.join("bootstrap/debug/rustdoc"))
|
|
|
|
.env("RUSTDOC_REAL", self.rustdoc(compiler))
|
|
|
|
.env("RUSTC_FLAGS", self.rustc_flags(target).join(" "));
|
|
|
|
|
|
|
|
self.add_bootstrap_key(compiler, &mut cargo);
|
|
|
|
|
|
|
|
// Specify some various options for build scripts used throughout
|
|
|
|
// the build.
|
|
|
|
//
|
|
|
|
// FIXME: the guard against msvc shouldn't need to be here
|
|
|
|
if !target.contains("msvc") {
|
|
|
|
cargo.env(format!("CC_{}", target), self.cc(target))
|
|
|
|
.env(format!("AR_{}", target), self.ar(target).unwrap()) // only msvc is None
|
|
|
|
.env(format!("CFLAGS_{}", target), self.cflags(target).join(" "));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Environment variables *required* needed throughout the build
|
|
|
|
//
|
|
|
|
// FIXME: should update code to not require this env var
|
|
|
|
cargo.env("CFG_COMPILER_HOST_TRIPLE", target);
|
|
|
|
|
|
|
|
if self.config.verbose || self.flags.verbose {
|
|
|
|
cargo.arg("-v");
|
|
|
|
}
|
|
|
|
if self.config.rust_optimize {
|
|
|
|
cargo.arg("--release");
|
|
|
|
}
|
|
|
|
return cargo
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Get a path to the compiler specified.
|
|
|
|
fn compiler_path(&self, compiler: &Compiler) -> PathBuf {
|
|
|
|
if compiler.is_snapshot(self) {
|
|
|
|
self.rustc.clone()
|
|
|
|
} else {
|
|
|
|
self.sysroot(compiler).join("bin").join(exe("rustc", compiler.host))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Get the specified tool built by the specified compiler
|
|
|
|
fn tool(&self, compiler: &Compiler, tool: &str) -> PathBuf {
|
|
|
|
self.cargo_out(compiler, Mode::Tool, compiler.host)
|
|
|
|
.join(exe(tool, compiler.host))
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Get the `rustdoc` executable next to the specified compiler
|
|
|
|
fn rustdoc(&self, compiler: &Compiler) -> PathBuf {
|
|
|
|
let mut rustdoc = self.compiler_path(compiler);
|
|
|
|
rustdoc.pop();
|
|
|
|
rustdoc.push(exe("rustdoc", compiler.host));
|
|
|
|
return rustdoc
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Get a `Command` which is ready to run `tool` in `stage` built for
|
|
|
|
/// `host`.
|
|
|
|
fn tool_cmd(&self, compiler: &Compiler, tool: &str) -> Command {
|
|
|
|
let mut cmd = Command::new(self.tool(&compiler, tool));
|
|
|
|
let host = compiler.host;
|
|
|
|
let paths = vec![
|
|
|
|
self.cargo_out(compiler, Mode::Libstd, host).join("deps"),
|
|
|
|
self.cargo_out(compiler, Mode::Libtest, host).join("deps"),
|
|
|
|
self.cargo_out(compiler, Mode::Librustc, host).join("deps"),
|
|
|
|
self.cargo_out(compiler, Mode::Tool, host).join("deps"),
|
|
|
|
];
|
|
|
|
add_lib_path(paths, &mut cmd);
|
|
|
|
return cmd
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Get the space-separated set of activated features for the standard
|
|
|
|
/// library.
|
|
|
|
fn std_features(&self) -> String {
|
|
|
|
let mut features = String::new();
|
|
|
|
if self.config.debug_jemalloc {
|
|
|
|
features.push_str(" debug-jemalloc");
|
|
|
|
}
|
|
|
|
if self.config.use_jemalloc {
|
|
|
|
features.push_str(" jemalloc");
|
|
|
|
}
|
2016-07-26 15:21:25 -05:00
|
|
|
if self.config.backtrace {
|
|
|
|
features.push_str(" backtrace");
|
|
|
|
}
|
2016-07-05 21:58:20 -07:00
|
|
|
return features
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Get the space-separated set of activated features for the compiler.
|
|
|
|
fn rustc_features(&self) -> String {
|
|
|
|
let mut features = String::new();
|
|
|
|
if self.config.use_jemalloc {
|
|
|
|
features.push_str(" jemalloc");
|
|
|
|
}
|
|
|
|
return features
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Component directory that Cargo will produce output into (e.g.
|
|
|
|
/// release/debug)
|
|
|
|
fn cargo_dir(&self) -> &'static str {
|
|
|
|
if self.config.rust_optimize {"release"} else {"debug"}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the sysroot for the `compiler` specified that *this build system
|
|
|
|
/// generates*.
|
|
|
|
///
|
|
|
|
/// That is, the sysroot for the stage0 compiler is not what the compiler
|
|
|
|
/// thinks it is by default, but it's the same as the default for stages
|
|
|
|
/// 1-3.
|
|
|
|
fn sysroot(&self, compiler: &Compiler) -> PathBuf {
|
|
|
|
if compiler.stage == 0 {
|
|
|
|
self.out.join(compiler.host).join("stage0-sysroot")
|
|
|
|
} else {
|
|
|
|
self.out.join(compiler.host).join(format!("stage{}", compiler.stage))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the libdir where the standard library and other artifacts are
|
|
|
|
/// found for a compiler's sysroot.
|
|
|
|
fn sysroot_libdir(&self, compiler: &Compiler, target: &str) -> PathBuf {
|
|
|
|
self.sysroot(compiler).join("lib").join("rustlib")
|
|
|
|
.join(target).join("lib")
|
|
|
|
}
|
|
|
|
|
|
|
|
/// 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.
|
|
|
|
fn stage_out(&self, compiler: &Compiler, mode: Mode) -> PathBuf {
|
|
|
|
let suffix = match mode {
|
|
|
|
Mode::Libstd => "-std",
|
|
|
|
Mode::Libtest => "-test",
|
|
|
|
Mode::Tool => "-tools",
|
|
|
|
Mode::Librustc => "-rustc",
|
|
|
|
};
|
|
|
|
self.out.join(compiler.host)
|
|
|
|
.join(format!("stage{}{}", compiler.stage, suffix))
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the root output directory for all Cargo output in a given stage,
|
|
|
|
/// running a particular comipler, wehther or not we're building the
|
|
|
|
/// standard library, and targeting the specified architecture.
|
|
|
|
fn cargo_out(&self,
|
|
|
|
compiler: &Compiler,
|
|
|
|
mode: Mode,
|
|
|
|
target: &str) -> PathBuf {
|
|
|
|
self.stage_out(compiler, mode).join(target).join(self.cargo_dir())
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Root output directory for LLVM compiled for `target`
|
|
|
|
///
|
|
|
|
/// Note that if LLVM is configured externally then the directory returned
|
|
|
|
/// will likely be empty.
|
|
|
|
fn llvm_out(&self, target: &str) -> PathBuf {
|
|
|
|
self.out.join(target).join("llvm")
|
|
|
|
}
|
|
|
|
|
2016-08-06 15:54:28 +10:00
|
|
|
/// Returns true if no custom `llvm-config` is set for the specified target.
|
|
|
|
///
|
|
|
|
/// If no custom `llvm-config` was specified then Rust's llvm will be used.
|
|
|
|
fn is_rust_llvm(&self, target: &str) -> bool {
|
|
|
|
match self.config.target_config.get(target) {
|
|
|
|
Some(ref c) => c.llvm_config.is_none(),
|
|
|
|
None => true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-07-05 21:58:20 -07:00
|
|
|
/// Returns the path to `llvm-config` for the specified target.
|
|
|
|
///
|
|
|
|
/// If a custom `llvm-config` was specified for target then that's returned
|
|
|
|
/// instead.
|
|
|
|
fn llvm_config(&self, target: &str) -> PathBuf {
|
|
|
|
let target_config = self.config.target_config.get(target);
|
|
|
|
if let Some(s) = target_config.and_then(|c| c.llvm_config.as_ref()) {
|
|
|
|
s.clone()
|
|
|
|
} else {
|
|
|
|
self.llvm_out(&self.config.build).join("bin")
|
|
|
|
.join(exe("llvm-config", target))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the path to `FileCheck` binary for the specified target
|
|
|
|
fn llvm_filecheck(&self, target: &str) -> PathBuf {
|
|
|
|
let target_config = self.config.target_config.get(target);
|
|
|
|
if let Some(s) = target_config.and_then(|c| c.llvm_config.as_ref()) {
|
|
|
|
s.parent().unwrap().join(exe("FileCheck", target))
|
|
|
|
} else {
|
|
|
|
let base = self.llvm_out(&self.config.build).join("build");
|
|
|
|
let exe = exe("FileCheck", target);
|
|
|
|
if self.config.build.contains("msvc") {
|
|
|
|
base.join("Release/bin").join(exe)
|
|
|
|
} else {
|
|
|
|
base.join("bin").join(exe)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Root output directory for rust_test_helpers library compiled for
|
|
|
|
/// `target`
|
|
|
|
fn test_helpers_out(&self, target: &str) -> PathBuf {
|
|
|
|
self.out.join(target).join("rust-test-helpers")
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Adds the compiler's directory of dynamic libraries to `cmd`'s dynamic
|
|
|
|
/// library lookup path.
|
|
|
|
fn add_rustc_lib_path(&self, compiler: &Compiler, cmd: &mut Command) {
|
|
|
|
// Windows doesn't need dylib path munging because the dlls for the
|
|
|
|
// compiler live next to the compiler and the system will find them
|
|
|
|
// automatically.
|
|
|
|
if cfg!(windows) {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
add_lib_path(vec![self.rustc_libdir(compiler)], cmd);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Adds the compiler's bootstrap key to the environment of `cmd`.
|
|
|
|
fn add_bootstrap_key(&self, compiler: &Compiler, cmd: &mut Command) {
|
|
|
|
// In stage0 we're using a previously released stable compiler, so we
|
|
|
|
// use the stage0 bootstrap key. Otherwise we use our own build's
|
|
|
|
// bootstrap key.
|
2016-07-14 19:39:55 +02:00
|
|
|
let bootstrap_key = if compiler.is_snapshot(self) && !self.local_rebuild {
|
2016-07-05 21:58:20 -07:00
|
|
|
&self.bootstrap_key_stage0
|
|
|
|
} else {
|
|
|
|
&self.bootstrap_key
|
|
|
|
};
|
|
|
|
cmd.env("RUSTC_BOOTSTRAP_KEY", bootstrap_key);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the compiler's libdir where it stores the dynamic libraries that
|
|
|
|
/// it itself links against.
|
|
|
|
///
|
|
|
|
/// For example this returns `<sysroot>/lib` on Unix and `<sysroot>/bin` on
|
|
|
|
/// Windows.
|
|
|
|
fn rustc_libdir(&self, compiler: &Compiler) -> PathBuf {
|
|
|
|
if compiler.is_snapshot(self) {
|
|
|
|
self.rustc_snapshot_libdir()
|
|
|
|
} else {
|
|
|
|
self.sysroot(compiler).join(libdir(compiler.host))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the libdir of the snapshot compiler.
|
|
|
|
fn rustc_snapshot_libdir(&self) -> PathBuf {
|
|
|
|
self.rustc.parent().unwrap().parent().unwrap()
|
|
|
|
.join(libdir(&self.config.build))
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Runs a command, printing out nice contextual information if it fails.
|
|
|
|
fn run(&self, cmd: &mut Command) {
|
|
|
|
self.verbose(&format!("running: {:?}", cmd));
|
|
|
|
run_silent(cmd)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Prints a message if this build is configured in verbose mode.
|
|
|
|
fn verbose(&self, msg: &str) {
|
|
|
|
if self.flags.verbose || self.config.verbose {
|
|
|
|
println!("{}", msg);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the number of parallel jobs that have been configured for this
|
|
|
|
/// build.
|
|
|
|
fn jobs(&self) -> u32 {
|
|
|
|
self.flags.jobs.unwrap_or(num_cpus::get() as u32)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the path to the C compiler for the target specified.
|
|
|
|
fn cc(&self, target: &str) -> &Path {
|
|
|
|
self.cc[target].0.path()
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns a list of flags to pass to the C compiler for the target
|
|
|
|
/// specified.
|
|
|
|
fn cflags(&self, target: &str) -> Vec<String> {
|
|
|
|
// Filter out -O and /O (the optimization flags) that we picked up from
|
|
|
|
// gcc-rs because the build scripts will determine that for themselves.
|
|
|
|
let mut base = self.cc[target].0.args().iter()
|
|
|
|
.map(|s| s.to_string_lossy().into_owned())
|
|
|
|
.filter(|s| !s.starts_with("-O") && !s.starts_with("/O"))
|
|
|
|
.collect::<Vec<_>>();
|
|
|
|
|
|
|
|
// If we're compiling on OSX then we add a few unconditional flags
|
|
|
|
// indicating that we want libc++ (more filled out than libstdc++) and
|
|
|
|
// we want to compile for 10.7. This way we can ensure that
|
|
|
|
// LLVM/jemalloc/etc are all properly compiled.
|
|
|
|
if target.contains("apple-darwin") {
|
|
|
|
base.push("-stdlib=libc++".into());
|
|
|
|
}
|
2016-08-02 09:54:05 -07:00
|
|
|
// This is a hack, because newer binutils broke things on some vms/distros
|
|
|
|
// (i.e., linking against unknown relocs disabled by the following flag)
|
2016-08-01 23:42:15 -07:00
|
|
|
// See: https://github.com/rust-lang/rust/issues/34978
|
2016-08-10 14:35:09 -05:00
|
|
|
match target {
|
|
|
|
"i586-unknown-linux-gnu" |
|
|
|
|
"i686-unknown-linux-musl" |
|
|
|
|
"x86_64-unknown-linux-musl" => {
|
|
|
|
base.push("-Wa,-mrelax-relocations=no".into());
|
2016-08-11 11:22:39 -05:00
|
|
|
},
|
|
|
|
_ => {},
|
2016-08-01 23:42:15 -07:00
|
|
|
}
|
2016-07-05 21:58:20 -07:00
|
|
|
return base
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the path to the `ar` archive utility for the target specified.
|
|
|
|
fn ar(&self, target: &str) -> Option<&Path> {
|
|
|
|
self.cc[target].1.as_ref().map(|p| &**p)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the path to the C++ compiler for the target specified, may panic
|
|
|
|
/// if no C++ compiler was configured for the target.
|
|
|
|
fn cxx(&self, target: &str) -> &Path {
|
2016-09-12 22:43:48 -07:00
|
|
|
match self.cxx.get(target) {
|
|
|
|
Some(p) => p.path(),
|
|
|
|
None => panic!("\n\ntarget `{}` is not configured as a host,
|
|
|
|
only as a target\n\n", target),
|
|
|
|
}
|
2016-07-05 21:58:20 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns flags to pass to the compiler to generate code for `target`.
|
|
|
|
fn rustc_flags(&self, target: &str) -> Vec<String> {
|
|
|
|
// New flags should be added here with great caution!
|
|
|
|
//
|
|
|
|
// It's quite unfortunate to **require** flags to generate code for a
|
|
|
|
// target, so it should only be passed here if absolutely necessary!
|
|
|
|
// Most default configuration should be done through target specs rather
|
|
|
|
// than an entry here.
|
|
|
|
|
|
|
|
let mut base = Vec::new();
|
2016-09-08 01:58:00 +00:00
|
|
|
if target != self.config.build && !target.contains("msvc") &&
|
|
|
|
!target.contains("emscripten") {
|
2016-07-05 21:58:20 -07:00
|
|
|
base.push(format!("-Clinker={}", self.cc(target).display()));
|
|
|
|
}
|
|
|
|
return base
|
2015-11-19 15:20:12 -08:00
|
|
|
}
|
2016-09-06 21:49:02 -05:00
|
|
|
|
|
|
|
/// Returns the "musl root" for this `target`, if defined
|
|
|
|
fn musl_root(&self, target: &str) -> Option<&Path> {
|
|
|
|
self.config.target_config[target].musl_root.as_ref()
|
|
|
|
.or(self.config.musl_root.as_ref())
|
|
|
|
.map(|p| &**p)
|
|
|
|
}
|
2015-11-19 15:20:12 -08:00
|
|
|
}
|
|
|
|
|
2016-07-05 21:58:20 -07:00
|
|
|
impl<'a> Compiler<'a> {
|
|
|
|
/// Creates a new complier for the specified stage/host
|
|
|
|
fn new(stage: u32, host: &'a str) -> Compiler<'a> {
|
|
|
|
Compiler { stage: stage, host: host }
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns whether this is a snapshot compiler for `build`'s configuration
|
|
|
|
fn is_snapshot(&self, build: &Build) -> bool {
|
|
|
|
self.stage == 0 && self.host == build.config.build
|
|
|
|
}
|
2015-11-19 15:20:12 -08:00
|
|
|
}
|