diff --git a/crates/ra_cli/src/analysis_bench.rs b/crates/ra_cli/src/analysis_bench.rs index 9e76bcebf4..01b96ec586 100644 --- a/crates/ra_cli/src/analysis_bench.rs +++ b/crates/ra_cli/src/analysis_bench.rs @@ -34,10 +34,11 @@ pub(crate) fn run(verbose: bool, path: &Path, op: Op) -> Result<()> { .iter() .find_map(|(source_root_id, project_root)| { if project_root.is_member() { - for (rel_path, file_id) in &db.source_root(*source_root_id).files { + for file_id in db.source_root(*source_root_id).walk() { + let rel_path = db.file_relative_path(file_id); let abs_path = rel_path.to_path(project_root.path()); if abs_path == path { - return Some(*file_id); + return Some(file_id); } } } diff --git a/crates/ra_db/src/input.rs b/crates/ra_db/src/input.rs index ad8e10c526..d1ee3c0368 100644 --- a/crates/ra_db/src/input.rs +++ b/crates/ra_db/src/input.rs @@ -5,7 +5,7 @@ /// Note that neither this module, nor any other part of the analyzer's core do /// actual IO. See `vfs` and `project_model` in the `ra_lsp_server` crate for how /// actual IO is done and lowered to input. -use relative_path::RelativePathBuf; +use relative_path::{RelativePath, RelativePathBuf}; use rustc_hash::FxHashMap; use ra_syntax::SmolStr; @@ -36,7 +36,7 @@ pub struct SourceRoot { /// Libraries are considered mostly immutable, this assumption is used to /// optimize salsa's query structure pub is_library: bool, - pub files: FxHashMap, + files: FxHashMap, } impl SourceRoot { @@ -46,6 +46,18 @@ impl SourceRoot { pub fn new_library() -> SourceRoot { SourceRoot { is_library: true, ..SourceRoot::new() } } + pub fn file_by_relative_path(&self, path: &RelativePath) -> Option { + self.files.get(path).copied() + } + pub fn insert_file(&mut self, path: RelativePathBuf, file_id: FileId) { + self.files.insert(path, file_id); + } + pub fn remove_file(&mut self, path: &RelativePath) { + self.files.remove(path); + } + pub fn walk(&self) -> impl Iterator + '_ { + self.files.values().copied() + } } /// `CrateGraph` is a bit of information which turns a set of text files into a diff --git a/crates/ra_db/src/lib.rs b/crates/ra_db/src/lib.rs index b82d1bda08..c54791b7ae 100644 --- a/crates/ra_db/src/lib.rs +++ b/crates/ra_db/src/lib.rs @@ -12,7 +12,7 @@ pub use crate::{ cancellation::Canceled, input::{CrateGraph, CrateId, Dependency, Edition, FileId, SourceRoot, SourceRootId}, }; -pub use ::salsa; +pub use salsa; pub trait CheckCanceled { /// Aborts current query if there are pending changes. @@ -93,8 +93,7 @@ pub trait SourceDatabase: CheckCanceled + std::fmt::Debug { fn source_root_crates(db: &impl SourceDatabase, id: SourceRootId) -> Arc> { let root = db.source_root(id); let graph = db.crate_graph(); - let res = - root.files.values().filter_map(|&it| graph.crate_id_for_crate_root(it)).collect::>(); + let res = root.walk().filter_map(|it| graph.crate_id_for_crate_root(it)).collect::>(); Arc::new(res) } diff --git a/crates/ra_hir/src/mock.rs b/crates/ra_hir/src/mock.rs index 77a44a275b..972f0ece59 100644 --- a/crates/ra_hir/src/mock.rs +++ b/crates/ra_hir/src/mock.rs @@ -157,7 +157,7 @@ impl MockDatabase { self.set_file_text(file_id, text); self.set_file_relative_path(file_id, rel_path.clone()); self.set_file_source_root(file_id, source_root_id); - source_root.files.insert(rel_path, file_id); + source_root.insert_file(rel_path, file_id); if is_crate_root { let mut crate_graph = CrateGraph::default(); diff --git a/crates/ra_hir/src/nameres/mod_resolution.rs b/crates/ra_hir/src/nameres/mod_resolution.rs index 94c9946ffd..de81fd422f 100644 --- a/crates/ra_hir/src/nameres/mod_resolution.rs +++ b/crates/ra_hir/src/nameres/mod_resolution.rs @@ -120,10 +120,12 @@ enum OutOfLineMode { impl OutOfLineMode { pub fn resolve(&self, source_root: Arc) -> Result { match self { - OutOfLineMode::RootOrModRs { file, directory } => match source_root.files.get(file) { - None => resolve_simple_path(source_root, directory).map_err(|_| file.clone()), - file_id => resolve_find_result(file_id, file), - }, + OutOfLineMode::RootOrModRs { file, directory } => { + match source_root.file_by_relative_path(file) { + None => resolve_simple_path(source_root, directory).map_err(|_| file.clone()), + file_id => resolve_find_result(file_id, file), + } + } OutOfLineMode::FileInDirectory(path) => resolve_simple_path(source_root, path), OutOfLineMode::WithAttributePath(path) => resolve_simple_path(source_root, path), } @@ -168,11 +170,11 @@ fn resolve_simple_path( source_root: Arc, path: &RelativePathBuf, ) -> Result { - resolve_find_result(source_root.files.get(path), path) + resolve_find_result(source_root.file_by_relative_path(path), path) } fn resolve_find_result( - file_id: Option<&FileId>, + file_id: Option, path: &RelativePathBuf, ) -> Result { match file_id { diff --git a/crates/ra_ide_api/src/change.rs b/crates/ra_ide_api/src/change.rs index 0234c572d3..89631935af 100644 --- a/crates/ra_ide_api/src/change.rs +++ b/crates/ra_ide_api/src/change.rs @@ -213,11 +213,11 @@ impl RootDatabase { durability, ); self.set_file_source_root_with_durability(add_file.file_id, root_id, durability); - source_root.files.insert(add_file.path, add_file.file_id); + source_root.insert_file(add_file.path, add_file.file_id); } for remove_file in root_change.removed { self.set_file_text_with_durability(remove_file.file_id, Default::default(), durability); - source_root.files.remove(&remove_file.path); + source_root.remove_file(&remove_file.path); } self.set_source_root_with_durability(root_id, Arc::new(source_root), durability); } diff --git a/crates/ra_ide_api/src/symbol_index.rs b/crates/ra_ide_api/src/symbol_index.rs index d4afddab4a..a5729c3688 100644 --- a/crates/ra_ide_api/src/symbol_index.rs +++ b/crates/ra_ide_api/src/symbol_index.rs @@ -87,7 +87,7 @@ pub(crate) fn world_symbols(db: &RootDatabase, query: Query) -> Vec let mut files = Vec::new(); for &root in db.local_roots().iter() { let sr = db.source_root(root); - files.extend(sr.files.values().copied()) + files.extend(sr.walk()) } let snap = Snap(db.snapshot());