mirror of
https://github.com/rust-lang/rust-analyzer
synced 2025-01-28 04:45:05 +00:00
Merge #7162
7162: Introduce queries to avoid problems when performing completion for enums with many variants r=matklad a=danielframpton This change introduces two new queries to compute: 1) attributes for all variants of an enum, and 2) attributes for all fields of a variant. The purpose of this change is to avoid the current n^2 behavior when rendering completion for variants (which prevents completion for enums with large numbers of variants). Co-authored-by: Daniel Frampton <Daniel.Frampton@microsoft.com>
This commit is contained in:
commit
4bc1ed7d59
2 changed files with 55 additions and 12 deletions
|
@ -2,6 +2,7 @@
|
||||||
|
|
||||||
use std::{ops, sync::Arc};
|
use std::{ops, sync::Arc};
|
||||||
|
|
||||||
|
use arena::map::ArenaMap;
|
||||||
use base_db::CrateId;
|
use base_db::CrateId;
|
||||||
use cfg::{CfgExpr, CfgOptions};
|
use cfg::{CfgExpr, CfgOptions};
|
||||||
use either::Either;
|
use either::Either;
|
||||||
|
@ -21,7 +22,8 @@ use crate::{
|
||||||
nameres::ModuleSource,
|
nameres::ModuleSource,
|
||||||
path::{ModPath, PathKind},
|
path::{ModPath, PathKind},
|
||||||
src::HasChildSource,
|
src::HasChildSource,
|
||||||
AdtId, AttrDefId, GenericParamId, Lookup,
|
AdtId, AttrDefId, EnumId, GenericParamId, HasModule, LocalEnumVariantId, LocalFieldId, Lookup,
|
||||||
|
VariantId,
|
||||||
};
|
};
|
||||||
|
|
||||||
/// Holds documentation
|
/// Holds documentation
|
||||||
|
@ -210,16 +212,10 @@ impl Attrs {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
AttrDefId::FieldId(it) => {
|
AttrDefId::FieldId(it) => {
|
||||||
let src = it.parent.child_source(db);
|
return db.fields_attrs(it.parent)[it.local_id].clone();
|
||||||
match &src.value[it.local_id] {
|
|
||||||
Either::Left(_tuple) => RawAttrs::default(),
|
|
||||||
Either::Right(record) => RawAttrs::from_attrs_owner(db, src.with_value(record)),
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
AttrDefId::EnumVariantId(var_id) => {
|
AttrDefId::EnumVariantId(it) => {
|
||||||
let src = var_id.parent.child_source(db);
|
return db.variants_attrs(it.parent)[it.local_id].clone();
|
||||||
let src = src.as_ref().map(|it| &it[var_id.local_id]);
|
|
||||||
RawAttrs::from_attrs_owner(db, src.map(|it| it as &dyn AttrsOwner))
|
|
||||||
}
|
}
|
||||||
AttrDefId::AdtId(it) => match it {
|
AttrDefId::AdtId(it) => match it {
|
||||||
AdtId::StructId(it) => attrs_from_item_tree(it.lookup(db).id, db),
|
AdtId::StructId(it) => attrs_from_item_tree(it.lookup(db).id, db),
|
||||||
|
@ -259,6 +255,46 @@ impl Attrs {
|
||||||
raw_attrs.filter(db, def.krate(db))
|
raw_attrs.filter(db, def.krate(db))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub(crate) fn variants_attrs_query(
|
||||||
|
db: &dyn DefDatabase,
|
||||||
|
e: EnumId,
|
||||||
|
) -> Arc<ArenaMap<LocalEnumVariantId, Attrs>> {
|
||||||
|
let krate = e.lookup(db).container.module(db).krate;
|
||||||
|
let src = e.child_source(db);
|
||||||
|
let mut res = ArenaMap::default();
|
||||||
|
|
||||||
|
for (id, var) in src.value.iter() {
|
||||||
|
let attrs = RawAttrs::from_attrs_owner(db, src.with_value(var as &dyn AttrsOwner))
|
||||||
|
.filter(db, krate);
|
||||||
|
|
||||||
|
res.insert(id, attrs)
|
||||||
|
}
|
||||||
|
|
||||||
|
Arc::new(res)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) fn fields_attrs_query(
|
||||||
|
db: &dyn DefDatabase,
|
||||||
|
v: VariantId,
|
||||||
|
) -> Arc<ArenaMap<LocalFieldId, Attrs>> {
|
||||||
|
let krate = v.module(db).krate;
|
||||||
|
let src = v.child_source(db);
|
||||||
|
let mut res = ArenaMap::default();
|
||||||
|
|
||||||
|
for (id, fld) in src.value.iter() {
|
||||||
|
let attrs = match fld {
|
||||||
|
Either::Left(_tuple) => Attrs::default(),
|
||||||
|
Either::Right(record) => {
|
||||||
|
RawAttrs::from_attrs_owner(db, src.with_value(record)).filter(db, krate)
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
res.insert(id, attrs);
|
||||||
|
}
|
||||||
|
|
||||||
|
Arc::new(res)
|
||||||
|
}
|
||||||
|
|
||||||
pub fn by_key(&self, key: &'static str) -> AttrQuery<'_> {
|
pub fn by_key(&self, key: &'static str) -> AttrQuery<'_> {
|
||||||
AttrQuery { attrs: self, key }
|
AttrQuery { attrs: self, key }
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
//! Defines database & queries for name resolution.
|
//! Defines database & queries for name resolution.
|
||||||
use std::sync::Arc;
|
use std::sync::Arc;
|
||||||
|
|
||||||
|
use arena::map::ArenaMap;
|
||||||
use base_db::{salsa, CrateId, SourceDatabase, Upcast};
|
use base_db::{salsa, CrateId, SourceDatabase, Upcast};
|
||||||
use hir_expand::{db::AstDatabase, HirFileId};
|
use hir_expand::{db::AstDatabase, HirFileId};
|
||||||
use syntax::SmolStr;
|
use syntax::SmolStr;
|
||||||
|
@ -16,8 +17,8 @@ use crate::{
|
||||||
lang_item::{LangItemTarget, LangItems},
|
lang_item::{LangItemTarget, LangItems},
|
||||||
nameres::CrateDefMap,
|
nameres::CrateDefMap,
|
||||||
AttrDefId, ConstId, ConstLoc, DefWithBodyId, EnumId, EnumLoc, FunctionId, FunctionLoc,
|
AttrDefId, ConstId, ConstLoc, DefWithBodyId, EnumId, EnumLoc, FunctionId, FunctionLoc,
|
||||||
GenericDefId, ImplId, ImplLoc, StaticId, StaticLoc, StructId, StructLoc, TraitId, TraitLoc,
|
GenericDefId, ImplId, ImplLoc, LocalEnumVariantId, LocalFieldId, StaticId, StaticLoc, StructId,
|
||||||
TypeAliasId, TypeAliasLoc, UnionId, UnionLoc,
|
StructLoc, TraitId, TraitLoc, TypeAliasId, TypeAliasLoc, UnionId, UnionLoc, VariantId,
|
||||||
};
|
};
|
||||||
|
|
||||||
#[salsa::query_group(InternDatabaseStorage)]
|
#[salsa::query_group(InternDatabaseStorage)]
|
||||||
|
@ -92,6 +93,12 @@ pub trait DefDatabase: InternDatabase + AstDatabase + Upcast<dyn AstDatabase> {
|
||||||
#[salsa::invoke(GenericParams::generic_params_query)]
|
#[salsa::invoke(GenericParams::generic_params_query)]
|
||||||
fn generic_params(&self, def: GenericDefId) -> Arc<GenericParams>;
|
fn generic_params(&self, def: GenericDefId) -> Arc<GenericParams>;
|
||||||
|
|
||||||
|
#[salsa::invoke(Attrs::variants_attrs_query)]
|
||||||
|
fn variants_attrs(&self, def: EnumId) -> Arc<ArenaMap<LocalEnumVariantId, Attrs>>;
|
||||||
|
|
||||||
|
#[salsa::invoke(Attrs::fields_attrs_query)]
|
||||||
|
fn fields_attrs(&self, def: VariantId) -> Arc<ArenaMap<LocalFieldId, Attrs>>;
|
||||||
|
|
||||||
#[salsa::invoke(Attrs::attrs_query)]
|
#[salsa::invoke(Attrs::attrs_query)]
|
||||||
fn attrs(&self, def: AttrDefId) -> Attrs;
|
fn attrs(&self, def: AttrDefId) -> Attrs;
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue