mirror of
https://github.com/rust-lang/rust-analyzer
synced 2025-01-15 22:54:00 +00:00
Auto merge of #14490 - Veykril:crategraph-dedup, r=Veykril
internal: Switch crate graph to use an Arena instead of a hashmap
This commit is contained in:
commit
25124a84de
15 changed files with 1499 additions and 1593 deletions
1
Cargo.lock
generated
1
Cargo.lock
generated
|
@ -87,6 +87,7 @@ name = "base-db"
|
||||||
version = "0.0.0"
|
version = "0.0.0"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"cfg",
|
"cfg",
|
||||||
|
"la-arena",
|
||||||
"profile",
|
"profile",
|
||||||
"rustc-hash",
|
"rustc-hash",
|
||||||
"salsa",
|
"salsa",
|
||||||
|
|
|
@ -15,6 +15,8 @@ doctest = false
|
||||||
salsa = "0.17.0-pre.2"
|
salsa = "0.17.0-pre.2"
|
||||||
rustc-hash = "1.1.0"
|
rustc-hash = "1.1.0"
|
||||||
|
|
||||||
|
la-arena = { version = "0.3.0", path = "../../lib/la-arena" }
|
||||||
|
|
||||||
# local deps
|
# local deps
|
||||||
cfg.workspace = true
|
cfg.workspace = true
|
||||||
profile.workspace = true
|
profile.workspace = true
|
||||||
|
|
|
@ -9,8 +9,8 @@
|
||||||
use std::{fmt, mem, ops, panic::RefUnwindSafe, str::FromStr, sync::Arc};
|
use std::{fmt, mem, ops, panic::RefUnwindSafe, str::FromStr, sync::Arc};
|
||||||
|
|
||||||
use cfg::CfgOptions;
|
use cfg::CfgOptions;
|
||||||
use rustc_hash::FxHashMap;
|
use la_arena::{Arena, Idx, RawIdx};
|
||||||
use stdx::hash::{NoHashHashMap, NoHashHashSet};
|
use rustc_hash::{FxHashMap, FxHashSet};
|
||||||
use syntax::SmolStr;
|
use syntax::SmolStr;
|
||||||
use tt::token_id::Subtree;
|
use tt::token_id::Subtree;
|
||||||
use vfs::{file_set::FileSet, AbsPathBuf, AnchoredPath, FileId, VfsPath};
|
use vfs::{file_set::FileSet, AbsPathBuf, AnchoredPath, FileId, VfsPath};
|
||||||
|
@ -84,17 +84,22 @@ impl SourceRoot {
|
||||||
///
|
///
|
||||||
/// `CrateGraph` is `!Serialize` by design, see
|
/// `CrateGraph` is `!Serialize` by design, see
|
||||||
/// <https://github.com/rust-lang/rust-analyzer/blob/master/docs/dev/architecture.md#serialization>
|
/// <https://github.com/rust-lang/rust-analyzer/blob/master/docs/dev/architecture.md#serialization>
|
||||||
#[derive(Debug, Clone, Default /* Serialize, Deserialize */)]
|
#[derive(Clone, Default)]
|
||||||
pub struct CrateGraph {
|
pub struct CrateGraph {
|
||||||
arena: NoHashHashMap<CrateId, CrateData>,
|
arena: Arena<CrateData>,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Clone, Copy, PartialEq, Eq, PartialOrd, Ord, Hash)]
|
impl fmt::Debug for CrateGraph {
|
||||||
pub struct CrateId(pub u32);
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
||||||
|
f.debug_map()
|
||||||
|
.entries(self.arena.iter().map(|(id, data)| (u32::from(id.into_raw()), data)))
|
||||||
|
.finish()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
impl stdx::hash::NoHashHashable for CrateId {}
|
pub type CrateId = Idx<CrateData>;
|
||||||
|
|
||||||
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
|
#[derive(Debug, Clone, PartialEq, Eq, Hash, PartialOrd, Ord)]
|
||||||
pub struct CrateName(SmolStr);
|
pub struct CrateName(SmolStr);
|
||||||
|
|
||||||
impl CrateName {
|
impl CrateName {
|
||||||
|
@ -182,7 +187,7 @@ impl fmt::Display for LangCrateOrigin {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
|
#[derive(Debug, Clone, PartialEq, Eq, Hash, PartialOrd, Ord)]
|
||||||
pub struct CrateDisplayName {
|
pub struct CrateDisplayName {
|
||||||
// The name we use to display various paths (with `_`).
|
// The name we use to display various paths (with `_`).
|
||||||
crate_name: CrateName,
|
crate_name: CrateName,
|
||||||
|
@ -261,7 +266,7 @@ pub struct ProcMacro {
|
||||||
pub expander: Arc<dyn ProcMacroExpander>,
|
pub expander: Arc<dyn ProcMacroExpander>,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Copy, Clone)]
|
#[derive(Debug, Copy, Clone, PartialEq, Eq, Hash, PartialOrd, Ord)]
|
||||||
pub enum ReleaseChannel {
|
pub enum ReleaseChannel {
|
||||||
Stable,
|
Stable,
|
||||||
Beta,
|
Beta,
|
||||||
|
@ -287,7 +292,7 @@ impl ReleaseChannel {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Clone)]
|
#[derive(Debug, Clone, PartialEq, Eq)]
|
||||||
pub struct CrateData {
|
pub struct CrateData {
|
||||||
pub root_file_id: FileId,
|
pub root_file_id: FileId,
|
||||||
pub edition: Edition,
|
pub edition: Edition,
|
||||||
|
@ -327,7 +332,7 @@ pub struct Env {
|
||||||
entries: FxHashMap<String, String>,
|
entries: FxHashMap<String, String>,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Clone, PartialEq, Eq)]
|
#[derive(Debug, Clone, PartialEq, Eq, Hash)]
|
||||||
pub struct Dependency {
|
pub struct Dependency {
|
||||||
pub crate_id: CrateId,
|
pub crate_id: CrateId,
|
||||||
pub name: CrateName,
|
pub name: CrateName,
|
||||||
|
@ -378,10 +383,7 @@ impl CrateGraph {
|
||||||
is_proc_macro,
|
is_proc_macro,
|
||||||
channel,
|
channel,
|
||||||
};
|
};
|
||||||
let crate_id = CrateId(self.arena.len() as u32);
|
self.arena.alloc(data)
|
||||||
let prev = self.arena.insert(crate_id, data);
|
|
||||||
assert!(prev.is_none());
|
|
||||||
crate_id
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn add_dep(
|
pub fn add_dep(
|
||||||
|
@ -394,14 +396,14 @@ impl CrateGraph {
|
||||||
// Check if adding a dep from `from` to `to` creates a cycle. To figure
|
// Check if adding a dep from `from` to `to` creates a cycle. To figure
|
||||||
// that out, look for a path in the *opposite* direction, from `to` to
|
// that out, look for a path in the *opposite* direction, from `to` to
|
||||||
// `from`.
|
// `from`.
|
||||||
if let Some(path) = self.find_path(&mut NoHashHashSet::default(), dep.crate_id, from) {
|
if let Some(path) = self.find_path(&mut FxHashSet::default(), dep.crate_id, from) {
|
||||||
let path = path.into_iter().map(|it| (it, self[it].display_name.clone())).collect();
|
let path = path.into_iter().map(|it| (it, self[it].display_name.clone())).collect();
|
||||||
let err = CyclicDependenciesError { path };
|
let err = CyclicDependenciesError { path };
|
||||||
assert!(err.from().0 == from && err.to().0 == dep.crate_id);
|
assert!(err.from().0 == from && err.to().0 == dep.crate_id);
|
||||||
return Err(err);
|
return Err(err);
|
||||||
}
|
}
|
||||||
|
|
||||||
self.arena.get_mut(&from).unwrap().add_dep(dep);
|
self.arena[from].add_dep(dep);
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -410,14 +412,14 @@ impl CrateGraph {
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn iter(&self) -> impl Iterator<Item = CrateId> + '_ {
|
pub fn iter(&self) -> impl Iterator<Item = CrateId> + '_ {
|
||||||
self.arena.keys().copied()
|
self.arena.iter().map(|(idx, _)| idx)
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Returns an iterator over all transitive dependencies of the given crate,
|
/// Returns an iterator over all transitive dependencies of the given crate,
|
||||||
/// including the crate itself.
|
/// including the crate itself.
|
||||||
pub fn transitive_deps(&self, of: CrateId) -> impl Iterator<Item = CrateId> {
|
pub fn transitive_deps(&self, of: CrateId) -> impl Iterator<Item = CrateId> {
|
||||||
let mut worklist = vec![of];
|
let mut worklist = vec![of];
|
||||||
let mut deps = NoHashHashSet::default();
|
let mut deps = FxHashSet::default();
|
||||||
|
|
||||||
while let Some(krate) = worklist.pop() {
|
while let Some(krate) = worklist.pop() {
|
||||||
if !deps.insert(krate) {
|
if !deps.insert(krate) {
|
||||||
|
@ -434,11 +436,11 @@ impl CrateGraph {
|
||||||
/// including the crate itself.
|
/// including the crate itself.
|
||||||
pub fn transitive_rev_deps(&self, of: CrateId) -> impl Iterator<Item = CrateId> {
|
pub fn transitive_rev_deps(&self, of: CrateId) -> impl Iterator<Item = CrateId> {
|
||||||
let mut worklist = vec![of];
|
let mut worklist = vec![of];
|
||||||
let mut rev_deps = NoHashHashSet::default();
|
let mut rev_deps = FxHashSet::default();
|
||||||
rev_deps.insert(of);
|
rev_deps.insert(of);
|
||||||
|
|
||||||
let mut inverted_graph = NoHashHashMap::<_, Vec<_>>::default();
|
let mut inverted_graph = FxHashMap::<_, Vec<_>>::default();
|
||||||
self.arena.iter().for_each(|(&krate, data)| {
|
self.arena.iter().for_each(|(krate, data)| {
|
||||||
data.dependencies
|
data.dependencies
|
||||||
.iter()
|
.iter()
|
||||||
.for_each(|dep| inverted_graph.entry(dep.crate_id).or_default().push(krate))
|
.for_each(|dep| inverted_graph.entry(dep.crate_id).or_default().push(krate))
|
||||||
|
@ -461,9 +463,9 @@ impl CrateGraph {
|
||||||
/// come before the crate itself).
|
/// come before the crate itself).
|
||||||
pub fn crates_in_topological_order(&self) -> Vec<CrateId> {
|
pub fn crates_in_topological_order(&self) -> Vec<CrateId> {
|
||||||
let mut res = Vec::new();
|
let mut res = Vec::new();
|
||||||
let mut visited = NoHashHashSet::default();
|
let mut visited = FxHashSet::default();
|
||||||
|
|
||||||
for krate in self.arena.keys().copied() {
|
for krate in self.iter() {
|
||||||
go(self, &mut visited, &mut res, krate);
|
go(self, &mut visited, &mut res, krate);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -471,7 +473,7 @@ impl CrateGraph {
|
||||||
|
|
||||||
fn go(
|
fn go(
|
||||||
graph: &CrateGraph,
|
graph: &CrateGraph,
|
||||||
visited: &mut NoHashHashSet<CrateId>,
|
visited: &mut FxHashSet<CrateId>,
|
||||||
res: &mut Vec<CrateId>,
|
res: &mut Vec<CrateId>,
|
||||||
source: CrateId,
|
source: CrateId,
|
||||||
) {
|
) {
|
||||||
|
@ -487,7 +489,7 @@ impl CrateGraph {
|
||||||
|
|
||||||
// FIXME: this only finds one crate with the given root; we could have multiple
|
// FIXME: this only finds one crate with the given root; we could have multiple
|
||||||
pub fn crate_id_for_crate_root(&self, file_id: FileId) -> Option<CrateId> {
|
pub fn crate_id_for_crate_root(&self, file_id: FileId) -> Option<CrateId> {
|
||||||
let (&crate_id, _) =
|
let (crate_id, _) =
|
||||||
self.arena.iter().find(|(_crate_id, data)| data.root_file_id == file_id)?;
|
self.arena.iter().find(|(_crate_id, data)| data.root_file_id == file_id)?;
|
||||||
Some(crate_id)
|
Some(crate_id)
|
||||||
}
|
}
|
||||||
|
@ -499,24 +501,26 @@ impl CrateGraph {
|
||||||
/// amount.
|
/// amount.
|
||||||
pub fn extend(&mut self, other: CrateGraph, proc_macros: &mut ProcMacroPaths) -> u32 {
|
pub fn extend(&mut self, other: CrateGraph, proc_macros: &mut ProcMacroPaths) -> u32 {
|
||||||
let start = self.arena.len() as u32;
|
let start = self.arena.len() as u32;
|
||||||
self.arena.extend(other.arena.into_iter().map(|(id, mut data)| {
|
self.arena.extend(other.arena.into_iter().map(|(_, mut data)| {
|
||||||
let new_id = id.shift(start);
|
|
||||||
for dep in &mut data.dependencies {
|
for dep in &mut data.dependencies {
|
||||||
dep.crate_id = dep.crate_id.shift(start);
|
dep.crate_id =
|
||||||
|
CrateId::from_raw(RawIdx::from(u32::from(dep.crate_id.into_raw()) + start));
|
||||||
}
|
}
|
||||||
(new_id, data)
|
data
|
||||||
}));
|
}));
|
||||||
|
|
||||||
*proc_macros = mem::take(proc_macros)
|
*proc_macros = mem::take(proc_macros)
|
||||||
.into_iter()
|
.into_iter()
|
||||||
.map(|(id, macros)| (id.shift(start), macros))
|
.map(|(id, macros)| {
|
||||||
|
(CrateId::from_raw(RawIdx::from(u32::from(id.into_raw()) + start)), macros)
|
||||||
|
})
|
||||||
.collect();
|
.collect();
|
||||||
start
|
start
|
||||||
}
|
}
|
||||||
|
|
||||||
fn find_path(
|
fn find_path(
|
||||||
&self,
|
&self,
|
||||||
visited: &mut NoHashHashSet<CrateId>,
|
visited: &mut FxHashSet<CrateId>,
|
||||||
from: CrateId,
|
from: CrateId,
|
||||||
to: CrateId,
|
to: CrateId,
|
||||||
) -> Option<Vec<CrateId>> {
|
) -> Option<Vec<CrateId>> {
|
||||||
|
@ -546,10 +550,8 @@ impl CrateGraph {
|
||||||
let std = self.hacky_find_crate("std");
|
let std = self.hacky_find_crate("std");
|
||||||
match (cfg_if, std) {
|
match (cfg_if, std) {
|
||||||
(Some(cfg_if), Some(std)) => {
|
(Some(cfg_if), Some(std)) => {
|
||||||
self.arena.get_mut(&cfg_if).unwrap().dependencies.clear();
|
self.arena[cfg_if].dependencies.clear();
|
||||||
self.arena
|
self.arena[std]
|
||||||
.get_mut(&std)
|
|
||||||
.unwrap()
|
|
||||||
.dependencies
|
.dependencies
|
||||||
.push(Dependency::new(CrateName::new("cfg_if").unwrap(), cfg_if));
|
.push(Dependency::new(CrateName::new("cfg_if").unwrap(), cfg_if));
|
||||||
true
|
true
|
||||||
|
@ -566,13 +568,7 @@ impl CrateGraph {
|
||||||
impl ops::Index<CrateId> for CrateGraph {
|
impl ops::Index<CrateId> for CrateGraph {
|
||||||
type Output = CrateData;
|
type Output = CrateData;
|
||||||
fn index(&self, crate_id: CrateId) -> &CrateData {
|
fn index(&self, crate_id: CrateId) -> &CrateData {
|
||||||
&self.arena[&crate_id]
|
&self.arena[crate_id]
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl CrateId {
|
|
||||||
fn shift(self, amount: u32) -> CrateId {
|
|
||||||
CrateId(self.0 + amount)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -8,7 +8,7 @@ pub mod fixture;
|
||||||
|
|
||||||
use std::{panic, sync::Arc};
|
use std::{panic, sync::Arc};
|
||||||
|
|
||||||
use stdx::hash::NoHashHashSet;
|
use rustc_hash::FxHashSet;
|
||||||
use syntax::{ast, Parse, SourceFile, TextRange, TextSize};
|
use syntax::{ast, Parse, SourceFile, TextRange, TextSize};
|
||||||
|
|
||||||
pub use crate::{
|
pub use crate::{
|
||||||
|
@ -59,7 +59,7 @@ pub trait FileLoader {
|
||||||
/// Text of the file.
|
/// Text of the file.
|
||||||
fn file_text(&self, file_id: FileId) -> Arc<String>;
|
fn file_text(&self, file_id: FileId) -> Arc<String>;
|
||||||
fn resolve_path(&self, path: AnchoredPath<'_>) -> Option<FileId>;
|
fn resolve_path(&self, path: AnchoredPath<'_>) -> Option<FileId>;
|
||||||
fn relevant_crates(&self, file_id: FileId) -> Arc<NoHashHashSet<CrateId>>;
|
fn relevant_crates(&self, file_id: FileId) -> Arc<FxHashSet<CrateId>>;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Database which stores all significant input facts: source code and project
|
/// Database which stores all significant input facts: source code and project
|
||||||
|
@ -99,10 +99,10 @@ pub trait SourceDatabaseExt: SourceDatabase {
|
||||||
#[salsa::input]
|
#[salsa::input]
|
||||||
fn source_root(&self, id: SourceRootId) -> Arc<SourceRoot>;
|
fn source_root(&self, id: SourceRootId) -> Arc<SourceRoot>;
|
||||||
|
|
||||||
fn source_root_crates(&self, id: SourceRootId) -> Arc<NoHashHashSet<CrateId>>;
|
fn source_root_crates(&self, id: SourceRootId) -> Arc<FxHashSet<CrateId>>;
|
||||||
}
|
}
|
||||||
|
|
||||||
fn source_root_crates(db: &dyn SourceDatabaseExt, id: SourceRootId) -> Arc<NoHashHashSet<CrateId>> {
|
fn source_root_crates(db: &dyn SourceDatabaseExt, id: SourceRootId) -> Arc<FxHashSet<CrateId>> {
|
||||||
let graph = db.crate_graph();
|
let graph = db.crate_graph();
|
||||||
let res = graph
|
let res = graph
|
||||||
.iter()
|
.iter()
|
||||||
|
@ -128,7 +128,7 @@ impl<T: SourceDatabaseExt> FileLoader for FileLoaderDelegate<&'_ T> {
|
||||||
source_root.resolve_path(path)
|
source_root.resolve_path(path)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn relevant_crates(&self, file_id: FileId) -> Arc<NoHashHashSet<CrateId>> {
|
fn relevant_crates(&self, file_id: FileId) -> Arc<FxHashSet<CrateId>> {
|
||||||
let _p = profile::span("relevant_crates");
|
let _p = profile::span("relevant_crates");
|
||||||
let source_root = self.0.file_source_root(file_id);
|
let source_root = self.0.file_source_root(file_id);
|
||||||
self.0.source_root_crates(source_root)
|
self.0.source_root_crates(source_root)
|
||||||
|
|
|
@ -148,8 +148,8 @@ fn f() {
|
||||||
}
|
}
|
||||||
"#,
|
"#,
|
||||||
expect![[r#"
|
expect![[r#"
|
||||||
BlockId(1) in ModuleId { krate: CrateId(0), block: Some(BlockId(0)), local_id: Idx::<ModuleData>(1) }
|
BlockId(1) in ModuleId { krate: Idx::<CrateData>(0), block: Some(BlockId(0)), local_id: Idx::<ModuleData>(1) }
|
||||||
BlockId(0) in ModuleId { krate: CrateId(0), block: None, local_id: Idx::<ModuleData>(0) }
|
BlockId(0) in ModuleId { krate: Idx::<CrateData>(0), block: None, local_id: Idx::<ModuleData>(0) }
|
||||||
crate scope
|
crate scope
|
||||||
"#]],
|
"#]],
|
||||||
);
|
);
|
||||||
|
|
|
@ -11,7 +11,7 @@ use base_db::{
|
||||||
Upcast,
|
Upcast,
|
||||||
};
|
};
|
||||||
use hir_expand::{db::ExpandDatabase, InFile};
|
use hir_expand::{db::ExpandDatabase, InFile};
|
||||||
use stdx::hash::NoHashHashSet;
|
use rustc_hash::FxHashSet;
|
||||||
use syntax::{algo, ast, AstNode};
|
use syntax::{algo, ast, AstNode};
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
|
@ -77,7 +77,7 @@ impl FileLoader for TestDB {
|
||||||
fn resolve_path(&self, path: AnchoredPath<'_>) -> Option<FileId> {
|
fn resolve_path(&self, path: AnchoredPath<'_>) -> Option<FileId> {
|
||||||
FileLoaderDelegate(self).resolve_path(path)
|
FileLoaderDelegate(self).resolve_path(path)
|
||||||
}
|
}
|
||||||
fn relevant_crates(&self, file_id: FileId) -> Arc<NoHashHashSet<CrateId>> {
|
fn relevant_crates(&self, file_id: FileId) -> Arc<FxHashSet<CrateId>> {
|
||||||
FileLoaderDelegate(self).relevant_crates(file_id)
|
FileLoaderDelegate(self).relevant_crates(file_id)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -11,7 +11,8 @@ use base_db::{
|
||||||
};
|
};
|
||||||
use hir_def::{db::DefDatabase, ModuleId};
|
use hir_def::{db::DefDatabase, ModuleId};
|
||||||
use hir_expand::db::ExpandDatabase;
|
use hir_expand::db::ExpandDatabase;
|
||||||
use stdx::hash::{NoHashHashMap, NoHashHashSet};
|
use rustc_hash::FxHashSet;
|
||||||
|
use stdx::hash::NoHashHashMap;
|
||||||
use syntax::TextRange;
|
use syntax::TextRange;
|
||||||
use test_utils::extract_annotations;
|
use test_utils::extract_annotations;
|
||||||
|
|
||||||
|
@ -81,7 +82,7 @@ impl FileLoader for TestDB {
|
||||||
fn resolve_path(&self, path: AnchoredPath<'_>) -> Option<FileId> {
|
fn resolve_path(&self, path: AnchoredPath<'_>) -> Option<FileId> {
|
||||||
FileLoaderDelegate(self).resolve_path(path)
|
FileLoaderDelegate(self).resolve_path(path)
|
||||||
}
|
}
|
||||||
fn relevant_crates(&self, file_id: FileId) -> Arc<NoHashHashSet<CrateId>> {
|
fn relevant_crates(&self, file_id: FileId) -> Arc<FxHashSet<CrateId>> {
|
||||||
FileLoaderDelegate(self).relevant_crates(file_id)
|
FileLoaderDelegate(self).relevant_crates(file_id)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -53,7 +53,6 @@ use hir::{
|
||||||
db::{DefDatabase, ExpandDatabase, HirDatabase},
|
db::{DefDatabase, ExpandDatabase, HirDatabase},
|
||||||
symbols::FileSymbolKind,
|
symbols::FileSymbolKind,
|
||||||
};
|
};
|
||||||
use stdx::hash::NoHashHashSet;
|
|
||||||
|
|
||||||
use crate::{line_index::LineIndex, symbol_index::SymbolsDatabase};
|
use crate::{line_index::LineIndex, symbol_index::SymbolsDatabase};
|
||||||
pub use rustc_hash::{FxHashMap, FxHashSet, FxHasher};
|
pub use rustc_hash::{FxHashMap, FxHashSet, FxHasher};
|
||||||
|
@ -120,7 +119,7 @@ impl FileLoader for RootDatabase {
|
||||||
fn resolve_path(&self, path: AnchoredPath<'_>) -> Option<FileId> {
|
fn resolve_path(&self, path: AnchoredPath<'_>) -> Option<FileId> {
|
||||||
FileLoaderDelegate(self).resolve_path(path)
|
FileLoaderDelegate(self).resolve_path(path)
|
||||||
}
|
}
|
||||||
fn relevant_crates(&self, file_id: FileId) -> Arc<NoHashHashSet<CrateId>> {
|
fn relevant_crates(&self, file_id: FileId) -> Arc<FxHashSet<CrateId>> {
|
||||||
FileLoaderDelegate(self).relevant_crates(file_id)
|
FileLoaderDelegate(self).relevant_crates(file_id)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -2,9 +2,7 @@
|
||||||
(
|
(
|
||||||
Module {
|
Module {
|
||||||
id: ModuleId {
|
id: ModuleId {
|
||||||
krate: CrateId(
|
krate: Idx::<CrateData>(0),
|
||||||
0,
|
|
||||||
),
|
|
||||||
block: None,
|
block: None,
|
||||||
local_id: Idx::<ModuleData>(0),
|
local_id: Idx::<ModuleData>(0),
|
||||||
},
|
},
|
||||||
|
@ -381,9 +379,7 @@
|
||||||
(
|
(
|
||||||
Module {
|
Module {
|
||||||
id: ModuleId {
|
id: ModuleId {
|
||||||
krate: CrateId(
|
krate: Idx::<CrateData>(0),
|
||||||
0,
|
|
||||||
),
|
|
||||||
block: None,
|
block: None,
|
||||||
local_id: Idx::<ModuleData>(1),
|
local_id: Idx::<ModuleData>(1),
|
||||||
},
|
},
|
||||||
|
@ -412,9 +408,7 @@
|
||||||
(
|
(
|
||||||
Module {
|
Module {
|
||||||
id: ModuleId {
|
id: ModuleId {
|
||||||
krate: CrateId(
|
krate: Idx::<CrateData>(0),
|
||||||
0,
|
|
||||||
),
|
|
||||||
block: None,
|
block: None,
|
||||||
local_id: Idx::<ModuleData>(2),
|
local_id: Idx::<ModuleData>(2),
|
||||||
},
|
},
|
||||||
|
|
|
@ -12,9 +12,8 @@ use ide_db::{
|
||||||
salsa::{Database, ParallelDatabase, Snapshot},
|
salsa::{Database, ParallelDatabase, Snapshot},
|
||||||
Cancelled, CrateGraph, CrateId, SourceDatabase, SourceDatabaseExt,
|
Cancelled, CrateGraph, CrateId, SourceDatabase, SourceDatabaseExt,
|
||||||
},
|
},
|
||||||
FxIndexMap,
|
FxHashSet, FxIndexMap,
|
||||||
};
|
};
|
||||||
use stdx::hash::NoHashHashSet;
|
|
||||||
|
|
||||||
use crate::RootDatabase;
|
use crate::RootDatabase;
|
||||||
|
|
||||||
|
@ -142,7 +141,7 @@ pub(crate) fn parallel_prime_caches(
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn compute_crates_to_prime(db: &RootDatabase, graph: &CrateGraph) -> NoHashHashSet<CrateId> {
|
fn compute_crates_to_prime(db: &RootDatabase, graph: &CrateGraph) -> FxHashSet<CrateId> {
|
||||||
// We're only interested in the workspace crates and the `ImportMap`s of their direct
|
// We're only interested in the workspace crates and the `ImportMap`s of their direct
|
||||||
// dependencies, though in practice the latter also compute the `DefMap`s.
|
// dependencies, though in practice the latter also compute the `DefMap`s.
|
||||||
// We don't prime transitive dependencies because they're generally not visible in
|
// We don't prime transitive dependencies because they're generally not visible in
|
||||||
|
|
|
@ -3,9 +3,8 @@ use std::sync::Arc;
|
||||||
use dot::{Id, LabelText};
|
use dot::{Id, LabelText};
|
||||||
use ide_db::{
|
use ide_db::{
|
||||||
base_db::{CrateGraph, CrateId, Dependency, SourceDatabase, SourceDatabaseExt},
|
base_db::{CrateGraph, CrateId, Dependency, SourceDatabase, SourceDatabaseExt},
|
||||||
RootDatabase,
|
FxHashSet, RootDatabase,
|
||||||
};
|
};
|
||||||
use stdx::hash::NoHashHashSet;
|
|
||||||
|
|
||||||
// Feature: View Crate Graph
|
// Feature: View Crate Graph
|
||||||
//
|
//
|
||||||
|
@ -42,7 +41,7 @@ pub(crate) fn view_crate_graph(db: &RootDatabase, full: bool) -> Result<String,
|
||||||
|
|
||||||
struct DotCrateGraph {
|
struct DotCrateGraph {
|
||||||
graph: Arc<CrateGraph>,
|
graph: Arc<CrateGraph>,
|
||||||
crates_to_render: NoHashHashSet<CrateId>,
|
crates_to_render: FxHashSet<CrateId>,
|
||||||
}
|
}
|
||||||
|
|
||||||
type Edge<'a> = (CrateId, &'a Dependency);
|
type Edge<'a> = (CrateId, &'a Dependency);
|
||||||
|
@ -80,7 +79,7 @@ impl<'a> dot::Labeller<'a, CrateId, Edge<'a>> for DotCrateGraph {
|
||||||
}
|
}
|
||||||
|
|
||||||
fn node_id(&'a self, n: &CrateId) -> Id<'a> {
|
fn node_id(&'a self, n: &CrateId) -> Id<'a> {
|
||||||
Id::new(format!("_{}", n.0)).unwrap()
|
Id::new(format!("_{}", u32::from(n.into_raw()))).unwrap()
|
||||||
}
|
}
|
||||||
|
|
||||||
fn node_shape(&'a self, _node: &CrateId) -> Option<LabelText<'a>> {
|
fn node_shape(&'a self, _node: &CrateId) -> Option<LabelText<'a>> {
|
||||||
|
|
|
@ -52,6 +52,7 @@
|
||||||
use std::path::PathBuf;
|
use std::path::PathBuf;
|
||||||
|
|
||||||
use base_db::{CrateDisplayName, CrateId, CrateName, Dependency, Edition};
|
use base_db::{CrateDisplayName, CrateId, CrateName, Dependency, Edition};
|
||||||
|
use la_arena::RawIdx;
|
||||||
use paths::{AbsPath, AbsPathBuf};
|
use paths::{AbsPath, AbsPathBuf};
|
||||||
use rustc_hash::FxHashMap;
|
use rustc_hash::FxHashMap;
|
||||||
use serde::{de, Deserialize};
|
use serde::{de, Deserialize};
|
||||||
|
@ -135,7 +136,10 @@ impl ProjectJson {
|
||||||
.deps
|
.deps
|
||||||
.into_iter()
|
.into_iter()
|
||||||
.map(|dep_data| {
|
.map(|dep_data| {
|
||||||
Dependency::new(dep_data.name, CrateId(dep_data.krate as u32))
|
Dependency::new(
|
||||||
|
dep_data.name,
|
||||||
|
CrateId::from_raw(RawIdx::from(dep_data.krate as u32)),
|
||||||
|
)
|
||||||
})
|
})
|
||||||
.collect::<Vec<_>>(),
|
.collect::<Vec<_>>(),
|
||||||
cfg: crate_data.cfg,
|
cfg: crate_data.cfg,
|
||||||
|
@ -162,7 +166,10 @@ impl ProjectJson {
|
||||||
|
|
||||||
/// Returns an iterator over the crates in the project.
|
/// Returns an iterator over the crates in the project.
|
||||||
pub fn crates(&self) -> impl Iterator<Item = (CrateId, &Crate)> + '_ {
|
pub fn crates(&self) -> impl Iterator<Item = (CrateId, &Crate)> + '_ {
|
||||||
self.crates.iter().enumerate().map(|(idx, krate)| (CrateId(idx as u32), krate))
|
self.crates
|
||||||
|
.iter()
|
||||||
|
.enumerate()
|
||||||
|
.map(|(idx, krate)| (CrateId::from_raw(RawIdx::from(idx as u32)), krate))
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Returns the path to the project's root folder.
|
/// Returns the path to the project's root folder.
|
||||||
|
|
File diff suppressed because it is too large
Load diff
|
@ -13,7 +13,7 @@ use cfg::{CfgDiff, CfgOptions};
|
||||||
use paths::{AbsPath, AbsPathBuf};
|
use paths::{AbsPath, AbsPathBuf};
|
||||||
use rustc_hash::{FxHashMap, FxHashSet};
|
use rustc_hash::{FxHashMap, FxHashSet};
|
||||||
use semver::Version;
|
use semver::Version;
|
||||||
use stdx::{always, hash::NoHashHashMap};
|
use stdx::always;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
build_scripts::BuildScriptOutput,
|
build_scripts::BuildScriptOutput,
|
||||||
|
@ -732,7 +732,7 @@ fn project_json_to_crate_graph(
|
||||||
});
|
});
|
||||||
|
|
||||||
let mut cfg_cache: FxHashMap<&str, Vec<CfgFlag>> = FxHashMap::default();
|
let mut cfg_cache: FxHashMap<&str, Vec<CfgFlag>> = FxHashMap::default();
|
||||||
let crates: NoHashHashMap<CrateId, CrateId> = project
|
let crates: FxHashMap<CrateId, CrateId> = project
|
||||||
.crates()
|
.crates()
|
||||||
.filter_map(|(crate_id, krate)| Some((crate_id, krate, load(&krate.root_module)?)))
|
.filter_map(|(crate_id, krate)| Some((crate_id, krate, load(&krate.root_module)?)))
|
||||||
.map(
|
.map(
|
||||||
|
|
|
@ -4,8 +4,9 @@
|
||||||
#![warn(missing_docs)]
|
#![warn(missing_docs)]
|
||||||
|
|
||||||
use std::{
|
use std::{
|
||||||
fmt,
|
cmp, fmt,
|
||||||
hash::{Hash, Hasher},
|
hash::{Hash, Hasher},
|
||||||
|
iter::Enumerate,
|
||||||
marker::PhantomData,
|
marker::PhantomData,
|
||||||
ops::{Index, IndexMut, Range, RangeInclusive},
|
ops::{Index, IndexMut, Range, RangeInclusive},
|
||||||
};
|
};
|
||||||
|
@ -47,6 +48,18 @@ pub struct Idx<T> {
|
||||||
_ty: PhantomData<fn() -> T>,
|
_ty: PhantomData<fn() -> T>,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl<T> Ord for Idx<T> {
|
||||||
|
fn cmp(&self, other: &Self) -> cmp::Ordering {
|
||||||
|
self.raw.cmp(&other.raw)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T> PartialOrd for Idx<T> {
|
||||||
|
fn partial_cmp(&self, other: &Self) -> Option<cmp::Ordering> {
|
||||||
|
self.raw.partial_cmp(&other.raw)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
impl<T> Clone for Idx<T> {
|
impl<T> Clone for Idx<T> {
|
||||||
fn clone(&self) -> Self {
|
fn clone(&self) -> Self {
|
||||||
*self
|
*self
|
||||||
|
@ -335,7 +348,7 @@ impl<T> Arena<T> {
|
||||||
/// assert_eq!(iterator.next(), Some(&40));
|
/// assert_eq!(iterator.next(), Some(&40));
|
||||||
/// assert_eq!(iterator.next(), Some(&60));
|
/// assert_eq!(iterator.next(), Some(&60));
|
||||||
/// ```
|
/// ```
|
||||||
pub fn values(&mut self) -> impl Iterator<Item = &T> + ExactSizeIterator + DoubleEndedIterator {
|
pub fn values(&self) -> impl Iterator<Item = &T> + ExactSizeIterator + DoubleEndedIterator {
|
||||||
self.data.iter()
|
self.data.iter()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -410,3 +423,32 @@ impl<T> FromIterator<T> for Arena<T> {
|
||||||
Arena { data: Vec::from_iter(iter) }
|
Arena { data: Vec::from_iter(iter) }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// An iterator over the arena’s elements.
|
||||||
|
pub struct IntoIter<T>(Enumerate<<Vec<T> as IntoIterator>::IntoIter>);
|
||||||
|
|
||||||
|
impl<T> Iterator for IntoIter<T> {
|
||||||
|
type Item = (Idx<T>, T);
|
||||||
|
|
||||||
|
fn next(&mut self) -> Option<Self::Item> {
|
||||||
|
self.0.next().map(|(idx, value)| (Idx::from_raw(RawIdx(idx as u32)), value))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T> IntoIterator for Arena<T> {
|
||||||
|
type Item = (Idx<T>, T);
|
||||||
|
|
||||||
|
type IntoIter = IntoIter<T>;
|
||||||
|
|
||||||
|
fn into_iter(self) -> Self::IntoIter {
|
||||||
|
IntoIter(self.data.into_iter().enumerate())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<T> Extend<T> for Arena<T> {
|
||||||
|
fn extend<II: IntoIterator<Item = T>>(&mut self, iter: II) {
|
||||||
|
for t in iter {
|
||||||
|
self.alloc(t);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in a new issue