rust/crates/ra_project_model/src/sysroot.rs

204 lines
5.4 KiB
Rust
Raw Normal View History

//! FIXME: write short doc here
use anyhow::{anyhow, bail, Context, Result};
2019-01-10 13:21:14 -06:00
use std::{
env,
2019-01-10 13:21:14 -06:00
path::{Path, PathBuf},
process::Command,
};
use ra_arena::{impl_arena_id, Arena, RawId};
2019-01-10 13:21:14 -06:00
2019-08-19 07:41:18 -05:00
#[derive(Default, Debug, Clone)]
2019-01-10 13:21:14 -06:00
pub struct Sysroot {
2019-01-10 13:47:05 -06:00
crates: Arena<SysrootCrate, SysrootCrateData>,
}
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
pub struct SysrootCrate(RawId);
impl_arena_id!(SysrootCrate);
#[derive(Debug, Clone)]
struct SysrootCrateData {
2019-02-09 03:51:06 -06:00
name: String,
2019-01-10 15:37:10 -06:00
root: PathBuf,
2019-01-10 13:47:05 -06:00
deps: Vec<SysrootCrate>,
2019-01-10 13:21:14 -06:00
}
impl Sysroot {
pub fn core(&self) -> Option<SysrootCrate> {
self.by_name("core")
}
2019-11-24 06:19:47 -06:00
pub fn alloc(&self) -> Option<SysrootCrate> {
self.by_name("alloc")
}
pub fn std(&self) -> Option<SysrootCrate> {
2019-01-10 14:05:22 -06:00
self.by_name("std")
}
2019-11-24 04:33:12 -06:00
pub fn proc_macro(&self) -> Option<SysrootCrate> {
self.by_name("proc_macro")
}
2019-08-06 03:54:51 -05:00
pub fn crates<'a>(&'a self) -> impl Iterator<Item = SysrootCrate> + ExactSizeIterator + 'a {
2019-01-10 15:37:10 -06:00
self.crates.iter().map(|(id, _data)| id)
}
pub fn discover(cargo_toml: &Path) -> Result<Sysroot> {
let src = get_or_install_rust_src(cargo_toml)?;
2019-01-11 12:27:25 -06:00
if !src.exists() {
Err(anyhow!(
"can't load standard library from sysroot\n\
{}\n\
(discovered via `rustc --print sysroot`)\n\
try running `rustup component add rust-src` or set `RUST_SRC_PATH`",
src.display(),
))?;
2019-01-11 12:27:25 -06:00
}
2019-01-10 13:21:14 -06:00
2019-02-08 05:49:43 -06:00
let mut sysroot = Sysroot { crates: Arena::default() };
2019-01-10 13:47:05 -06:00
for name in SYSROOT_CRATES.trim().lines() {
2019-01-10 15:37:10 -06:00
let root = src.join(format!("lib{}", name)).join("lib.rs");
if root.exists() {
2019-01-10 13:47:05 -06:00
sysroot.crates.alloc(SysrootCrateData {
name: name.into(),
2019-01-10 15:37:10 -06:00
root,
2019-01-10 13:47:05 -06:00
deps: Vec::new(),
});
}
}
2019-01-10 14:05:22 -06:00
if let Some(std) = sysroot.std() {
2019-01-10 13:47:05 -06:00
for dep in STD_DEPS.trim().lines() {
if let Some(dep) = sysroot.by_name(dep) {
sysroot.crates[std].deps.push(dep)
}
}
}
2019-06-13 14:59:50 -05:00
if let Some(alloc) = sysroot.by_name("alloc") {
2019-11-24 04:33:12 -06:00
if let Some(core) = sysroot.core() {
2019-06-13 14:59:50 -05:00
sysroot.crates[alloc].deps.push(core);
}
}
2019-01-10 13:47:05 -06:00
Ok(sysroot)
}
2019-01-10 13:21:14 -06:00
2019-01-10 13:47:05 -06:00
fn by_name(&self, name: &str) -> Option<SysrootCrate> {
2019-02-08 05:49:43 -06:00
self.crates.iter().find(|(_id, data)| data.name == name).map(|(id, _data)| id)
2019-01-10 13:21:14 -06:00
}
}
2019-01-10 13:47:05 -06:00
fn get_or_install_rust_src(cargo_toml: &Path) -> Result<PathBuf> {
fn try_find_src_path(cargo_toml: &Path) -> Result<PathBuf> {
if let Ok(path) = env::var("RUST_SRC_PATH") {
return Ok(path.into());
}
let rustc_output = Command::new("rustc")
.current_dir(cargo_toml.parent().unwrap())
.args(&["--print", "sysroot"])
.output()
.context("rustc --print sysroot failed")?;
if !rustc_output.status.success() {
match rustc_output.status.code() {
Some(code) => bail!(
"failed to locate sysroot: rustc --print sysroot exited with code {}",
code
),
None => {
bail!("failed to locate sysroot: rustc --print sysroot terminated by signal")
}
};
}
let stdout = String::from_utf8(rustc_output.stdout)?;
let sysroot_path = Path::new(stdout.trim());
Ok(sysroot_path.join("lib/rustlib/src/rust/src"))
}
fn try_install_rust_src(cargo_toml: &Path) -> Result<PathBuf> {
let rustup_output = Command::new("rustup")
.current_dir(cargo_toml.parent().unwrap())
.args(&["component", "add", "rust-src"])
.output()
.context("rustup component add rust-src failed")?;
if !rustup_output.status.success() {
match rustup_output.status.code() {
Some(code) => bail!(
"failed to install rust-src: rustup component add rust-src exited with code {}",
code
),
None => bail!(
"failed to install rust-src: rustup component add rust-src terminated by signal"
),
};
}
try_find_src_path(cargo_toml)
}
let src = try_find_src_path(cargo_toml)?;
if !src.exists() {
try_install_rust_src(cargo_toml)
} else {
Ok(src)
2020-02-17 15:33:48 -06:00
}
}
2019-01-10 15:37:10 -06:00
impl SysrootCrate {
2019-02-09 03:51:06 -06:00
pub fn name(self, sysroot: &Sysroot) -> &str {
2019-01-10 15:37:10 -06:00
&sysroot.crates[self].name
}
pub fn root(self, sysroot: &Sysroot) -> &Path {
2019-01-10 15:37:10 -06:00
sysroot.crates[self].root.as_path()
}
pub fn root_dir(self, sysroot: &Sysroot) -> &Path {
2019-01-10 15:37:10 -06:00
self.root(sysroot).parent().unwrap()
}
pub fn deps<'a>(self, sysroot: &'a Sysroot) -> impl Iterator<Item = SysrootCrate> + 'a {
2019-07-04 12:26:44 -05:00
sysroot.crates[self].deps.iter().copied()
2019-01-10 15:37:10 -06:00
}
}
2019-01-10 13:47:05 -06:00
const SYSROOT_CRATES: &str = "
std
core
alloc
collections
libc
panic_unwind
proc_macro
rustc_unicode
std_unicode
test
alloc_jemalloc
alloc_system
compiler_builtins
getopts
panic_unwind
panic_abort
rand
term
unwind
build_helper
rustc_asan
rustc_lsan
rustc_msan
rustc_tsan
syntax";
const STD_DEPS: &str = "
2019-02-03 16:23:59 -06:00
alloc
2019-01-10 13:47:05 -06:00
alloc_jemalloc
alloc_system
2019-02-03 09:35:42 -06:00
core
2019-01-10 13:47:05 -06:00
panic_abort
rand
compiler_builtins
unwind
rustc_asan
rustc_lsan
rustc_msan
rustc_tsan
build_helper";