Unnamed repository; edit this file 'description' to name the repository.
Diffstat (limited to 'crates/project-model/src/sysroot.rs')
-rw-r--r--crates/project-model/src/sysroot.rs126
1 files changed, 97 insertions, 29 deletions
diff --git a/crates/project-model/src/sysroot.rs b/crates/project-model/src/sysroot.rs
index 9e19a52583..b0a8f0d4a4 100644
--- a/crates/project-model/src/sysroot.rs
+++ b/crates/project-model/src/sysroot.rs
@@ -4,7 +4,7 @@
//! but we can't process `.rlib` and need source code instead. The source code
//! is typically installed with `rustup component add rust-src` command.
-use std::{env, fs, iter, ops, path::PathBuf, process::Command};
+use std::{env, fs, iter, ops, path::PathBuf, process::Command, sync::Arc};
use anyhow::{format_err, Context, Result};
use base_db::CrateName;
@@ -12,16 +12,30 @@ use itertools::Itertools;
use la_arena::{Arena, Idx};
use paths::{AbsPath, AbsPathBuf};
use rustc_hash::FxHashMap;
+use toolchain::{probe_for_binary, Tool};
use crate::{utf8_stdout, CargoConfig, CargoWorkspace, ManifestPath};
-#[derive(Debug, Clone, Eq, PartialEq)]
+#[derive(Debug, Clone)]
pub struct Sysroot {
root: AbsPathBuf,
- src_root: AbsPathBuf,
+ src_root: Option<Result<AbsPathBuf, Arc<anyhow::Error>>>,
mode: SysrootMode,
}
+impl Eq for Sysroot {}
+impl PartialEq for Sysroot {
+ fn eq(&self, other: &Self) -> bool {
+ self.root == other.root
+ && self.mode == other.mode
+ && match (&self.src_root, &other.src_root) {
+ (Some(Ok(this)), Some(Ok(other))) => this == other,
+ (None, None) | (Some(Err(_)), Some(Err(_))) => true,
+ _ => false,
+ }
+ }
+}
+
#[derive(Debug, Clone, Eq, PartialEq)]
pub(crate) enum SysrootMode {
Workspace(CargoWorkspace),
@@ -86,8 +100,8 @@ impl Sysroot {
/// Returns the sysroot "source" directory, where stdlib sources are located, like:
/// `$HOME/.rustup/toolchains/nightly-2022-07-23-x86_64-unknown-linux-gnu/lib/rustlib/src/rust/library`
- pub fn src_root(&self) -> &AbsPath {
- &self.src_root
+ pub fn src_root(&self) -> Option<&AbsPath> {
+ self.src_root.as_ref()?.as_deref().ok()
}
pub fn is_empty(&self) -> bool {
@@ -98,6 +112,11 @@ impl Sysroot {
}
pub fn loading_warning(&self) -> Option<String> {
+ let src_root = match &self.src_root {
+ None => return Some(format!("sysroot at `{}` has no library sources", self.root)),
+ Some(Ok(src_root)) => src_root,
+ Some(Err(e)) => return Some(e.to_string()),
+ };
let has_core = match &self.mode {
SysrootMode::Workspace(ws) => ws.packages().any(|p| ws[p].name == "core"),
SysrootMode::Stitched(stitched) => stitched.by_name("core").is_some(),
@@ -108,10 +127,7 @@ impl Sysroot {
} else {
" try running `rustup component add rust-src` to possible fix this"
};
- Some(format!(
- "could not find libcore in loaded sysroot at `{}`{var_note}",
- self.src_root.as_path(),
- ))
+ Some(format!("could not find libcore in loaded sysroot at `{}`{var_note}", src_root,))
} else {
None
}
@@ -140,8 +156,19 @@ impl Sysroot {
tracing::debug!("discovering sysroot for {dir}");
let sysroot_dir = discover_sysroot_dir(dir, extra_env)?;
let sysroot_src_dir =
- discover_sysroot_src_dir_or_add_component(&sysroot_dir, dir, extra_env)?;
- Ok(Sysroot::load(sysroot_dir, sysroot_src_dir, metadata))
+ discover_sysroot_src_dir_or_add_component(&sysroot_dir, dir, extra_env);
+ Ok(Sysroot::load(sysroot_dir, Some(sysroot_src_dir), metadata))
+ }
+
+ pub fn discover_no_source(
+ dir: &AbsPath,
+ extra_env: &FxHashMap<String, String>,
+ ) -> Result<Sysroot> {
+ tracing::debug!("discovering sysroot for {dir}");
+ let sysroot_dir = discover_sysroot_dir(dir, extra_env)?;
+ let sysroot_src_dir =
+ discover_sysroot_src_dir_or_add_component(&sysroot_dir, dir, extra_env);
+ Ok(Sysroot::load(sysroot_dir, Some(sysroot_src_dir), false))
}
pub fn discover_with_src_override(
@@ -152,33 +179,73 @@ impl Sysroot {
) -> Result<Sysroot> {
tracing::debug!("discovering sysroot for {current_dir}");
let sysroot_dir = discover_sysroot_dir(current_dir, extra_env)?;
- Ok(Sysroot::load(sysroot_dir, src, metadata))
+ Ok(Sysroot::load(sysroot_dir, Some(Ok(src)), metadata))
}
pub fn discover_rustc_src(&self) -> Option<ManifestPath> {
get_rustc_src(&self.root)
}
- pub fn discover_rustc(&self) -> anyhow::Result<AbsPathBuf> {
- let rustc = self.root.join("bin/rustc");
- tracing::debug!(?rustc, "checking for rustc binary at location");
- match fs::metadata(&rustc) {
- Ok(_) => Ok(rustc),
- Err(e) => Err(e).context(format!(
- "failed to discover rustc in sysroot: {:?}",
- AsRef::<std::path::Path>::as_ref(&self.root)
- )),
- }
- }
-
pub fn with_sysroot_dir(sysroot_dir: AbsPathBuf, metadata: bool) -> Result<Sysroot> {
let sysroot_src_dir = discover_sysroot_src_dir(&sysroot_dir).ok_or_else(|| {
format_err!("can't load standard library from sysroot path {sysroot_dir}")
- })?;
- Ok(Sysroot::load(sysroot_dir, sysroot_src_dir, metadata))
+ });
+ Ok(Sysroot::load(sysroot_dir, Some(sysroot_src_dir), metadata))
}
- pub fn load(sysroot_dir: AbsPathBuf, sysroot_src_dir: AbsPathBuf, metadata: bool) -> Sysroot {
+ pub fn discover_binary(&self, binary: &str) -> anyhow::Result<AbsPathBuf> {
+ toolchain::probe_for_binary(self.root.join("bin").join(binary).into())
+ .ok_or_else(|| anyhow::anyhow!("no rustc binary found in {}", self.root.join("bin")))
+ .and_then(|rustc| {
+ fs::metadata(&rustc).map(|_| AbsPathBuf::assert(rustc)).with_context(|| {
+ format!(
+ "failed to discover rustc in sysroot: {:?}",
+ AsRef::<std::path::Path>::as_ref(&self.root)
+ )
+ })
+ })
+ }
+
+ pub fn discover_tool(sysroot: Option<&Self>, tool: Tool) -> anyhow::Result<PathBuf> {
+ match sysroot {
+ Some(sysroot) => sysroot.discover_binary(tool.name()).map(Into::into),
+ None => Ok(tool.path()),
+ }
+ }
+
+ pub fn discover_proc_macro_srv(&self) -> anyhow::Result<AbsPathBuf> {
+ ["libexec", "lib"]
+ .into_iter()
+ .map(|segment| self.root().join(segment).join("rust-analyzer-proc-macro-srv"))
+ .find_map(|server_path| probe_for_binary(server_path.into()))
+ .map(AbsPathBuf::assert)
+ .ok_or_else(|| {
+ anyhow::format_err!("cannot find proc-macro server in sysroot `{}`", self.root())
+ })
+ }
+
+ pub fn load(
+ sysroot_dir: AbsPathBuf,
+ sysroot_src_dir: Option<Result<AbsPathBuf, anyhow::Error>>,
+ metadata: bool,
+ ) -> Sysroot {
+ let sysroot_src_dir = match sysroot_src_dir {
+ Some(Ok(sysroot_src_dir)) => sysroot_src_dir,
+ Some(Err(e)) => {
+ return Sysroot {
+ root: sysroot_dir,
+ src_root: Some(Err(Arc::new(e))),
+ mode: SysrootMode::Stitched(Stitched { crates: Arena::default() }),
+ }
+ }
+ None => {
+ return Sysroot {
+ root: sysroot_dir,
+ src_root: None,
+ mode: SysrootMode::Stitched(Stitched { crates: Arena::default() }),
+ }
+ }
+ };
if metadata {
let sysroot: Option<_> = (|| {
let sysroot_cargo_toml = ManifestPath::try_from(
@@ -191,6 +258,7 @@ impl Sysroot {
&sysroot_cargo_toml,
&current_dir,
&CargoConfig::default(),
+ None,
&|_| (),
)
.map_err(|e| {
@@ -274,7 +342,7 @@ impl Sysroot {
let cargo_workspace = CargoWorkspace::new(res);
Some(Sysroot {
root: sysroot_dir.clone(),
- src_root: sysroot_src_dir.clone(),
+ src_root: Some(Ok(sysroot_src_dir.clone())),
mode: SysrootMode::Workspace(cargo_workspace),
})
})();
@@ -326,7 +394,7 @@ impl Sysroot {
}
Sysroot {
root: sysroot_dir,
- src_root: sysroot_src_dir,
+ src_root: Some(Ok(sysroot_src_dir)),
mode: SysrootMode::Stitched(stitched),
}
}