2019-10-30 10:10:38 +00:00
|
|
|
//! `hir_def` crate contains everything between macro expansion and type
|
|
|
|
//! inference.
|
|
|
|
//!
|
|
|
|
//! It defines various items (structs, enums, traits) which comprises Rust code,
|
|
|
|
//! as well as an algorithm for resolving paths to such entities.
|
|
|
|
//!
|
|
|
|
//! Note that `hir_def` is a work in progress, so not all of the above is
|
|
|
|
//! actually true.
|
|
|
|
|
2020-04-06 14:58:16 +00:00
|
|
|
#[allow(unused)]
|
|
|
|
macro_rules! eprintln {
|
|
|
|
($($tt:tt)*) => { stdx::eprintln!($($tt)*) };
|
|
|
|
}
|
|
|
|
|
2019-10-30 10:10:38 +00:00
|
|
|
pub mod db;
|
2019-11-24 14:00:10 +00:00
|
|
|
|
2019-10-30 13:12:55 +00:00
|
|
|
pub mod attr;
|
|
|
|
pub mod path;
|
|
|
|
pub mod type_ref;
|
2019-10-31 07:51:54 +00:00
|
|
|
pub mod builtin_type;
|
2020-12-11 13:04:33 +00:00
|
|
|
pub mod builtin_attr;
|
2019-11-24 14:00:10 +00:00
|
|
|
pub mod per_ns;
|
2019-12-20 14:38:17 +00:00
|
|
|
pub mod item_scope;
|
2019-11-24 14:00:10 +00:00
|
|
|
|
2019-12-05 22:34:12 +00:00
|
|
|
pub mod dyn_map;
|
|
|
|
pub mod keys;
|
|
|
|
|
2020-03-25 14:33:01 +00:00
|
|
|
pub mod item_tree;
|
2021-04-05 01:50:10 +00:00
|
|
|
pub mod intern;
|
2020-03-25 14:33:01 +00:00
|
|
|
|
2019-11-24 14:00:10 +00:00
|
|
|
pub mod adt;
|
2019-11-22 14:32:10 +00:00
|
|
|
pub mod data;
|
2019-11-24 14:00:10 +00:00
|
|
|
pub mod generics;
|
2019-11-23 09:58:01 +00:00
|
|
|
pub mod lang_item;
|
2019-11-24 14:00:10 +00:00
|
|
|
|
|
|
|
pub mod expr;
|
|
|
|
pub mod body;
|
|
|
|
pub mod resolver;
|
2019-10-30 13:12:55 +00:00
|
|
|
|
2019-11-22 18:43:36 +00:00
|
|
|
mod trace;
|
2019-11-27 14:46:02 +00:00
|
|
|
pub mod nameres;
|
2019-11-22 18:43:36 +00:00
|
|
|
|
2019-11-28 15:05:28 +00:00
|
|
|
pub mod src;
|
2019-12-05 22:34:12 +00:00
|
|
|
pub mod child_by_source;
|
2019-11-28 15:05:28 +00:00
|
|
|
|
2019-12-24 19:32:42 +00:00
|
|
|
pub mod visibility;
|
2019-12-30 13:25:19 +00:00
|
|
|
pub mod find_path;
|
2020-05-20 21:51:20 +00:00
|
|
|
pub mod import_map;
|
2019-12-24 19:32:42 +00:00
|
|
|
|
2019-11-03 17:53:17 +00:00
|
|
|
#[cfg(test)]
|
|
|
|
mod test_db;
|
|
|
|
|
2021-01-22 15:31:40 +00:00
|
|
|
use std::{
|
|
|
|
hash::{Hash, Hasher},
|
|
|
|
sync::Arc,
|
|
|
|
};
|
2019-10-30 10:10:38 +00:00
|
|
|
|
2021-05-31 11:37:11 +00:00
|
|
|
use attr::Attr;
|
2020-08-13 14:25:38 +00:00
|
|
|
use base_db::{impl_intern_key, salsa, CrateId};
|
2020-02-17 04:57:24 +00:00
|
|
|
use hir_expand::{
|
2021-02-28 11:12:11 +00:00
|
|
|
ast_id_map::FileAstId,
|
2021-03-17 06:31:14 +00:00
|
|
|
eager::{expand_eager_macro, ErrorEmitted, ErrorSink},
|
2021-02-28 11:12:11 +00:00
|
|
|
hygiene::Hygiene,
|
2021-05-10 14:35:06 +00:00
|
|
|
AstId, FragmentKind, HirFileId, InFile, MacroCallId, MacroCallKind, MacroDefId, MacroDefKind,
|
2020-02-17 04:57:24 +00:00
|
|
|
};
|
2021-01-14 15:47:42 +00:00
|
|
|
use la_arena::Idx;
|
2021-01-22 15:31:40 +00:00
|
|
|
use nameres::DefMap;
|
2021-04-16 13:48:03 +00:00
|
|
|
use path::ModPath;
|
internal: move diagnostics to hir
The idea here is to eventually get rid of `dyn Diagnostic` and
`DiagnosticSink` infrastructure altogether, and just have a `enum
hir::Diagnostic` instead.
The problem with `dyn Diagnostic` is that it is defined in the lowest
level of the stack (hir_expand), but is used by the highest level (ide).
As a first step, we free hir_expand and hir_def from `dyn Diagnostic`
and kick the can up to `hir_ty`, as an intermediate state. The plan is
then to move DiagnosticSink similarly to the hir crate, and, as final
third step, remove its usage from the ide.
One currently unsolved problem is testing. You can notice that the test
which checks precise diagnostic ranges, unresolved_import_in_use_tree,
was moved to the ide layer. Logically, only IDE should have the infra to
render a specific range.
At the same time, the range is determined with the data produced in
hir_def and hir crates, so this layering is rather unfortunate. Working
on hir_def shouldn't require compiling `ide` for testing.
2021-05-23 20:31:59 +00:00
|
|
|
use stdx::impl_from;
|
2020-08-12 16:26:51 +00:00
|
|
|
use syntax::ast;
|
2019-10-30 10:10:38 +00:00
|
|
|
|
internal: move diagnostics to hir
The idea here is to eventually get rid of `dyn Diagnostic` and
`DiagnosticSink` infrastructure altogether, and just have a `enum
hir::Diagnostic` instead.
The problem with `dyn Diagnostic` is that it is defined in the lowest
level of the stack (hir_expand), but is used by the highest level (ide).
As a first step, we free hir_expand and hir_def from `dyn Diagnostic`
and kick the can up to `hir_ty`, as an intermediate state. The plan is
then to move DiagnosticSink similarly to the hir crate, and, as final
third step, remove its usage from the ide.
One currently unsolved problem is testing. You can notice that the test
which checks precise diagnostic ranges, unresolved_import_in_use_tree,
was moved to the ide layer. Logically, only IDE should have the infra to
render a specific range.
At the same time, the range is determined with the data produced in
hir_def and hir crates, so this layering is rather unfortunate. Working
on hir_def shouldn't require compiling `ide` for testing.
2021-05-23 20:31:59 +00:00
|
|
|
use crate::{
|
|
|
|
adt::VariantData,
|
|
|
|
attr::AttrId,
|
|
|
|
builtin_type::BuiltinType,
|
|
|
|
item_tree::{
|
|
|
|
Const, Enum, Function, Impl, ItemTreeId, ItemTreeNode, ModItem, Static, Struct, Trait,
|
|
|
|
TypeAlias, Union,
|
|
|
|
},
|
2020-06-22 13:07:06 +00:00
|
|
|
};
|
2019-10-30 10:10:38 +00:00
|
|
|
|
2019-11-23 13:49:05 +00:00
|
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
|
2019-10-30 09:27:54 +00:00
|
|
|
pub struct ModuleId {
|
2021-01-22 17:09:55 +00:00
|
|
|
krate: CrateId,
|
2021-03-18 23:06:35 +00:00
|
|
|
/// If this `ModuleId` was derived from a `DefMap` for a block expression, this stores the
|
|
|
|
/// `BlockId` of that block expression. If `None`, this module is part of the crate-level
|
|
|
|
/// `DefMap` of `krate`.
|
2021-01-25 18:02:05 +00:00
|
|
|
block: Option<BlockId>,
|
2021-03-18 23:06:35 +00:00
|
|
|
/// The module's ID in its originating `DefMap`.
|
2019-11-27 18:31:51 +00:00
|
|
|
pub local_id: LocalModuleId,
|
2019-10-30 09:27:54 +00:00
|
|
|
}
|
|
|
|
|
2021-01-22 15:31:40 +00:00
|
|
|
impl ModuleId {
|
|
|
|
pub fn def_map(&self, db: &dyn db::DefDatabase) -> Arc<DefMap> {
|
2021-01-25 18:02:05 +00:00
|
|
|
match self.block {
|
2021-02-01 12:32:43 +00:00
|
|
|
Some(block) => {
|
|
|
|
db.block_def_map(block).unwrap_or_else(|| {
|
|
|
|
// NOTE: This should be unreachable - all `ModuleId`s come from their `DefMap`s,
|
|
|
|
// so the `DefMap` here must exist.
|
2021-03-18 23:06:35 +00:00
|
|
|
unreachable!("no `block_def_map` for `ModuleId` {:?}", self);
|
2021-02-01 12:32:43 +00:00
|
|
|
})
|
|
|
|
}
|
2021-01-25 18:02:05 +00:00
|
|
|
None => db.crate_def_map(self.krate),
|
|
|
|
}
|
2021-01-22 15:31:40 +00:00
|
|
|
}
|
2021-01-22 17:09:55 +00:00
|
|
|
|
|
|
|
pub fn krate(&self) -> CrateId {
|
|
|
|
self.krate
|
|
|
|
}
|
2021-03-01 18:36:34 +00:00
|
|
|
|
|
|
|
pub fn containing_module(&self, db: &dyn db::DefDatabase) -> Option<ModuleId> {
|
|
|
|
self.def_map(db).containing_module(self.local_id)
|
|
|
|
}
|
2021-04-18 23:06:04 +00:00
|
|
|
|
|
|
|
/// Returns `true` if this module represents a block expression.
|
|
|
|
///
|
|
|
|
/// Returns `false` if this module is a submodule *inside* a block expression
|
|
|
|
/// (eg. `m` in `{ mod m {} }`).
|
|
|
|
pub fn is_block_root(&self, db: &dyn db::DefDatabase) -> bool {
|
|
|
|
if self.block.is_none() {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
self.def_map(db)[self.local_id].parent.is_none()
|
|
|
|
}
|
2021-01-22 15:31:40 +00:00
|
|
|
}
|
|
|
|
|
2019-10-30 09:27:54 +00:00
|
|
|
/// An ID of a module, **local** to a specific crate
|
2020-03-19 15:00:11 +00:00
|
|
|
pub type LocalModuleId = Idx<nameres::ModuleData>;
|
2019-10-30 10:10:38 +00:00
|
|
|
|
2020-06-22 13:07:06 +00:00
|
|
|
#[derive(Debug)]
|
|
|
|
pub struct ItemLoc<N: ItemTreeNode> {
|
2021-03-09 18:09:02 +00:00
|
|
|
pub container: ModuleId,
|
2020-06-22 13:07:06 +00:00
|
|
|
pub id: ItemTreeId<N>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<N: ItemTreeNode> Clone for ItemLoc<N> {
|
|
|
|
fn clone(&self) -> Self {
|
|
|
|
Self { container: self.container, id: self.id }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<N: ItemTreeNode> Copy for ItemLoc<N> {}
|
|
|
|
|
|
|
|
impl<N: ItemTreeNode> PartialEq for ItemLoc<N> {
|
|
|
|
fn eq(&self, other: &Self) -> bool {
|
|
|
|
self.container == other.container && self.id == other.id
|
|
|
|
}
|
2019-12-20 12:11:01 +00:00
|
|
|
}
|
2019-10-30 10:10:38 +00:00
|
|
|
|
2020-06-22 13:07:06 +00:00
|
|
|
impl<N: ItemTreeNode> Eq for ItemLoc<N> {}
|
|
|
|
|
|
|
|
impl<N: ItemTreeNode> Hash for ItemLoc<N> {
|
|
|
|
fn hash<H: Hasher>(&self, state: &mut H) {
|
|
|
|
self.container.hash(state);
|
|
|
|
self.id.hash(state);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug)]
|
|
|
|
pub struct AssocItemLoc<N: ItemTreeNode> {
|
2019-12-20 10:59:50 +00:00
|
|
|
pub container: AssocContainerId,
|
2020-06-22 13:07:06 +00:00
|
|
|
pub id: ItemTreeId<N>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<N: ItemTreeNode> Clone for AssocItemLoc<N> {
|
|
|
|
fn clone(&self) -> Self {
|
|
|
|
Self { container: self.container, id: self.id }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<N: ItemTreeNode> Copy for AssocItemLoc<N> {}
|
|
|
|
|
|
|
|
impl<N: ItemTreeNode> PartialEq for AssocItemLoc<N> {
|
|
|
|
fn eq(&self, other: &Self) -> bool {
|
|
|
|
self.container == other.container && self.id == other.id
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<N: ItemTreeNode> Eq for AssocItemLoc<N> {}
|
|
|
|
|
|
|
|
impl<N: ItemTreeNode> Hash for AssocItemLoc<N> {
|
|
|
|
fn hash<H: Hasher>(&self, state: &mut H) {
|
|
|
|
self.container.hash(state);
|
|
|
|
self.id.hash(state);
|
|
|
|
}
|
2019-11-20 13:03:59 +00:00
|
|
|
}
|
|
|
|
|
2019-12-20 12:19:41 +00:00
|
|
|
macro_rules! impl_intern {
|
|
|
|
($id:ident, $loc:ident, $intern:ident, $lookup:ident) => {
|
|
|
|
impl_intern_key!($id);
|
|
|
|
|
|
|
|
impl Intern for $loc {
|
|
|
|
type ID = $id;
|
2020-03-13 15:05:46 +00:00
|
|
|
fn intern(self, db: &dyn db::DefDatabase) -> $id {
|
2019-12-20 12:19:41 +00:00
|
|
|
db.$intern(self)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Lookup for $id {
|
|
|
|
type Data = $loc;
|
2020-03-13 15:05:46 +00:00
|
|
|
fn lookup(&self, db: &dyn db::DefDatabase) -> $loc {
|
2019-12-20 12:19:41 +00:00
|
|
|
db.$lookup(*self)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2019-12-20 12:11:01 +00:00
|
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
|
|
|
|
pub struct FunctionId(salsa::InternId);
|
2020-06-22 13:07:06 +00:00
|
|
|
type FunctionLoc = AssocItemLoc<Function>;
|
2019-12-20 12:19:41 +00:00
|
|
|
impl_intern!(FunctionId, FunctionLoc, intern_function, lookup_intern_function);
|
2019-10-30 10:10:38 +00:00
|
|
|
|
2020-07-15 19:47:45 +00:00
|
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash, PartialOrd, Ord)]
|
2019-11-25 14:30:50 +00:00
|
|
|
pub struct StructId(salsa::InternId);
|
2020-06-22 13:07:06 +00:00
|
|
|
type StructLoc = ItemLoc<Struct>;
|
2019-12-20 12:19:41 +00:00
|
|
|
impl_intern!(StructId, StructLoc, intern_struct, lookup_intern_struct);
|
2019-10-30 10:10:38 +00:00
|
|
|
|
2020-07-15 19:47:45 +00:00
|
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash, PartialOrd, Ord)]
|
2019-11-25 14:30:50 +00:00
|
|
|
pub struct UnionId(salsa::InternId);
|
2020-06-22 13:07:06 +00:00
|
|
|
pub type UnionLoc = ItemLoc<Union>;
|
2019-12-20 12:19:41 +00:00
|
|
|
impl_intern!(UnionId, UnionLoc, intern_union, lookup_intern_union);
|
2019-10-30 13:12:55 +00:00
|
|
|
|
2020-07-15 19:47:45 +00:00
|
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash, PartialOrd, Ord)]
|
2019-10-30 10:10:38 +00:00
|
|
|
pub struct EnumId(salsa::InternId);
|
2020-06-22 13:07:06 +00:00
|
|
|
pub type EnumLoc = ItemLoc<Enum>;
|
2019-12-20 12:19:41 +00:00
|
|
|
impl_intern!(EnumId, EnumLoc, intern_enum, lookup_intern_enum);
|
2019-10-30 10:10:38 +00:00
|
|
|
|
2019-10-30 13:12:55 +00:00
|
|
|
// FIXME: rename to `VariantId`, only enums can ave variants
|
|
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
|
|
|
|
pub struct EnumVariantId {
|
2019-10-31 15:45:10 +00:00
|
|
|
pub parent: EnumId,
|
|
|
|
pub local_id: LocalEnumVariantId,
|
2019-10-30 13:12:55 +00:00
|
|
|
}
|
|
|
|
|
2020-03-19 15:00:11 +00:00
|
|
|
pub type LocalEnumVariantId = Idx<adt::EnumVariantData>;
|
2019-10-30 13:12:55 +00:00
|
|
|
|
2019-10-31 13:40:36 +00:00
|
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
|
2020-04-25 12:23:34 +00:00
|
|
|
pub struct FieldId {
|
2019-11-23 08:14:10 +00:00
|
|
|
pub parent: VariantId,
|
2020-04-25 12:23:34 +00:00
|
|
|
pub local_id: LocalFieldId,
|
2019-10-31 13:40:36 +00:00
|
|
|
}
|
|
|
|
|
2020-04-25 12:23:34 +00:00
|
|
|
pub type LocalFieldId = Idx<adt::FieldData>;
|
2019-10-31 13:40:36 +00:00
|
|
|
|
2019-10-30 10:10:38 +00:00
|
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
|
|
|
|
pub struct ConstId(salsa::InternId);
|
2020-06-22 13:07:06 +00:00
|
|
|
type ConstLoc = AssocItemLoc<Const>;
|
2019-12-20 12:19:41 +00:00
|
|
|
impl_intern!(ConstId, ConstLoc, intern_const, lookup_intern_const);
|
2019-10-30 10:10:38 +00:00
|
|
|
|
|
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
|
|
|
|
pub struct StaticId(salsa::InternId);
|
2020-06-22 13:07:06 +00:00
|
|
|
pub type StaticLoc = ItemLoc<Static>;
|
2019-12-20 12:19:41 +00:00
|
|
|
impl_intern!(StaticId, StaticLoc, intern_static, lookup_intern_static);
|
2019-10-30 10:10:38 +00:00
|
|
|
|
|
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
|
|
|
|
pub struct TraitId(salsa::InternId);
|
2020-06-22 13:07:06 +00:00
|
|
|
pub type TraitLoc = ItemLoc<Trait>;
|
2019-12-20 12:19:41 +00:00
|
|
|
impl_intern!(TraitId, TraitLoc, intern_trait, lookup_intern_trait);
|
2019-10-30 10:10:38 +00:00
|
|
|
|
|
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
|
|
|
|
pub struct TypeAliasId(salsa::InternId);
|
2020-06-22 13:07:06 +00:00
|
|
|
type TypeAliasLoc = AssocItemLoc<TypeAlias>;
|
2019-12-20 12:19:41 +00:00
|
|
|
impl_intern!(TypeAliasId, TypeAliasLoc, intern_type_alias, lookup_intern_type_alias);
|
2019-10-31 08:23:30 +00:00
|
|
|
|
2020-06-18 23:29:34 +00:00
|
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash, Ord, PartialOrd)]
|
2019-11-15 16:14:50 +00:00
|
|
|
pub struct ImplId(salsa::InternId);
|
2020-06-22 13:07:06 +00:00
|
|
|
type ImplLoc = ItemLoc<Impl>;
|
2019-12-20 12:47:44 +00:00
|
|
|
impl_intern!(ImplId, ImplLoc, intern_impl, lookup_intern_impl);
|
2019-11-15 16:14:50 +00:00
|
|
|
|
2021-01-25 18:02:05 +00:00
|
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash, Ord, PartialOrd)]
|
|
|
|
pub struct BlockId(salsa::InternId);
|
|
|
|
#[derive(Debug, Hash, PartialEq, Eq, Clone)]
|
|
|
|
pub struct BlockLoc {
|
|
|
|
ast_id: AstId<ast::BlockExpr>,
|
2021-02-01 12:32:43 +00:00
|
|
|
/// The containing module.
|
2021-01-25 18:02:05 +00:00
|
|
|
module: ModuleId,
|
|
|
|
}
|
|
|
|
impl_intern!(BlockId, BlockLoc, intern_block, lookup_intern_block);
|
|
|
|
|
2019-12-20 12:11:01 +00:00
|
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
|
|
|
|
pub struct TypeParamId {
|
|
|
|
pub parent: GenericDefId,
|
|
|
|
pub local_id: LocalTypeParamId,
|
|
|
|
}
|
|
|
|
|
2020-03-19 15:00:11 +00:00
|
|
|
pub type LocalTypeParamId = Idx<generics::TypeParamData>;
|
2019-12-20 12:11:01 +00:00
|
|
|
|
2019-11-20 14:49:57 +00:00
|
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
|
2020-12-11 12:49:32 +00:00
|
|
|
pub struct LifetimeParamId {
|
|
|
|
pub parent: GenericDefId,
|
|
|
|
pub local_id: LocalLifetimeParamId,
|
|
|
|
}
|
|
|
|
pub type LocalLifetimeParamId = Idx<generics::LifetimeParamData>;
|
|
|
|
|
2021-01-01 09:06:42 +00:00
|
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
|
|
|
|
pub struct ConstParamId {
|
|
|
|
pub parent: GenericDefId,
|
|
|
|
pub local_id: LocalConstParamId,
|
|
|
|
}
|
|
|
|
pub type LocalConstParamId = Idx<generics::ConstParamData>;
|
|
|
|
|
2019-12-20 11:07:23 +00:00
|
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
|
|
|
|
pub enum AssocContainerId {
|
2021-03-09 17:27:16 +00:00
|
|
|
ModuleId(ModuleId),
|
2019-11-20 14:49:57 +00:00
|
|
|
ImplId(ImplId),
|
|
|
|
TraitId(TraitId),
|
|
|
|
}
|
2021-03-09 17:27:16 +00:00
|
|
|
impl_from!(ModuleId for AssocContainerId);
|
2019-11-20 14:49:57 +00:00
|
|
|
|
2019-10-31 08:23:30 +00:00
|
|
|
/// A Data Type
|
2020-07-15 19:47:45 +00:00
|
|
|
#[derive(Clone, Copy, Debug, PartialEq, Eq, Hash, PartialOrd, Ord)]
|
2019-10-31 08:23:30 +00:00
|
|
|
pub enum AdtId {
|
|
|
|
StructId(StructId),
|
|
|
|
UnionId(UnionId),
|
|
|
|
EnumId(EnumId),
|
|
|
|
}
|
2020-07-13 14:16:53 +00:00
|
|
|
impl_from!(StructId, UnionId, EnumId for AdtId);
|
2019-10-31 08:23:30 +00:00
|
|
|
|
2021-01-01 23:42:07 +00:00
|
|
|
/// A generic param
|
|
|
|
#[derive(Clone, Copy, Debug, PartialEq, Eq, Hash)]
|
|
|
|
pub enum GenericParamId {
|
|
|
|
TypeParamId(TypeParamId),
|
|
|
|
LifetimeParamId(LifetimeParamId),
|
|
|
|
ConstParamId(ConstParamId),
|
|
|
|
}
|
|
|
|
impl_from!(TypeParamId, LifetimeParamId, ConstParamId for GenericParamId);
|
|
|
|
|
2019-10-31 08:23:30 +00:00
|
|
|
/// The defs which can be visible in the module.
|
|
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
|
|
|
|
pub enum ModuleDefId {
|
|
|
|
ModuleId(ModuleId),
|
|
|
|
FunctionId(FunctionId),
|
|
|
|
AdtId(AdtId),
|
|
|
|
// Can't be directly declared, but can be imported.
|
|
|
|
EnumVariantId(EnumVariantId),
|
|
|
|
ConstId(ConstId),
|
|
|
|
StaticId(StaticId),
|
|
|
|
TraitId(TraitId),
|
|
|
|
TypeAliasId(TypeAliasId),
|
|
|
|
BuiltinType(BuiltinType),
|
|
|
|
}
|
2020-07-13 14:16:53 +00:00
|
|
|
impl_from!(
|
|
|
|
ModuleId,
|
2019-10-31 08:23:30 +00:00
|
|
|
FunctionId,
|
|
|
|
AdtId(StructId, EnumId, UnionId),
|
|
|
|
EnumVariantId,
|
|
|
|
ConstId,
|
|
|
|
StaticId,
|
|
|
|
TraitId,
|
|
|
|
TypeAliasId,
|
|
|
|
BuiltinType
|
2020-07-13 14:16:53 +00:00
|
|
|
for ModuleDefId
|
2019-10-31 08:23:30 +00:00
|
|
|
);
|
2019-11-14 14:37:22 +00:00
|
|
|
|
|
|
|
/// The defs which have a body.
|
|
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
|
|
|
|
pub enum DefWithBodyId {
|
|
|
|
FunctionId(FunctionId),
|
|
|
|
StaticId(StaticId),
|
|
|
|
ConstId(ConstId),
|
|
|
|
}
|
|
|
|
|
2020-07-13 14:16:53 +00:00
|
|
|
impl_from!(FunctionId, ConstId, StaticId for DefWithBodyId);
|
2019-11-15 18:28:00 +00:00
|
|
|
|
2021-03-13 19:38:11 +00:00
|
|
|
impl DefWithBodyId {
|
|
|
|
pub fn as_generic_def_id(self) -> Option<GenericDefId> {
|
|
|
|
match self {
|
|
|
|
DefWithBodyId::FunctionId(f) => Some(f.into()),
|
|
|
|
DefWithBodyId::StaticId(_) => None,
|
|
|
|
DefWithBodyId::ConstId(c) => Some(c.into()),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-11-15 18:28:00 +00:00
|
|
|
#[derive(Debug, Copy, Clone, PartialEq, Eq, Hash)]
|
|
|
|
pub enum AssocItemId {
|
|
|
|
FunctionId(FunctionId),
|
|
|
|
ConstId(ConstId),
|
|
|
|
TypeAliasId(TypeAliasId),
|
|
|
|
}
|
|
|
|
// FIXME: not every function, ... is actually an assoc item. maybe we should make
|
|
|
|
// sure that you can only turn actual assoc items into AssocItemIds. This would
|
|
|
|
// require not implementing From, and instead having some checked way of
|
|
|
|
// casting them, and somehow making the constructors private, which would be annoying.
|
2020-07-13 14:16:53 +00:00
|
|
|
impl_from!(FunctionId, ConstId, TypeAliasId for AssocItemId);
|
2019-11-20 09:25:02 +00:00
|
|
|
|
|
|
|
#[derive(Clone, Copy, PartialEq, Eq, Debug, Hash)]
|
|
|
|
pub enum GenericDefId {
|
|
|
|
FunctionId(FunctionId),
|
|
|
|
AdtId(AdtId),
|
|
|
|
TraitId(TraitId),
|
|
|
|
TypeAliasId(TypeAliasId),
|
|
|
|
ImplId(ImplId),
|
|
|
|
// enum variants cannot have generics themselves, but their parent enums
|
|
|
|
// can, and this makes some code easier to write
|
|
|
|
EnumVariantId(EnumVariantId),
|
|
|
|
// consts can have type parameters from their parents (i.e. associated consts of traits)
|
|
|
|
ConstId(ConstId),
|
|
|
|
}
|
2020-07-13 14:16:53 +00:00
|
|
|
impl_from!(
|
|
|
|
FunctionId,
|
2019-11-20 09:25:02 +00:00
|
|
|
AdtId(StructId, EnumId, UnionId),
|
|
|
|
TraitId,
|
|
|
|
TypeAliasId,
|
|
|
|
ImplId,
|
|
|
|
EnumVariantId,
|
|
|
|
ConstId
|
2020-07-13 14:16:53 +00:00
|
|
|
for GenericDefId
|
2019-11-20 09:25:02 +00:00
|
|
|
);
|
2019-11-20 13:03:59 +00:00
|
|
|
|
2019-11-27 09:31:40 +00:00
|
|
|
impl From<AssocItemId> for GenericDefId {
|
|
|
|
fn from(item: AssocItemId) -> Self {
|
|
|
|
match item {
|
|
|
|
AssocItemId::FunctionId(f) => f.into(),
|
|
|
|
AssocItemId::ConstId(c) => c.into(),
|
|
|
|
AssocItemId::TypeAliasId(t) => t.into(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-11-23 08:14:10 +00:00
|
|
|
#[derive(Clone, Copy, Debug, PartialEq, Eq, Hash)]
|
|
|
|
pub enum AttrDefId {
|
|
|
|
ModuleId(ModuleId),
|
2020-04-25 12:23:34 +00:00
|
|
|
FieldId(FieldId),
|
2019-11-23 08:14:10 +00:00
|
|
|
AdtId(AdtId),
|
|
|
|
FunctionId(FunctionId),
|
|
|
|
EnumVariantId(EnumVariantId),
|
|
|
|
StaticId(StaticId),
|
|
|
|
ConstId(ConstId),
|
|
|
|
TraitId(TraitId),
|
|
|
|
TypeAliasId(TypeAliasId),
|
|
|
|
MacroDefId(MacroDefId),
|
2019-11-23 09:01:56 +00:00
|
|
|
ImplId(ImplId),
|
2021-01-01 23:42:07 +00:00
|
|
|
GenericParamId(GenericParamId),
|
2019-11-23 08:14:10 +00:00
|
|
|
}
|
|
|
|
|
2020-07-13 14:16:53 +00:00
|
|
|
impl_from!(
|
|
|
|
ModuleId,
|
2020-04-25 12:23:34 +00:00
|
|
|
FieldId,
|
2019-11-23 08:14:10 +00:00
|
|
|
AdtId(StructId, EnumId, UnionId),
|
|
|
|
EnumVariantId,
|
|
|
|
StaticId,
|
|
|
|
ConstId,
|
|
|
|
FunctionId,
|
|
|
|
TraitId,
|
|
|
|
TypeAliasId,
|
2019-11-23 09:01:56 +00:00
|
|
|
MacroDefId,
|
2021-01-01 23:42:07 +00:00
|
|
|
ImplId,
|
|
|
|
GenericParamId
|
2020-07-13 14:16:53 +00:00
|
|
|
for AttrDefId
|
2019-11-23 08:14:10 +00:00
|
|
|
);
|
|
|
|
|
2021-04-08 11:37:34 +00:00
|
|
|
impl From<AssocContainerId> for AttrDefId {
|
|
|
|
fn from(acid: AssocContainerId) -> Self {
|
|
|
|
match acid {
|
|
|
|
AssocContainerId::ModuleId(mid) => AttrDefId::ModuleId(mid),
|
|
|
|
AssocContainerId::ImplId(iid) => AttrDefId::ImplId(iid),
|
|
|
|
AssocContainerId::TraitId(tid) => AttrDefId::TraitId(tid),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-11-24 20:48:39 +00:00
|
|
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
|
|
|
|
pub enum VariantId {
|
|
|
|
EnumVariantId(EnumVariantId),
|
|
|
|
StructId(StructId),
|
2019-11-25 14:34:15 +00:00
|
|
|
UnionId(UnionId),
|
2019-11-24 20:48:39 +00:00
|
|
|
}
|
2020-07-13 14:16:53 +00:00
|
|
|
impl_from!(EnumVariantId, StructId, UnionId for VariantId);
|
2019-11-24 20:48:39 +00:00
|
|
|
|
2021-04-06 15:59:18 +00:00
|
|
|
impl VariantId {
|
|
|
|
pub fn variant_data(self, db: &dyn db::DefDatabase) -> Arc<VariantData> {
|
|
|
|
match self {
|
|
|
|
VariantId::StructId(it) => db.struct_data(it).variant_data.clone(),
|
|
|
|
VariantId::UnionId(it) => db.union_data(it).variant_data.clone(),
|
|
|
|
VariantId::EnumVariantId(it) => {
|
|
|
|
db.enum_data(it.parent).variants[it.local_id].variant_data.clone()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-04-06 20:25:44 +00:00
|
|
|
|
|
|
|
pub fn file_id(self, db: &dyn db::DefDatabase) -> HirFileId {
|
|
|
|
match self {
|
|
|
|
VariantId::EnumVariantId(it) => it.parent.lookup(db).id.file_id(),
|
|
|
|
VariantId::StructId(it) => it.lookup(db).id.file_id(),
|
|
|
|
VariantId::UnionId(it) => it.lookup(db).id.file_id(),
|
|
|
|
}
|
|
|
|
}
|
2021-05-23 16:10:40 +00:00
|
|
|
|
|
|
|
pub fn adt_id(self) -> AdtId {
|
|
|
|
match self {
|
|
|
|
VariantId::EnumVariantId(it) => it.parent.into(),
|
|
|
|
VariantId::StructId(it) => it.into(),
|
|
|
|
VariantId::UnionId(it) => it.into(),
|
|
|
|
}
|
|
|
|
}
|
2021-04-06 15:59:18 +00:00
|
|
|
}
|
|
|
|
|
2019-11-20 13:03:59 +00:00
|
|
|
trait Intern {
|
|
|
|
type ID;
|
2020-03-13 15:05:46 +00:00
|
|
|
fn intern(self, db: &dyn db::DefDatabase) -> Self::ID;
|
2019-11-20 13:03:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub trait Lookup {
|
|
|
|
type Data;
|
2020-03-13 15:05:46 +00:00
|
|
|
fn lookup(&self, db: &dyn db::DefDatabase) -> Self::Data;
|
2019-11-20 13:03:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub trait HasModule {
|
2020-03-13 15:05:46 +00:00
|
|
|
fn module(&self, db: &dyn db::DefDatabase) -> ModuleId;
|
2019-11-20 13:03:59 +00:00
|
|
|
}
|
|
|
|
|
2019-12-20 10:59:50 +00:00
|
|
|
impl HasModule for AssocContainerId {
|
2020-03-13 15:05:46 +00:00
|
|
|
fn module(&self, db: &dyn db::DefDatabase) -> ModuleId {
|
2019-12-19 17:12:46 +00:00
|
|
|
match *self {
|
2021-03-09 17:27:16 +00:00
|
|
|
AssocContainerId::ModuleId(it) => it,
|
2021-03-09 18:09:02 +00:00
|
|
|
AssocContainerId::ImplId(it) => it.lookup(db).container,
|
|
|
|
AssocContainerId::TraitId(it) => it.lookup(db).container,
|
2019-11-20 13:03:59 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-06-22 13:07:06 +00:00
|
|
|
impl<N: ItemTreeNode> HasModule for AssocItemLoc<N> {
|
2020-03-13 15:05:46 +00:00
|
|
|
fn module(&self, db: &dyn db::DefDatabase) -> ModuleId {
|
2019-12-19 17:12:46 +00:00
|
|
|
self.container.module(db)
|
2019-11-20 15:00:01 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-11-24 19:47:58 +00:00
|
|
|
impl HasModule for AdtId {
|
2020-03-13 15:05:46 +00:00
|
|
|
fn module(&self, db: &dyn db::DefDatabase) -> ModuleId {
|
2019-11-24 19:47:58 +00:00
|
|
|
match self {
|
2019-12-12 13:58:04 +00:00
|
|
|
AdtId::StructId(it) => it.lookup(db).container,
|
2019-12-12 14:11:57 +00:00
|
|
|
AdtId::UnionId(it) => it.lookup(db).container,
|
|
|
|
AdtId::EnumId(it) => it.lookup(db).container,
|
2019-11-24 19:47:58 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-12-17 13:29:29 +00:00
|
|
|
impl HasModule for VariantId {
|
|
|
|
fn module(&self, db: &dyn db::DefDatabase) -> ModuleId {
|
|
|
|
match self {
|
2021-03-09 18:09:02 +00:00
|
|
|
VariantId::EnumVariantId(it) => it.parent.lookup(db).container,
|
|
|
|
VariantId::StructId(it) => it.lookup(db).container,
|
|
|
|
VariantId::UnionId(it) => it.lookup(db).container,
|
2020-12-17 13:29:29 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-11-26 11:02:57 +00:00
|
|
|
impl HasModule for DefWithBodyId {
|
2020-03-13 15:05:46 +00:00
|
|
|
fn module(&self, db: &dyn db::DefDatabase) -> ModuleId {
|
2019-11-26 11:02:57 +00:00
|
|
|
match self {
|
|
|
|
DefWithBodyId::FunctionId(it) => it.lookup(db).module(db),
|
|
|
|
DefWithBodyId::StaticId(it) => it.lookup(db).module(db),
|
|
|
|
DefWithBodyId::ConstId(it) => it.lookup(db).module(db),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-06-22 13:07:06 +00:00
|
|
|
impl DefWithBodyId {
|
|
|
|
pub fn as_mod_item(self, db: &dyn db::DefDatabase) -> ModItem {
|
|
|
|
match self {
|
|
|
|
DefWithBodyId::FunctionId(it) => it.lookup(db).id.value.into(),
|
|
|
|
DefWithBodyId::StaticId(it) => it.lookup(db).id.value.into(),
|
|
|
|
DefWithBodyId::ConstId(it) => it.lookup(db).id.value.into(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-12-07 18:52:09 +00:00
|
|
|
impl HasModule for GenericDefId {
|
2020-03-13 15:05:46 +00:00
|
|
|
fn module(&self, db: &dyn db::DefDatabase) -> ModuleId {
|
2019-12-07 18:52:09 +00:00
|
|
|
match self {
|
|
|
|
GenericDefId::FunctionId(it) => it.lookup(db).module(db),
|
|
|
|
GenericDefId::AdtId(it) => it.module(db),
|
2021-03-09 18:09:02 +00:00
|
|
|
GenericDefId::TraitId(it) => it.lookup(db).container,
|
2019-12-07 18:52:09 +00:00
|
|
|
GenericDefId::TypeAliasId(it) => it.lookup(db).module(db),
|
2021-03-09 18:09:02 +00:00
|
|
|
GenericDefId::ImplId(it) => it.lookup(db).container,
|
|
|
|
GenericDefId::EnumVariantId(it) => it.parent.lookup(db).container,
|
2019-12-07 18:52:09 +00:00
|
|
|
GenericDefId::ConstId(it) => it.lookup(db).module(db),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-11-24 12:13:56 +00:00
|
|
|
impl HasModule for StaticLoc {
|
2021-03-09 18:09:02 +00:00
|
|
|
fn module(&self, _db: &dyn db::DefDatabase) -> ModuleId {
|
|
|
|
self.container
|
2019-11-24 12:13:56 +00:00
|
|
|
}
|
|
|
|
}
|
2020-02-17 04:57:24 +00:00
|
|
|
|
2021-03-01 18:36:34 +00:00
|
|
|
impl ModuleDefId {
|
|
|
|
/// Returns the module containing `self` (or `self`, if `self` is itself a module).
|
|
|
|
///
|
|
|
|
/// Returns `None` if `self` refers to a primitive type.
|
|
|
|
pub fn module(&self, db: &dyn db::DefDatabase) -> Option<ModuleId> {
|
|
|
|
Some(match self {
|
|
|
|
ModuleDefId::ModuleId(id) => *id,
|
|
|
|
ModuleDefId::FunctionId(id) => id.lookup(db).module(db),
|
|
|
|
ModuleDefId::AdtId(id) => id.module(db),
|
2021-03-09 18:09:02 +00:00
|
|
|
ModuleDefId::EnumVariantId(id) => id.parent.lookup(db).container,
|
2021-03-01 18:36:34 +00:00
|
|
|
ModuleDefId::ConstId(id) => id.lookup(db).container.module(db),
|
2021-03-09 18:09:02 +00:00
|
|
|
ModuleDefId::StaticId(id) => id.lookup(db).container,
|
|
|
|
ModuleDefId::TraitId(id) => id.lookup(db).container,
|
2021-03-01 18:36:34 +00:00
|
|
|
ModuleDefId::TypeAliasId(id) => id.lookup(db).module(db),
|
|
|
|
ModuleDefId::BuiltinType(_) => return None,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-12-17 13:29:29 +00:00
|
|
|
impl AttrDefId {
|
|
|
|
pub fn krate(&self, db: &dyn db::DefDatabase) -> CrateId {
|
|
|
|
match self {
|
|
|
|
AttrDefId::ModuleId(it) => it.krate,
|
|
|
|
AttrDefId::FieldId(it) => it.parent.module(db).krate,
|
|
|
|
AttrDefId::AdtId(it) => it.module(db).krate,
|
|
|
|
AttrDefId::FunctionId(it) => it.lookup(db).module(db).krate,
|
2021-03-09 18:09:02 +00:00
|
|
|
AttrDefId::EnumVariantId(it) => it.parent.lookup(db).container.krate,
|
2020-12-17 13:29:29 +00:00
|
|
|
AttrDefId::StaticId(it) => it.lookup(db).module(db).krate,
|
|
|
|
AttrDefId::ConstId(it) => it.lookup(db).module(db).krate,
|
2021-03-09 18:09:02 +00:00
|
|
|
AttrDefId::TraitId(it) => it.lookup(db).container.krate,
|
2020-12-17 13:29:29 +00:00
|
|
|
AttrDefId::TypeAliasId(it) => it.lookup(db).module(db).krate,
|
2021-03-09 18:09:02 +00:00
|
|
|
AttrDefId::ImplId(it) => it.lookup(db).container.krate,
|
2021-01-01 23:42:07 +00:00
|
|
|
AttrDefId::GenericParamId(it) => {
|
|
|
|
match it {
|
|
|
|
GenericParamId::TypeParamId(it) => it.parent,
|
|
|
|
GenericParamId::LifetimeParamId(it) => it.parent,
|
|
|
|
GenericParamId::ConstParamId(it) => it.parent,
|
|
|
|
}
|
|
|
|
.module(db)
|
|
|
|
.krate
|
|
|
|
}
|
2020-12-17 13:29:29 +00:00
|
|
|
// FIXME: `MacroDefId` should store the defining module, then this can implement
|
|
|
|
// `HasModule`
|
|
|
|
AttrDefId::MacroDefId(it) => it.krate,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-02-17 04:57:24 +00:00
|
|
|
/// A helper trait for converting to MacroCallId
|
|
|
|
pub trait AsMacroCall {
|
|
|
|
fn as_call_id(
|
|
|
|
&self,
|
2020-03-13 15:05:46 +00:00
|
|
|
db: &dyn db::DefDatabase,
|
2020-06-11 10:08:24 +00:00
|
|
|
krate: CrateId,
|
2020-02-17 04:57:24 +00:00
|
|
|
resolver: impl Fn(path::ModPath) -> Option<MacroDefId>,
|
2020-12-02 15:52:14 +00:00
|
|
|
) -> Option<MacroCallId> {
|
2021-03-17 06:31:14 +00:00
|
|
|
self.as_call_id_with_errors(db, krate, resolver, &mut |_| ()).ok()?.ok()
|
2020-12-02 15:52:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
fn as_call_id_with_errors(
|
|
|
|
&self,
|
|
|
|
db: &dyn db::DefDatabase,
|
|
|
|
krate: CrateId,
|
|
|
|
resolver: impl Fn(path::ModPath) -> Option<MacroDefId>,
|
|
|
|
error_sink: &mut dyn FnMut(mbe::ExpandError),
|
2021-03-17 06:31:14 +00:00
|
|
|
) -> Result<Result<MacroCallId, ErrorEmitted>, UnresolvedMacro>;
|
2020-02-17 04:57:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl AsMacroCall for InFile<&ast::MacroCall> {
|
2020-12-02 15:52:14 +00:00
|
|
|
fn as_call_id_with_errors(
|
2020-02-17 04:57:24 +00:00
|
|
|
&self,
|
2020-03-13 15:05:46 +00:00
|
|
|
db: &dyn db::DefDatabase,
|
2020-06-11 10:08:24 +00:00
|
|
|
krate: CrateId,
|
2020-02-17 04:57:24 +00:00
|
|
|
resolver: impl Fn(path::ModPath) -> Option<MacroDefId>,
|
2021-03-17 06:31:14 +00:00
|
|
|
mut error_sink: &mut dyn FnMut(mbe::ExpandError),
|
|
|
|
) -> Result<Result<MacroCallId, ErrorEmitted>, UnresolvedMacro> {
|
2021-05-08 23:36:06 +00:00
|
|
|
let fragment = hir_expand::to_fragment_kind(self.value);
|
2020-02-17 04:57:24 +00:00
|
|
|
let ast_id = AstId::new(self.file_id, db.ast_id_map(self.file_id).ast_id(self.value));
|
2020-03-13 15:05:46 +00:00
|
|
|
let h = Hygiene::new(db.upcast(), self.file_id);
|
2021-05-06 17:59:54 +00:00
|
|
|
let path = self.value.path().and_then(|path| path::ModPath::from_src(db, path, &h));
|
2020-12-02 15:52:14 +00:00
|
|
|
|
2021-03-17 06:31:14 +00:00
|
|
|
let path = match error_sink
|
|
|
|
.option(path, || mbe::ExpandError::Other("malformed macro invocation".into()))
|
|
|
|
{
|
|
|
|
Ok(path) => path,
|
|
|
|
Err(error) => {
|
|
|
|
return Ok(Err(error));
|
2021-03-16 07:46:57 +00:00
|
|
|
}
|
|
|
|
};
|
2020-02-17 04:57:24 +00:00
|
|
|
|
2021-02-28 11:12:11 +00:00
|
|
|
macro_call_as_call_id(
|
2021-03-16 07:46:57 +00:00
|
|
|
&AstIdWithPath::new(ast_id.file_id, ast_id.value, path),
|
2021-05-08 23:36:06 +00:00
|
|
|
fragment,
|
2021-02-28 11:12:11 +00:00
|
|
|
db,
|
|
|
|
krate,
|
|
|
|
resolver,
|
|
|
|
error_sink,
|
|
|
|
)
|
2020-02-17 04:57:24 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Helper wrapper for `AstId` with `ModPath`
|
|
|
|
#[derive(Clone, Debug, Eq, PartialEq)]
|
|
|
|
struct AstIdWithPath<T: ast::AstNode> {
|
2020-11-02 12:13:32 +00:00
|
|
|
ast_id: AstId<T>,
|
|
|
|
path: path::ModPath,
|
2020-02-17 04:57:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl<T: ast::AstNode> AstIdWithPath<T> {
|
2020-11-02 12:13:32 +00:00
|
|
|
fn new(file_id: HirFileId, ast_id: FileAstId<T>, path: path::ModPath) -> AstIdWithPath<T> {
|
2020-02-17 04:57:24 +00:00
|
|
|
AstIdWithPath { ast_id: AstId::new(file_id, ast_id), path }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-04-10 15:49:12 +00:00
|
|
|
#[derive(Debug)]
|
2021-04-16 13:48:03 +00:00
|
|
|
pub struct UnresolvedMacro {
|
|
|
|
pub path: ModPath,
|
|
|
|
}
|
2021-02-28 11:12:11 +00:00
|
|
|
|
|
|
|
fn macro_call_as_call_id(
|
|
|
|
call: &AstIdWithPath<ast::MacroCall>,
|
2021-05-08 23:36:06 +00:00
|
|
|
fragment: FragmentKind,
|
2021-02-28 11:12:11 +00:00
|
|
|
db: &dyn db::DefDatabase,
|
|
|
|
krate: CrateId,
|
|
|
|
resolver: impl Fn(path::ModPath) -> Option<MacroDefId>,
|
|
|
|
error_sink: &mut dyn FnMut(mbe::ExpandError),
|
|
|
|
) -> Result<Result<MacroCallId, ErrorEmitted>, UnresolvedMacro> {
|
2021-04-16 13:48:03 +00:00
|
|
|
let def: MacroDefId =
|
|
|
|
resolver(call.path.clone()).ok_or_else(|| UnresolvedMacro { path: call.path.clone() })?;
|
2021-02-28 11:12:11 +00:00
|
|
|
|
2021-03-18 14:37:14 +00:00
|
|
|
let res = if let MacroDefKind::BuiltInEager(..) = def.kind {
|
2021-02-28 11:12:11 +00:00
|
|
|
let macro_call = InFile::new(call.ast_id.file_id, call.ast_id.to_node(db.upcast()));
|
|
|
|
let hygiene = Hygiene::new(db.upcast(), call.ast_id.file_id);
|
|
|
|
|
|
|
|
expand_eager_macro(
|
|
|
|
db.upcast(),
|
|
|
|
krate,
|
|
|
|
macro_call,
|
|
|
|
def,
|
2021-05-06 17:59:54 +00:00
|
|
|
&|path: ast::Path| resolver(path::ModPath::from_src(db, path, &hygiene)?),
|
2021-02-28 11:12:11 +00:00
|
|
|
error_sink,
|
|
|
|
)
|
|
|
|
.map(MacroCallId::from)
|
|
|
|
} else {
|
2021-04-08 18:43:07 +00:00
|
|
|
Ok(def
|
2021-05-08 23:36:06 +00:00
|
|
|
.as_lazy_macro(
|
|
|
|
db.upcast(),
|
|
|
|
krate,
|
|
|
|
MacroCallKind::FnLike { ast_id: call.ast_id, fragment },
|
|
|
|
)
|
2021-04-08 18:43:07 +00:00
|
|
|
.into())
|
2021-02-28 11:12:11 +00:00
|
|
|
};
|
|
|
|
Ok(res)
|
2020-02-17 04:57:24 +00:00
|
|
|
}
|
|
|
|
|
2021-03-19 13:23:13 +00:00
|
|
|
fn derive_macro_as_call_id(
|
2021-02-28 11:12:11 +00:00
|
|
|
item_attr: &AstIdWithPath<ast::Item>,
|
2021-04-09 11:38:01 +00:00
|
|
|
derive_attr: AttrId,
|
2021-02-28 11:12:11 +00:00
|
|
|
db: &dyn db::DefDatabase,
|
|
|
|
krate: CrateId,
|
|
|
|
resolver: impl Fn(path::ModPath) -> Option<MacroDefId>,
|
|
|
|
) -> Result<MacroCallId, UnresolvedMacro> {
|
2021-04-16 13:48:03 +00:00
|
|
|
let def: MacroDefId = resolver(item_attr.path.clone())
|
|
|
|
.ok_or_else(|| UnresolvedMacro { path: item_attr.path.clone() })?;
|
|
|
|
let last_segment = item_attr
|
|
|
|
.path
|
|
|
|
.segments()
|
|
|
|
.last()
|
|
|
|
.ok_or_else(|| UnresolvedMacro { path: item_attr.path.clone() })?;
|
2021-02-28 11:12:11 +00:00
|
|
|
let res = def
|
|
|
|
.as_lazy_macro(
|
|
|
|
db.upcast(),
|
|
|
|
krate,
|
2021-04-08 18:43:07 +00:00
|
|
|
MacroCallKind::Derive {
|
|
|
|
ast_id: item_attr.ast_id,
|
|
|
|
derive_name: last_segment.to_string(),
|
2021-05-10 19:50:42 +00:00
|
|
|
derive_attr_index: derive_attr.ast_index,
|
2021-04-08 18:43:07 +00:00
|
|
|
},
|
2020-03-18 09:47:59 +00:00
|
|
|
)
|
2021-02-28 11:12:11 +00:00
|
|
|
.into();
|
|
|
|
Ok(res)
|
2020-02-17 04:57:24 +00:00
|
|
|
}
|
2021-05-31 11:37:11 +00:00
|
|
|
|
|
|
|
fn attr_macro_as_call_id(
|
|
|
|
item_attr: &AstIdWithPath<ast::Item>,
|
|
|
|
macro_attr: &Attr,
|
|
|
|
db: &dyn db::DefDatabase,
|
|
|
|
krate: CrateId,
|
|
|
|
resolver: impl Fn(path::ModPath) -> Option<MacroDefId>,
|
|
|
|
) -> Result<MacroCallId, UnresolvedMacro> {
|
|
|
|
let def: MacroDefId = resolver(item_attr.path.clone())
|
|
|
|
.ok_or_else(|| UnresolvedMacro { path: item_attr.path.clone() })?;
|
|
|
|
let last_segment = item_attr
|
|
|
|
.path
|
|
|
|
.segments()
|
|
|
|
.last()
|
|
|
|
.ok_or_else(|| UnresolvedMacro { path: item_attr.path.clone() })?;
|
|
|
|
let mut arg = match ¯o_attr.input {
|
|
|
|
Some(input) => match &**input {
|
|
|
|
attr::AttrInput::Literal(_) => tt::Subtree::default(),
|
|
|
|
attr::AttrInput::TokenTree(tt) => tt.clone(),
|
|
|
|
},
|
|
|
|
None => tt::Subtree::default(),
|
|
|
|
};
|
|
|
|
// The parentheses are always disposed here.
|
|
|
|
arg.delimiter = None;
|
|
|
|
|
|
|
|
let res = def
|
|
|
|
.as_lazy_macro(
|
|
|
|
db.upcast(),
|
|
|
|
krate,
|
|
|
|
MacroCallKind::Attr {
|
|
|
|
ast_id: item_attr.ast_id,
|
|
|
|
attr_name: last_segment.to_string(),
|
|
|
|
attr_args: arg,
|
|
|
|
invoc_attr_index: macro_attr.id.ast_index,
|
|
|
|
},
|
|
|
|
)
|
|
|
|
.into();
|
|
|
|
Ok(res)
|
|
|
|
}
|