use std::{
env,
hash::{Hash, Hasher},
path::{Path, PathBuf},
process::Command,
sync::OnceLock,
};
pub use rustc_version::version_meta as version;
use serde_json::{self, Value};
use crate::{cargo::Root, errors::*, extensions::CommandExt, rustc, util};
fn command() -> Command {
env::var_os("RUSTC")
.map(Command::new)
.unwrap_or_else(|| Command::new("rustc"))
}
static TARGETS: OnceLock<Vec<String>> = OnceLock::new();
pub fn targets(verbose: bool) -> Result<Vec<String>> {
if let Some(cached) = TARGETS.get() {
return Ok(cached.clone());
}
let list: Vec<String> = command()
.args(["--print", "target-list"])
.run_and_get_stdout(verbose)?
.lines()
.map(|l| l.to_owned())
.collect();
let _ = TARGETS.set(list.clone());
Ok(list)
}
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("ZARGO_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 }))
} else {
let mut json = root.path().join(&triple);
json.set_extension("json");
if json.exists() {
return Ok(Some(Target::Custom { json, 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, 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(())
}
}