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.
|
|
|
|
|
|
|
|
#![deny(warnings)]
|
|
|
|
|
2017-02-01 00:27:51 +03:00
|
|
|
extern crate filetime;
|
|
|
|
|
|
|
|
use std::fs;
|
2015-11-19 15:20:12 -08:00
|
|
|
use std::process::{Command, Stdio};
|
|
|
|
use std::path::{Path, PathBuf};
|
|
|
|
|
2017-02-01 00:27:51 +03:00
|
|
|
use filetime::FileTime;
|
|
|
|
|
|
|
|
/// 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
|
|
|
|
/// using a `Result` with `try!`, but this may change one day...
|
|
|
|
#[macro_export]
|
|
|
|
macro_rules! t {
|
|
|
|
($e:expr) => (match $e {
|
|
|
|
Ok(e) => e,
|
|
|
|
Err(e) => panic!("{} failed with {}", stringify!($e), e),
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2015-11-19 15:20:12 -08:00
|
|
|
pub fn run(cmd: &mut Command) {
|
|
|
|
println!("running: {:?}", cmd);
|
|
|
|
run_silent(cmd);
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn run_silent(cmd: &mut Command) {
|
|
|
|
let status = match cmd.status() {
|
|
|
|
Ok(status) => status,
|
2016-12-07 17:26:48 -08:00
|
|
|
Err(e) => fail(&format!("failed to execute command: {:?}\nerror: {}",
|
|
|
|
cmd, e)),
|
2015-11-19 15:20:12 -08:00
|
|
|
};
|
|
|
|
if !status.success() {
|
|
|
|
fail(&format!("command did not execute successfully: {:?}\n\
|
2016-10-16 15:41:01 +05:30
|
|
|
expected success, got: {}",
|
|
|
|
cmd,
|
|
|
|
status));
|
2015-11-19 15:20:12 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn gnu_target(target: &str) -> String {
|
|
|
|
match target {
|
|
|
|
"i686-pc-windows-msvc" => "i686-pc-win32".to_string(),
|
|
|
|
"x86_64-pc-windows-msvc" => "x86_64-pc-win32".to_string(),
|
|
|
|
"i686-pc-windows-gnu" => "i686-w64-mingw32".to_string(),
|
|
|
|
"x86_64-pc-windows-gnu" => "x86_64-w64-mingw32".to_string(),
|
|
|
|
s => s.to_string(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-06-16 08:38:06 -04:00
|
|
|
pub fn cc2ar(cc: &Path, target: &str) -> Option<PathBuf> {
|
|
|
|
if target.contains("msvc") {
|
|
|
|
None
|
|
|
|
} else if target.contains("musl") {
|
|
|
|
Some(PathBuf::from("ar"))
|
2016-12-17 11:22:41 +01:00
|
|
|
} else if target.contains("openbsd") {
|
|
|
|
Some(PathBuf::from("ar"))
|
2015-11-19 15:20:12 -08:00
|
|
|
} else {
|
2016-04-04 19:44:38 +10:00
|
|
|
let parent = cc.parent().unwrap();
|
2015-11-19 15:20:12 -08:00
|
|
|
let file = cc.file_name().unwrap().to_str().unwrap();
|
2016-04-04 19:44:38 +10:00
|
|
|
for suffix in &["gcc", "cc", "clang"] {
|
|
|
|
if let Some(idx) = file.rfind(suffix) {
|
|
|
|
let mut file = file[..idx].to_owned();
|
2016-04-04 21:14:15 +10:00
|
|
|
file.push_str("ar");
|
2016-06-16 08:38:06 -04:00
|
|
|
return Some(parent.join(&file));
|
2016-04-04 19:44:38 +10:00
|
|
|
}
|
|
|
|
}
|
2016-06-16 08:38:06 -04:00
|
|
|
Some(parent.join(file))
|
2015-11-19 15:20:12 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-12-17 20:09:23 +01:00
|
|
|
pub fn make(host: &str) -> PathBuf {
|
|
|
|
if host.contains("bitrig") || host.contains("dragonfly") ||
|
|
|
|
host.contains("freebsd") || host.contains("netbsd") ||
|
|
|
|
host.contains("openbsd") {
|
|
|
|
PathBuf::from("gmake")
|
|
|
|
} else {
|
|
|
|
PathBuf::from("make")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-11-19 15:20:12 -08:00
|
|
|
pub fn output(cmd: &mut Command) -> String {
|
|
|
|
let output = match cmd.stderr(Stdio::inherit()).output() {
|
|
|
|
Ok(status) => status,
|
2016-12-07 17:26:48 -08:00
|
|
|
Err(e) => fail(&format!("failed to execute command: {:?}\nerror: {}",
|
|
|
|
cmd, e)),
|
2015-11-19 15:20:12 -08:00
|
|
|
};
|
|
|
|
if !output.status.success() {
|
|
|
|
panic!("command did not execute successfully: {:?}\n\
|
2016-10-16 15:41:01 +05:30
|
|
|
expected success, got: {}",
|
|
|
|
cmd,
|
|
|
|
output.status);
|
2015-11-19 15:20:12 -08:00
|
|
|
}
|
|
|
|
String::from_utf8(output.stdout).unwrap()
|
|
|
|
}
|
|
|
|
|
2017-01-27 01:57:30 +03:00
|
|
|
pub fn rerun_if_changed_anything_in_dir(dir: &Path) {
|
|
|
|
let mut stack = dir.read_dir().unwrap()
|
|
|
|
.map(|e| e.unwrap())
|
|
|
|
.filter(|e| &*e.file_name() != ".git")
|
|
|
|
.collect::<Vec<_>>();
|
|
|
|
while let Some(entry) = stack.pop() {
|
|
|
|
let path = entry.path();
|
|
|
|
if entry.file_type().unwrap().is_dir() {
|
|
|
|
stack.extend(path.read_dir().unwrap().map(|e| e.unwrap()));
|
|
|
|
} else {
|
|
|
|
println!("cargo:rerun-if-changed={}", path.display());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-02-01 00:27:51 +03:00
|
|
|
/// Returns the last-modified time for `path`, or zero if it doesn't exist.
|
|
|
|
pub fn mtime(path: &Path) -> FileTime {
|
|
|
|
fs::metadata(path).map(|f| {
|
|
|
|
FileTime::from_last_modification_time(&f)
|
|
|
|
}).unwrap_or(FileTime::zero())
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns whether `dst` is up to date given that the file or files in `src`
|
|
|
|
/// are used to generate it.
|
|
|
|
///
|
|
|
|
/// Uses last-modified time checks to verify this.
|
|
|
|
pub fn up_to_date(src: &Path, dst: &Path) -> bool {
|
|
|
|
let threshold = mtime(dst);
|
|
|
|
let meta = match fs::metadata(src) {
|
|
|
|
Ok(meta) => meta,
|
|
|
|
Err(e) => panic!("source {:?} failed to get metadata: {}", src, e),
|
|
|
|
};
|
|
|
|
if meta.is_dir() {
|
|
|
|
dir_up_to_date(src, &threshold)
|
|
|
|
} else {
|
|
|
|
FileTime::from_last_modification_time(&meta) <= threshold
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn dir_up_to_date(src: &Path, threshold: &FileTime) -> bool {
|
|
|
|
t!(fs::read_dir(src)).map(|e| t!(e)).all(|e| {
|
|
|
|
let meta = t!(e.metadata());
|
|
|
|
if meta.is_dir() {
|
|
|
|
dir_up_to_date(&e.path(), threshold)
|
|
|
|
} else {
|
|
|
|
FileTime::from_last_modification_time(&meta) < *threshold
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2015-11-19 15:20:12 -08:00
|
|
|
fn fail(s: &str) -> ! {
|
|
|
|
println!("\n\n{}\n\n", s);
|
|
|
|
std::process::exit(1);
|
|
|
|
}
|