rust-analyzer/crates/ra_hir/src/impl_block.rs

239 lines
7.3 KiB
Rust
Raw Normal View History

use rustc_hash::FxHashMap;
use std::sync::Arc;
use ra_arena::{impl_arena_id, map::ArenaMap, Arena, RawId};
use ra_syntax::{
ast::{self, AstNode},
2019-07-19 07:43:01 +00:00
AstPtr, SourceFile,
2019-03-14 10:14:54 +00:00
};
use crate::{
code_model::{Module, ModuleSource},
generics::HasGenericParams,
2019-01-24 21:26:54 +00:00
ids::LocationCtx,
2019-01-23 22:08:41 +00:00
resolve::Resolver,
ty::Ty,
type_ref::TypeRef,
AstDatabase, Const, DefDatabase, Function, HasSource, HirDatabase, HirFileId, Source, TraitRef,
TypeAlias,
};
#[derive(Debug, Default, PartialEq, Eq)]
pub struct ImplSourceMap {
map: ArenaMap<ImplId, AstPtr<ast::ImplBlock>>,
}
impl ImplSourceMap {
fn insert(&mut self, impl_id: ImplId, impl_block: &ast::ImplBlock) {
self.map.insert(impl_id, AstPtr::new(impl_block))
}
2019-07-19 07:43:01 +00:00
pub fn get(&self, source: &ModuleSource, impl_id: ImplId) -> ast::ImplBlock {
let file = match source {
2019-07-19 07:43:01 +00:00
ModuleSource::SourceFile(file) => file.clone(),
ModuleSource::Module(m) => m.syntax().ancestors().find_map(SourceFile::cast).unwrap(),
};
2019-05-13 16:39:06 +00:00
self.map[impl_id].to_node(file.syntax()).to_owned()
}
}
2019-02-16 20:19:24 +00:00
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
pub struct ImplBlock {
module: Module,
impl_id: ImplId,
}
2019-06-11 14:36:52 +00:00
impl HasSource for ImplBlock {
2019-07-19 07:43:01 +00:00
type Ast = ast::ImplBlock;
fn source(self, db: &(impl DefDatabase + AstDatabase)) -> Source<ast::ImplBlock> {
2019-06-11 14:36:52 +00:00
let source_map = db.impls_in_module_with_source_map(self.module).1;
2019-06-11 14:47:24 +00:00
let src = self.module.definition_source(db);
2019-06-11 15:07:42 +00:00
Source { file_id: src.file_id, ast: source_map.get(&src.ast, self.impl_id) }
2019-06-11 14:36:52 +00:00
}
}
impl ImplBlock {
pub(crate) fn containing(
module_impl_blocks: Arc<ModuleImplBlocks>,
2019-01-24 12:28:50 +00:00
item: ImplItem,
) -> Option<ImplBlock> {
2019-01-24 12:28:50 +00:00
let impl_id = *module_impl_blocks.impls_by_def.get(&item)?;
Some(ImplBlock { module: module_impl_blocks.module, impl_id })
}
pub(crate) fn from_id(module: Module, impl_id: ImplId) -> ImplBlock {
ImplBlock { module, impl_id }
}
pub fn id(&self) -> ImplId {
self.impl_id
}
2019-01-23 22:08:41 +00:00
pub fn module(&self) -> Module {
self.module
2019-01-23 22:08:41 +00:00
}
pub fn target_trait(&self, db: &impl DefDatabase) -> Option<TypeRef> {
db.impls_in_module(self.module).impls[self.impl_id].target_trait().cloned()
}
pub fn target_type(&self, db: &impl DefDatabase) -> TypeRef {
db.impls_in_module(self.module).impls[self.impl_id].target_type().clone()
}
2019-01-26 21:52:04 +00:00
pub fn target_ty(&self, db: &impl HirDatabase) -> Ty {
Ty::from_hir(db, &self.resolver(db), &self.target_type(db))
2019-01-26 21:52:04 +00:00
}
pub fn target_trait_ref(&self, db: &impl HirDatabase) -> Option<TraitRef> {
let target_ty = self.target_ty(db);
TraitRef::from_hir(db, &self.resolver(db), &self.target_trait(db)?, Some(target_ty))
2019-01-26 21:52:04 +00:00
}
pub fn items(&self, db: &impl DefDatabase) -> Vec<ImplItem> {
db.impls_in_module(self.module).impls[self.impl_id].items().to_vec()
}
2019-01-23 22:08:41 +00:00
pub fn is_negative(&self, db: &impl DefDatabase) -> bool {
db.impls_in_module(self.module).impls[self.impl_id].negative
}
pub(crate) fn resolver(&self, db: &impl DefDatabase) -> Resolver {
2019-01-23 22:08:41 +00:00
let r = self.module().resolver(db);
2019-02-16 20:21:36 +00:00
// add generic params, if present
let p = self.generic_params(db);
let r = if !p.params.is_empty() { r.push_generic_params_scope(p) } else { r };
2019-01-23 22:08:41 +00:00
let r = r.push_impl_block_scope(self.clone());
r
}
}
#[derive(Debug, Clone, PartialEq, Eq)]
pub struct ImplData {
target_trait: Option<TypeRef>,
target_type: TypeRef,
items: Vec<ImplItem>,
negative: bool,
}
impl ImplData {
pub(crate) fn from_ast(
2019-06-01 18:17:57 +00:00
db: &(impl DefDatabase + AstDatabase),
file_id: HirFileId,
module: Module,
2019-01-08 08:28:42 +00:00
node: &ast::ImplBlock,
) -> Self {
let target_trait = node.target_trait().map(TypeRef::from_ast);
let target_type = TypeRef::from_ast_opt(node.target_type());
2019-01-24 21:26:54 +00:00
let ctx = LocationCtx::new(db, module, file_id);
let negative = node.is_negative();
let items = if let Some(item_list) = node.item_list() {
item_list
.impl_items()
2019-08-19 11:04:51 +00:00
.map(|item_node| match item_node {
ast::ImplItem::FnDef(it) => Function { id: ctx.to_def(&it) }.into(),
ast::ImplItem::ConstDef(it) => Const { id: ctx.to_def(&it) }.into(),
ast::ImplItem::TypeAliasDef(it) => TypeAlias { id: ctx.to_def(&it) }.into(),
})
.collect()
} else {
Vec::new()
};
ImplData { target_trait, target_type, items, negative }
}
pub fn target_trait(&self) -> Option<&TypeRef> {
self.target_trait.as_ref()
}
pub fn target_type(&self) -> &TypeRef {
&self.target_type
}
pub fn items(&self) -> &[ImplItem] {
&self.items
}
}
2019-01-24 12:28:50 +00:00
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
2019-03-23 07:53:48 +00:00
//FIXME: rename to ImplDef?
pub enum ImplItem {
Method(Function),
2019-01-24 22:38:21 +00:00
Const(Const),
2019-02-24 20:36:49 +00:00
TypeAlias(TypeAlias),
// Existential
}
2019-02-24 20:36:49 +00:00
impl_froms!(ImplItem: Const, TypeAlias);
2019-01-24 12:28:50 +00:00
impl From<Function> for ImplItem {
fn from(func: Function) -> ImplItem {
ImplItem::Method(func)
}
}
#[derive(Debug, Clone, Copy, PartialEq, Eq, PartialOrd, Ord, Hash)]
pub struct ImplId(pub RawId);
impl_arena_id!(ImplId);
/// The collection of impl blocks is a two-step process: first we collect the
/// blocks per-module; then we build an index of all impl blocks in the crate.
/// This way, we avoid having to do this process for the whole crate whenever
/// a file is changed; as long as the impl blocks in the file don't change,
/// we don't need to do the second step again.
#[derive(Debug, PartialEq, Eq)]
pub struct ModuleImplBlocks {
pub(crate) module: Module,
pub(crate) impls: Arena<ImplId, ImplData>,
2019-01-24 12:28:50 +00:00
impls_by_def: FxHashMap<ImplItem, ImplId>,
}
impl ModuleImplBlocks {
2019-06-01 18:17:57 +00:00
fn collect(
db: &(impl DefDatabase + AstDatabase),
module: Module,
source_map: &mut ImplSourceMap,
) -> Self {
2019-01-23 22:08:41 +00:00
let mut m = ModuleImplBlocks {
module,
impls: Arena::default(),
impls_by_def: FxHashMap::default(),
};
2019-06-11 14:47:24 +00:00
let src = m.module.definition_source(db);
let node = match &src.ast {
2019-07-19 07:43:01 +00:00
ModuleSource::SourceFile(node) => node.syntax().clone(),
2019-02-08 11:49:43 +00:00
ModuleSource::Module(node) => {
2019-07-19 07:43:01 +00:00
node.item_list().expect("inline module should have item list").syntax().clone()
2019-02-08 11:49:43 +00:00
}
};
for impl_block_ast in node.children().filter_map(ast::ImplBlock::cast) {
2019-07-19 07:43:01 +00:00
let impl_block = ImplData::from_ast(db, src.file_id, m.module, &impl_block_ast);
2019-01-23 22:08:41 +00:00
let id = m.impls.alloc(impl_block);
for &impl_item in &m.impls[id].items {
m.impls_by_def.insert(impl_item, id);
}
2019-07-19 07:43:01 +00:00
source_map.insert(id, &impl_block_ast);
}
2019-01-23 22:08:41 +00:00
m
}
}
pub(crate) fn impls_in_module_with_source_map_query(
2019-06-01 18:17:57 +00:00
db: &(impl DefDatabase + AstDatabase),
module: Module,
) -> (Arc<ModuleImplBlocks>, Arc<ImplSourceMap>) {
let mut source_map = ImplSourceMap::default();
2019-01-23 22:08:41 +00:00
let result = ModuleImplBlocks::collect(db, module, &mut source_map);
(Arc::new(result), Arc::new(source_map))
}
pub(crate) fn impls_in_module(db: &impl DefDatabase, module: Module) -> Arc<ModuleImplBlocks> {
db.impls_in_module_with_source_map(module).0
}