Store GenericParams in arena

This commit is contained in:
Aleksey Kladov 2019-12-06 17:35:05 +01:00
parent 35fc983dd9
commit 30fefcc08c
4 changed files with 25 additions and 14 deletions

View file

@ -5,18 +5,19 @@
use std::sync::Arc; use std::sync::Arc;
use hir_expand::name::{self, AsName, Name}; use hir_expand::name::{self, AsName, Name};
use ra_arena::Arena;
use ra_syntax::ast::{self, NameOwner, TypeBoundsOwner, TypeParamsOwner}; use ra_syntax::ast::{self, NameOwner, TypeBoundsOwner, TypeParamsOwner};
use crate::{ use crate::{
db::DefDatabase, db::DefDatabase,
src::HasSource, src::HasSource,
type_ref::{TypeBound, TypeRef}, type_ref::{TypeBound, TypeRef},
AdtId, AstItemDef, ContainerId, GenericDefId, Lookup, AdtId, AstItemDef, ContainerId, GenericDefId, LocalGenericParamId, Lookup,
}; };
/// Data about a generic parameter (to a function, struct, impl, ...). /// Data about a generic parameter (to a function, struct, impl, ...).
#[derive(Clone, PartialEq, Eq, Debug)] #[derive(Clone, PartialEq, Eq, Debug)]
pub struct GenericParam { pub struct GenericParamData {
// FIXME: give generic params proper IDs // FIXME: give generic params proper IDs
pub idx: u32, pub idx: u32,
pub name: Name, pub name: Name,
@ -27,7 +28,7 @@ pub struct GenericParam {
#[derive(Clone, PartialEq, Eq, Debug)] #[derive(Clone, PartialEq, Eq, Debug)]
pub struct GenericParams { pub struct GenericParams {
pub parent_params: Option<Arc<GenericParams>>, pub parent_params: Option<Arc<GenericParams>>,
pub params: Vec<GenericParam>, pub params: Arena<LocalGenericParamId, GenericParamData>,
pub where_predicates: Vec<WherePredicate>, pub where_predicates: Vec<WherePredicate>,
} }
@ -56,7 +57,7 @@ impl GenericParams {
parent_params: Option<Arc<GenericParams>>, parent_params: Option<Arc<GenericParams>>,
) -> GenericParams { ) -> GenericParams {
let mut generics = let mut generics =
GenericParams { params: Vec::new(), parent_params, where_predicates: Vec::new() }; GenericParams { params: Arena::default(), parent_params, where_predicates: Vec::new() };
let start = generics.parent_params.as_ref().map(|p| p.params.len()).unwrap_or(0) as u32; let start = generics.parent_params.as_ref().map(|p| p.params.len()).unwrap_or(0) as u32;
// FIXME: add `: Sized` bound for everything except for `Self` in traits // FIXME: add `: Sized` bound for everything except for `Self` in traits
match def { match def {
@ -66,7 +67,7 @@ impl GenericParams {
GenericDefId::AdtId(AdtId::EnumId(it)) => generics.fill(&it.source(db).value, start), GenericDefId::AdtId(AdtId::EnumId(it)) => generics.fill(&it.source(db).value, start),
GenericDefId::TraitId(it) => { GenericDefId::TraitId(it) => {
// traits get the Self type as an implicit first type parameter // traits get the Self type as an implicit first type parameter
generics.params.push(GenericParam { generics.params.alloc(GenericParamData {
idx: start, idx: start,
name: name::SELF_TYPE, name: name::SELF_TYPE,
default: None, default: None,
@ -110,8 +111,8 @@ impl GenericParams {
let name = type_param.name().map_or_else(Name::missing, |it| it.as_name()); let name = type_param.name().map_or_else(Name::missing, |it| it.as_name());
// FIXME: Use `Path::from_src` // FIXME: Use `Path::from_src`
let default = type_param.default_type().map(TypeRef::from_ast); let default = type_param.default_type().map(TypeRef::from_ast);
let param = GenericParam { idx: idx as u32 + start, name: name.clone(), default }; let param = GenericParamData { idx: idx as u32 + start, name: name.clone(), default };
self.params.push(param); self.params.alloc(param);
let type_ref = TypeRef::Path(name.into()); let type_ref = TypeRef::Path(name.into());
self.fill_bounds(&type_param, type_ref); self.fill_bounds(&type_param, type_ref);
@ -140,8 +141,8 @@ impl GenericParams {
self.where_predicates.push(WherePredicate { type_ref, bound }); self.where_predicates.push(WherePredicate { type_ref, bound });
} }
pub fn find_by_name(&self, name: &Name) -> Option<&GenericParam> { pub fn find_by_name(&self, name: &Name) -> Option<&GenericParamData> {
self.params.iter().find(|p| &p.name == name) self.params.iter().map(|(_id, data)| data).find(|p| &p.name == name)
} }
pub fn count_parent_params(&self) -> usize { pub fn count_parent_params(&self) -> usize {
@ -153,14 +154,14 @@ impl GenericParams {
parent_count + self.params.len() parent_count + self.params.len()
} }
fn for_each_param<'a>(&'a self, f: &mut impl FnMut(&'a GenericParam)) { fn for_each_param<'a>(&'a self, f: &mut impl FnMut(&'a GenericParamData)) {
if let Some(parent) = &self.parent_params { if let Some(parent) = &self.parent_params {
parent.for_each_param(f); parent.for_each_param(f);
} }
self.params.iter().for_each(f); self.params.iter().map(|(_id, data)| data).for_each(f);
} }
pub fn params_including_parent(&self) -> Vec<&GenericParam> { pub fn params_including_parent(&self) -> Vec<&GenericParamData> {
let mut vec = Vec::with_capacity(self.count_params_including_parent()); let mut vec = Vec::with_capacity(self.count_params_including_parent());
self.for_each_param(&mut |p| vec.push(p)); self.for_each_param(&mut |p| vec.push(p));
vec vec

View file

@ -317,6 +317,16 @@ macro_rules! impl_froms {
} }
} }
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
pub struct GenericParamId {
pub parent: GenericDefId,
pub local_id: LocalGenericParamId,
}
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
pub struct LocalGenericParamId(RawId);
impl_arena_id!(LocalGenericParamId);
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)] #[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
pub enum ContainerId { pub enum ContainerId {
ModuleId(ModuleId), ModuleId(ModuleId),

View file

@ -426,7 +426,7 @@ impl Scope {
} }
} }
Scope::GenericParams { params, .. } => { Scope::GenericParams { params, .. } => {
for param in params.params.iter() { for (_id, param) in params.params.iter() {
f(param.name.clone(), ScopeDef::GenericParam(param.idx)) f(param.name.clone(), ScopeDef::GenericParam(param.idx))
} }
} }

View file

@ -662,7 +662,7 @@ impl<'a, D: HirDatabase> InferenceContext<'a, D> {
let mut substs = Vec::with_capacity(parent_param_count + param_count); let mut substs = Vec::with_capacity(parent_param_count + param_count);
// Parent arguments are unknown, except for the receiver type // Parent arguments are unknown, except for the receiver type
if let Some(parent_generics) = def_generics.and_then(|p| p.parent_params.clone()) { if let Some(parent_generics) = def_generics.and_then(|p| p.parent_params.clone()) {
for param in &parent_generics.params { for (_id, param) in parent_generics.params.iter() {
if param.name == name::SELF_TYPE { if param.name == name::SELF_TYPE {
substs.push(receiver_ty.clone()); substs.push(receiver_ty.clone());
} else { } else {