2018-12-28 13:34:00 +00:00
|
|
|
use std::sync::Arc;
|
|
|
|
use rustc_hash::FxHashMap;
|
|
|
|
|
|
|
|
use ra_arena::{Arena, RawId, impl_arena_id};
|
|
|
|
use ra_syntax::ast::{self, AstNode};
|
|
|
|
|
|
|
|
use crate::{
|
2019-01-04 18:29:53 +00:00
|
|
|
DefId, DefLoc, DefKind, SourceItemId, SourceFileItems,
|
2019-01-24 12:28:50 +00:00
|
|
|
Function, HirFileId,
|
2018-12-28 13:34:00 +00:00
|
|
|
db::HirDatabase,
|
|
|
|
type_ref::TypeRef,
|
|
|
|
};
|
|
|
|
|
2019-01-06 12:58:45 +00:00
|
|
|
use crate::code_model_api::{Module, ModuleSource};
|
2019-01-06 12:16:21 +00:00
|
|
|
|
2018-12-28 13:34:00 +00:00
|
|
|
#[derive(Debug, Clone, PartialEq, Eq)]
|
|
|
|
pub struct ImplBlock {
|
2019-01-04 18:29:53 +00:00
|
|
|
module_impl_blocks: Arc<ModuleImplBlocks>,
|
2018-12-28 13:34:00 +00:00
|
|
|
impl_id: ImplId,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl ImplBlock {
|
|
|
|
pub(crate) fn containing(
|
2019-01-04 18:29:53 +00:00
|
|
|
module_impl_blocks: Arc<ModuleImplBlocks>,
|
2019-01-24 12:28:50 +00:00
|
|
|
item: ImplItem,
|
2018-12-28 13:34:00 +00:00
|
|
|
) -> Option<ImplBlock> {
|
2019-01-24 12:28:50 +00:00
|
|
|
let impl_id = *module_impl_blocks.impls_by_def.get(&item)?;
|
2018-12-28 13:34:00 +00:00
|
|
|
Some(ImplBlock {
|
2019-01-04 18:29:53 +00:00
|
|
|
module_impl_blocks,
|
2018-12-28 13:34:00 +00:00
|
|
|
impl_id,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2019-01-07 12:44:54 +00:00
|
|
|
pub(crate) fn from_id(module_impl_blocks: Arc<ModuleImplBlocks>, impl_id: ImplId) -> ImplBlock {
|
|
|
|
ImplBlock {
|
|
|
|
module_impl_blocks,
|
|
|
|
impl_id,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-12-28 13:34:00 +00:00
|
|
|
fn impl_data(&self) -> &ImplData {
|
2019-01-04 18:29:53 +00:00
|
|
|
&self.module_impl_blocks.impls[self.impl_id]
|
2018-12-28 13:34:00 +00:00
|
|
|
}
|
|
|
|
|
2018-12-30 18:59:49 +00:00
|
|
|
pub fn target_trait(&self) -> Option<&TypeRef> {
|
2019-01-07 12:44:54 +00:00
|
|
|
self.impl_data().target_trait()
|
2018-12-30 18:59:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn target_type(&self) -> &TypeRef {
|
2019-01-07 12:44:54 +00:00
|
|
|
self.impl_data().target_type()
|
2018-12-28 13:34:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn items(&self) -> &[ImplItem] {
|
2019-01-07 12:44:54 +00:00
|
|
|
self.impl_data().items()
|
2018-12-28 13:34:00 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug, Clone, PartialEq, Eq)]
|
|
|
|
pub struct ImplData {
|
2018-12-30 18:59:49 +00:00
|
|
|
target_trait: Option<TypeRef>,
|
|
|
|
target_type: TypeRef,
|
2018-12-28 13:34:00 +00:00
|
|
|
items: Vec<ImplItem>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl ImplData {
|
|
|
|
pub(crate) fn from_ast(
|
2019-01-24 12:28:50 +00:00
|
|
|
db: &impl HirDatabase,
|
2019-01-23 20:14:13 +00:00
|
|
|
file_id: HirFileId,
|
2018-12-28 13:34:00 +00:00
|
|
|
file_items: &SourceFileItems,
|
2019-01-23 20:14:13 +00:00
|
|
|
module: Module,
|
2019-01-08 08:28:42 +00:00
|
|
|
node: &ast::ImplBlock,
|
2018-12-28 13:34:00 +00:00
|
|
|
) -> Self {
|
2019-01-07 12:44:54 +00:00
|
|
|
let target_trait = node.target_trait().map(TypeRef::from_ast);
|
2018-12-30 18:59:49 +00:00
|
|
|
let target_type = TypeRef::from_ast_opt(node.target_type());
|
2018-12-28 13:34:00 +00:00
|
|
|
let items = if let Some(item_list) = node.item_list() {
|
|
|
|
item_list
|
|
|
|
.impl_items()
|
|
|
|
.map(|item_node| {
|
2019-01-08 08:28:42 +00:00
|
|
|
let kind = match item_node.kind() {
|
2019-01-24 12:31:12 +00:00
|
|
|
ast::ImplItemKind::FnDef(it) => {
|
|
|
|
return ImplItem::Method(Function::from_ast(db, module, file_id, it));
|
|
|
|
}
|
2019-01-08 08:28:42 +00:00
|
|
|
ast::ImplItemKind::ConstDef(..) => DefKind::Item,
|
|
|
|
ast::ImplItemKind::TypeDef(..) => DefKind::Item,
|
2018-12-28 13:34:00 +00:00
|
|
|
};
|
|
|
|
let item_id = file_items.id_of_unchecked(item_node.syntax());
|
2019-01-06 12:16:21 +00:00
|
|
|
let source_item_id = SourceItemId {
|
2019-01-23 20:14:13 +00:00
|
|
|
file_id,
|
2019-01-06 12:16:21 +00:00
|
|
|
item_id: Some(item_id),
|
|
|
|
};
|
2018-12-28 13:34:00 +00:00
|
|
|
let def_loc = DefLoc {
|
2019-01-23 20:14:13 +00:00
|
|
|
module,
|
2018-12-28 13:34:00 +00:00
|
|
|
kind,
|
2019-01-06 12:16:21 +00:00
|
|
|
source_item_id,
|
2018-12-28 13:34:00 +00:00
|
|
|
};
|
|
|
|
let def_id = def_loc.id(db);
|
2019-01-08 08:28:42 +00:00
|
|
|
match item_node.kind() {
|
2019-01-24 12:31:12 +00:00
|
|
|
ast::ImplItemKind::FnDef(_) => unreachable!(),
|
2019-01-08 08:28:42 +00:00
|
|
|
ast::ImplItemKind::ConstDef(..) => ImplItem::Const(def_id),
|
|
|
|
ast::ImplItemKind::TypeDef(..) => ImplItem::Type(def_id),
|
2018-12-28 13:34:00 +00:00
|
|
|
}
|
|
|
|
})
|
|
|
|
.collect()
|
|
|
|
} else {
|
|
|
|
Vec::new()
|
|
|
|
};
|
2018-12-30 18:59:49 +00:00
|
|
|
ImplData {
|
|
|
|
target_trait,
|
|
|
|
target_type,
|
|
|
|
items,
|
|
|
|
}
|
2018-12-28 13:34:00 +00:00
|
|
|
}
|
2019-01-07 12:44:54 +00:00
|
|
|
|
|
|
|
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
|
|
|
|
}
|
2018-12-28 13:34:00 +00:00
|
|
|
}
|
|
|
|
|
2019-01-24 12:28:50 +00:00
|
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
|
|
|
|
//TODO: rename to ImplDef?
|
2018-12-28 13:34:00 +00:00
|
|
|
pub enum ImplItem {
|
|
|
|
Method(Function),
|
|
|
|
// these don't have their own types yet
|
|
|
|
Const(DefId),
|
|
|
|
Type(DefId),
|
|
|
|
// Existential
|
|
|
|
}
|
|
|
|
|
2019-01-24 12:28:50 +00:00
|
|
|
impl From<Function> for ImplItem {
|
|
|
|
fn from(func: Function) -> ImplItem {
|
|
|
|
ImplItem::Method(func)
|
2018-12-28 13:34:00 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, PartialOrd, Ord, Hash)]
|
|
|
|
pub struct ImplId(pub RawId);
|
|
|
|
impl_arena_id!(ImplId);
|
|
|
|
|
2019-01-08 23:47:12 +00:00
|
|
|
/// 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.
|
2018-12-28 13:34:00 +00:00
|
|
|
#[derive(Debug, PartialEq, Eq)]
|
2019-01-04 18:29:53 +00:00
|
|
|
pub struct ModuleImplBlocks {
|
2019-01-07 12:44:54 +00:00
|
|
|
pub(crate) impls: Arena<ImplId, ImplData>,
|
2019-01-24 12:28:50 +00:00
|
|
|
impls_by_def: FxHashMap<ImplItem, ImplId>,
|
2018-12-28 13:34:00 +00:00
|
|
|
}
|
|
|
|
|
2019-01-04 18:29:53 +00:00
|
|
|
impl ModuleImplBlocks {
|
2018-12-28 13:34:00 +00:00
|
|
|
fn new() -> Self {
|
2019-01-04 18:29:53 +00:00
|
|
|
ModuleImplBlocks {
|
2018-12-28 13:34:00 +00:00
|
|
|
impls: Arena::default(),
|
|
|
|
impls_by_def: FxHashMap::default(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-15 16:18:52 +00:00
|
|
|
fn collect(&mut self, db: &impl HirDatabase, module: Module) {
|
2019-01-15 15:26:29 +00:00
|
|
|
let (file_id, module_source) = module.definition_source(db);
|
2019-01-23 20:14:13 +00:00
|
|
|
let file_id: HirFileId = file_id.into();
|
2019-01-06 12:58:45 +00:00
|
|
|
let node = match &module_source {
|
2019-01-08 08:28:42 +00:00
|
|
|
ModuleSource::SourceFile(node) => node.syntax(),
|
2019-01-12 20:51:56 +00:00
|
|
|
ModuleSource::Module(node) => node
|
|
|
|
.item_list()
|
|
|
|
.expect("inline module should have item list")
|
|
|
|
.syntax(),
|
2018-12-28 13:34:00 +00:00
|
|
|
};
|
|
|
|
|
2019-01-23 20:14:13 +00:00
|
|
|
let source_file_items = db.file_items(file_id);
|
2018-12-28 13:34:00 +00:00
|
|
|
|
|
|
|
for impl_block_ast in node.children().filter_map(ast::ImplBlock::cast) {
|
2019-01-23 20:14:13 +00:00
|
|
|
let impl_block =
|
|
|
|
ImplData::from_ast(db, file_id, &source_file_items, module, impl_block_ast);
|
2018-12-28 13:34:00 +00:00
|
|
|
let id = self.impls.alloc(impl_block);
|
2019-01-24 12:28:50 +00:00
|
|
|
for &impl_item in &self.impls[id].items {
|
|
|
|
self.impls_by_def.insert(impl_item, id);
|
2018-12-28 13:34:00 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-23 20:14:13 +00:00
|
|
|
pub(crate) fn impls_in_module(db: &impl HirDatabase, module: Module) -> Arc<ModuleImplBlocks> {
|
2019-01-04 18:29:53 +00:00
|
|
|
let mut result = ModuleImplBlocks::new();
|
2019-01-15 16:18:52 +00:00
|
|
|
result.collect(db, module);
|
|
|
|
Arc::new(result)
|
2018-12-28 13:34:00 +00:00
|
|
|
}
|