use std::env;
use std::hash::{Hash, Hasher};
use std::path::{Path, PathBuf};
use std::process::Command;
pub use rustc_version::version_meta as version;
use serde_json::Value;
use serde_json;
use errors::*;
use extensions::CommandExt;
use {rustc, util};
use cargo::Root;
fn command() -> Command {
env::var_os("RUSTC")
.map(Command::new)
.unwrap_or_else(|| Command::new("rustc"))
}
pub fn targets(verbose: bool) -> Result<Vec<String>> {
command()
.args(&["--print", "target-list"])
.run_and_get_stdout(verbose)
.map(|t| t.lines().map(|l| l.to_owned()).collect())
}
pub fn sysroot(verbose: bool) -> Result<Sysroot> {
command()
.args(&["--print", "sysroot"])
.run_and_get_stdout(verbose)
.map(|l| {
Sysroot {
path: PathBuf::from(l.trim()),
}
})
}
pub struct Src {
path: PathBuf,
}
impl Src {
pub fn from_env() -> Option<Self> {
env::var_os("XARGO_RUST_SRC").map(|s| {
let path = PathBuf::from(s);
let path = path.canonicalize().unwrap_or(path);
Src { path }
})
}
pub fn path(&self) -> &Path {
&self.path
}
}
pub struct Sysroot {
path: PathBuf,
}
impl Sysroot {
pub fn path(&self) -> &Path {
&self.path
}
pub fn src(&self) -> Result<Src> {
let src = self.path().join("lib").join("rustlib").join("src");
if src.join("rust").join("src").join("libstd").join("Cargo.toml").is_file() {
return Ok(Src {
path: src.join("rust").join("src"),
});
}
if src.join("rust").join("library").join("std").join("Cargo.toml").is_file() {
return Ok(Src {
path: src.join("rust").join("library"),
});
}
Err(
"`rust-src` component not found. Run `rustup component add \
rust-src`.",
)?
}
}
#[derive(Debug)]
pub enum Target {
Builtin { triple: String },
Custom { json: PathBuf, triple: String },
}
impl Target {
pub fn new(triple: &str, root: &Root, verbose: bool) -> Result<Option<Target>> {
let triple = triple.to_owned();
if rustc::targets(verbose)?.iter().any(|t| t == &triple) {
Ok(Some(Target::Builtin { triple: triple }))
} else {
let mut json = root.path().join(&triple);
json.set_extension("json");
if json.exists() {
return Ok(Some(Target::Custom {
json: json,
triple: triple,
}));
} else {
if let Some(p) = env::var_os("RUST_TARGET_PATH") {
let mut json = PathBuf::from(p);
json.push(&triple);
json.set_extension("json");
if json.exists() {
return Ok(Some(Target::Custom {
json: json,
triple: triple,
}));
}
}
}
Ok(None)
}
}
pub fn triple(&self) -> &str {
match *self {
Target::Builtin { ref triple } => triple,
Target::Custom { ref triple, .. } => triple,
}
}
pub fn hash<H>(&self, hasher: &mut H) -> Result<()>
where
H: Hasher,
{
if let Target::Custom { ref json, .. } = *self {
serde_json::from_str::<Value>(&util::read(json)?)
.chain_err(|| format!("{} is not valid JSON", json.display()))?
.to_string()
.hash(hasher);
}
Ok(())
}
}