2015-11-19 17:20:12 -06:00
|
|
|
// Copyright 2016 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.
|
|
|
|
|
|
|
|
use std::collections::HashSet;
|
|
|
|
|
|
|
|
use build::{Build, Compiler};
|
|
|
|
|
|
|
|
#[derive(Hash, Eq, PartialEq, Clone, Debug)]
|
|
|
|
pub struct Step<'a> {
|
|
|
|
pub src: Source<'a>,
|
|
|
|
pub target: &'a str,
|
|
|
|
}
|
|
|
|
|
|
|
|
macro_rules! targets {
|
|
|
|
($m:ident) => {
|
|
|
|
$m! {
|
2016-02-24 23:12:42 -06:00
|
|
|
// Step representing building the stageN compiler. This is just the
|
|
|
|
// compiler executable itself, not any of the support libraries
|
2015-11-19 17:20:12 -06:00
|
|
|
(rustc, Rustc { stage: u32 }),
|
2016-02-24 23:12:42 -06:00
|
|
|
|
|
|
|
// Steps for the two main cargo builds, one for the standard library
|
|
|
|
// and one for the compiler itself. These are parameterized over the
|
|
|
|
// stage output they're going to be placed in along with the
|
|
|
|
// compiler which is producing the copy of libstd or librustc
|
2015-11-19 17:20:12 -06:00
|
|
|
(libstd, Libstd { stage: u32, compiler: Compiler<'a> }),
|
|
|
|
(librustc, Librustc { stage: u32, compiler: Compiler<'a> }),
|
2016-02-24 23:12:42 -06:00
|
|
|
|
2016-02-25 01:50:32 -06:00
|
|
|
// Links the standard library/librustc produced by the compiler
|
|
|
|
// provided into the host's directory also provided.
|
|
|
|
(libstd_link, LibstdLink {
|
|
|
|
stage: u32,
|
|
|
|
compiler: Compiler<'a>,
|
|
|
|
host: &'a str
|
|
|
|
}),
|
|
|
|
(librustc_link, LibrustcLink {
|
|
|
|
stage: u32,
|
|
|
|
compiler: Compiler<'a>,
|
|
|
|
host: &'a str
|
|
|
|
}),
|
|
|
|
|
2016-02-24 23:12:42 -06:00
|
|
|
// Steps for long-running native builds. Ideally these wouldn't
|
|
|
|
// actually exist and would be part of build scripts, but for now
|
|
|
|
// these are here.
|
|
|
|
//
|
|
|
|
// There aren't really any parameters to this, but empty structs
|
|
|
|
// with braces are unstable so we just pick something that works.
|
2015-11-19 17:20:12 -06:00
|
|
|
(llvm, Llvm { _dummy: () }),
|
|
|
|
(compiler_rt, CompilerRt { _dummy: () }),
|
2016-02-16 12:26:43 -06:00
|
|
|
(doc, Doc { stage: u32 }),
|
|
|
|
(doc_book, DocBook { stage: u32 }),
|
|
|
|
(doc_nomicon, DocNomicon { stage: u32 }),
|
|
|
|
(doc_style, DocStyle { stage: u32 }),
|
|
|
|
(doc_standalone, DocStandalone { stage: u32 }),
|
2015-11-19 17:20:12 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
macro_rules! item { ($a:item) => ($a) }
|
|
|
|
|
|
|
|
macro_rules! define_source {
|
|
|
|
($(($short:ident, $name:ident { $($args:tt)* }),)*) => {
|
|
|
|
item! {
|
|
|
|
#[derive(Hash, Eq, PartialEq, Clone, Debug)]
|
|
|
|
pub enum Source<'a> {
|
|
|
|
$($name { $($args)* }),*
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
targets!(define_source);
|
|
|
|
|
|
|
|
pub fn all(build: &Build) -> Vec<Step> {
|
|
|
|
let mut ret = Vec::new();
|
|
|
|
let mut all = HashSet::new();
|
|
|
|
for target in top_level(build) {
|
|
|
|
fill(build, &target, &mut ret, &mut all);
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
|
|
|
|
fn fill<'a>(build: &'a Build,
|
|
|
|
target: &Step<'a>,
|
|
|
|
ret: &mut Vec<Step<'a>>,
|
|
|
|
set: &mut HashSet<Step<'a>>) {
|
|
|
|
if set.insert(target.clone()) {
|
|
|
|
for dep in target.deps(build) {
|
|
|
|
fill(build, &dep, ret, set);
|
|
|
|
}
|
|
|
|
ret.push(target.clone());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn top_level(build: &Build) -> Vec<Step> {
|
|
|
|
let mut targets = Vec::new();
|
|
|
|
let stage = build.flags.stage.unwrap_or(2);
|
|
|
|
|
|
|
|
let host = Step {
|
|
|
|
src: Source::Llvm { _dummy: () },
|
|
|
|
target: build.flags.host.iter().next()
|
|
|
|
.unwrap_or(&build.config.build),
|
|
|
|
};
|
|
|
|
let target = Step {
|
|
|
|
src: Source::Llvm { _dummy: () },
|
|
|
|
target: build.flags.target.iter().next().map(|x| &x[..])
|
|
|
|
.unwrap_or(host.target)
|
|
|
|
};
|
|
|
|
|
|
|
|
add_steps(build, stage, &host, &target, &mut targets);
|
|
|
|
|
|
|
|
if targets.len() == 0 {
|
|
|
|
let t = Step {
|
|
|
|
src: Source::Llvm { _dummy: () },
|
|
|
|
target: &build.config.build,
|
|
|
|
};
|
2016-02-16 12:26:43 -06:00
|
|
|
targets.push(t.doc(stage));
|
2015-11-19 17:20:12 -06:00
|
|
|
for host in build.config.host.iter() {
|
|
|
|
if !build.flags.host.contains(host) {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
let host = t.target(host);
|
2016-02-25 01:50:32 -06:00
|
|
|
if host.target == build.config.build {
|
|
|
|
targets.push(host.librustc(stage, host.compiler(stage)));
|
|
|
|
} else {
|
|
|
|
targets.push(host.librustc_link(stage, t.compiler(stage),
|
|
|
|
host.target));
|
|
|
|
}
|
2015-11-19 17:20:12 -06:00
|
|
|
for target in build.config.target.iter() {
|
|
|
|
if !build.flags.target.contains(target) {
|
|
|
|
continue
|
|
|
|
}
|
2016-02-25 01:50:32 -06:00
|
|
|
|
|
|
|
if host.target == build.config.build {
|
|
|
|
targets.push(host.target(target)
|
|
|
|
.libstd(stage, host.compiler(stage)));
|
|
|
|
} else {
|
|
|
|
targets.push(host.target(target)
|
|
|
|
.libstd_link(stage, t.compiler(stage),
|
|
|
|
host.target));
|
|
|
|
}
|
2015-11-19 17:20:12 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return targets
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
fn add_steps<'a>(build: &'a Build,
|
|
|
|
stage: u32,
|
|
|
|
host: &Step<'a>,
|
|
|
|
target: &Step<'a>,
|
|
|
|
targets: &mut Vec<Step<'a>>) {
|
|
|
|
for step in build.flags.step.iter() {
|
2016-02-25 01:50:32 -06:00
|
|
|
let compiler = host.target(&build.config.build).compiler(stage);
|
2015-11-19 17:20:12 -06:00
|
|
|
match &step[..] {
|
|
|
|
"libstd" => targets.push(target.libstd(stage, compiler)),
|
2016-02-24 19:13:23 -06:00
|
|
|
"librustc" => targets.push(target.librustc(stage, compiler)),
|
2016-02-25 01:50:32 -06:00
|
|
|
"libstd-link" => targets.push(target.libstd_link(stage, compiler,
|
|
|
|
host.target)),
|
|
|
|
"librustc-link" => targets.push(target.librustc_link(stage, compiler,
|
|
|
|
host.target)),
|
2015-11-19 17:20:12 -06:00
|
|
|
"rustc" => targets.push(host.rustc(stage)),
|
|
|
|
"llvm" => targets.push(target.llvm(())),
|
|
|
|
"compiler-rt" => targets.push(target.compiler_rt(())),
|
2016-02-16 12:26:43 -06:00
|
|
|
"doc-style" => targets.push(host.doc_style(stage)),
|
|
|
|
"doc-standalone" => targets.push(host.doc_standalone(stage)),
|
|
|
|
"doc-nomicon" => targets.push(host.doc_nomicon(stage)),
|
|
|
|
"doc-book" => targets.push(host.doc_book(stage)),
|
|
|
|
"doc" => targets.push(host.doc(stage)),
|
2015-11-19 17:20:12 -06:00
|
|
|
_ => panic!("unknown build target: `{}`", step),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
macro_rules! constructors {
|
|
|
|
($(($short:ident, $name:ident { $($arg:ident: $t:ty),* }),)*) => {$(
|
|
|
|
fn $short(&self, $($arg: $t),*) -> Step<'a> {
|
|
|
|
Step {
|
|
|
|
src: Source::$name { $($arg: $arg),* },
|
|
|
|
target: self.target,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
)*}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a> Step<'a> {
|
|
|
|
fn compiler(&self, stage: u32) -> Compiler<'a> {
|
|
|
|
Compiler::new(stage, self.target)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn target(&self, target: &'a str) -> Step<'a> {
|
|
|
|
Step { target: target, src: self.src.clone() }
|
|
|
|
}
|
|
|
|
|
|
|
|
targets!(constructors);
|
|
|
|
|
|
|
|
pub fn deps(&self, build: &'a Build) -> Vec<Step<'a>> {
|
|
|
|
match self.src {
|
|
|
|
Source::Rustc { stage: 0 } => {
|
2016-02-24 01:00:48 -06:00
|
|
|
Vec::new()
|
2015-11-19 17:20:12 -06:00
|
|
|
}
|
|
|
|
Source::Rustc { stage } => {
|
2016-02-24 01:00:48 -06:00
|
|
|
let compiler = Compiler::new(stage - 1, &build.config.build);
|
|
|
|
vec![self.librustc(stage - 1, compiler)]
|
2015-11-19 17:20:12 -06:00
|
|
|
}
|
|
|
|
Source::Librustc { stage, compiler } => {
|
|
|
|
vec![self.libstd(stage, compiler), self.llvm(())]
|
|
|
|
}
|
|
|
|
Source::Libstd { stage: _, compiler } => {
|
|
|
|
vec![self.compiler_rt(()),
|
|
|
|
self.rustc(compiler.stage).target(compiler.host)]
|
|
|
|
}
|
2016-02-25 01:50:32 -06:00
|
|
|
Source::LibrustcLink { stage, compiler, host } => {
|
|
|
|
vec![self.librustc(stage, compiler),
|
|
|
|
self.libstd_link(stage, compiler, host)]
|
|
|
|
}
|
|
|
|
Source::LibstdLink { stage, compiler, host } => {
|
|
|
|
vec![self.libstd(stage, compiler),
|
|
|
|
self.target(host).rustc(stage)]
|
|
|
|
}
|
2015-11-19 17:20:12 -06:00
|
|
|
Source::CompilerRt { _dummy } => {
|
|
|
|
vec![self.llvm(()).target(&build.config.build)]
|
|
|
|
}
|
|
|
|
Source::Llvm { _dummy } => Vec::new(),
|
2016-02-16 12:26:43 -06:00
|
|
|
Source::DocBook { stage } |
|
|
|
|
Source::DocNomicon { stage } |
|
|
|
|
Source::DocStyle { stage } |
|
|
|
|
Source::DocStandalone { stage } => {
|
|
|
|
vec![self.rustc(stage)]
|
|
|
|
}
|
|
|
|
Source::Doc { stage } => {
|
|
|
|
vec![self.doc_book(stage), self.doc_nomicon(stage),
|
|
|
|
self.doc_style(stage), self.doc_standalone(stage)]
|
|
|
|
}
|
2015-11-19 17:20:12 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|