]> git.lizzy.rs Git - rust.git/blob - crates/hir_def/src/db.rs
Merge #11866
[rust.git] / crates / hir_def / src / db.rs
1 //! Defines database & queries for name resolution.
2 use std::sync::Arc;
3
4 use base_db::{salsa, CrateId, SourceDatabase, Upcast};
5 use either::Either;
6 use hir_expand::{db::AstDatabase, HirFileId};
7 use la_arena::ArenaMap;
8 use syntax::{ast, AstPtr, SmolStr};
9
10 use crate::{
11     adt::{EnumData, StructData},
12     attr::{Attrs, AttrsWithOwner},
13     body::{scope::ExprScopes, Body, BodySourceMap},
14     data::{
15         ConstData, FunctionData, ImplData, Macro2Data, MacroRulesData, ProcMacroData, StaticData,
16         TraitData, TypeAliasData,
17     },
18     generics::GenericParams,
19     import_map::ImportMap,
20     intern::Interned,
21     item_tree::ItemTree,
22     lang_item::{LangItemTarget, LangItems},
23     nameres::DefMap,
24     visibility::{self, Visibility},
25     AttrDefId, BlockId, BlockLoc, ConstId, ConstLoc, DefWithBodyId, EnumId, EnumLoc, ExternBlockId,
26     ExternBlockLoc, FunctionId, FunctionLoc, GenericDefId, ImplId, ImplLoc, LocalEnumVariantId,
27     LocalFieldId, Macro2Id, Macro2Loc, MacroRulesId, MacroRulesLoc, ProcMacroId, ProcMacroLoc,
28     StaticId, StaticLoc, StructId, StructLoc, TraitId, TraitLoc, TypeAliasId, TypeAliasLoc,
29     UnionId, UnionLoc, VariantId,
30 };
31
32 #[salsa::query_group(InternDatabaseStorage)]
33 pub trait InternDatabase: SourceDatabase {
34     #[salsa::interned]
35     fn intern_function(&self, loc: FunctionLoc) -> FunctionId;
36     #[salsa::interned]
37     fn intern_struct(&self, loc: StructLoc) -> StructId;
38     #[salsa::interned]
39     fn intern_union(&self, loc: UnionLoc) -> UnionId;
40     #[salsa::interned]
41     fn intern_enum(&self, loc: EnumLoc) -> EnumId;
42     #[salsa::interned]
43     fn intern_const(&self, loc: ConstLoc) -> ConstId;
44     #[salsa::interned]
45     fn intern_static(&self, loc: StaticLoc) -> StaticId;
46     #[salsa::interned]
47     fn intern_trait(&self, loc: TraitLoc) -> TraitId;
48     #[salsa::interned]
49     fn intern_type_alias(&self, loc: TypeAliasLoc) -> TypeAliasId;
50     #[salsa::interned]
51     fn intern_impl(&self, loc: ImplLoc) -> ImplId;
52     #[salsa::interned]
53     fn intern_extern_block(&self, loc: ExternBlockLoc) -> ExternBlockId;
54     #[salsa::interned]
55     fn intern_block(&self, loc: BlockLoc) -> BlockId;
56     #[salsa::interned]
57     fn intern_macro2(&self, loc: Macro2Loc) -> Macro2Id;
58     #[salsa::interned]
59     fn intern_proc_macro(&self, loc: ProcMacroLoc) -> ProcMacroId;
60     #[salsa::interned]
61     fn intern_macro_rules(&self, loc: MacroRulesLoc) -> MacroRulesId;
62 }
63
64 #[salsa::query_group(DefDatabaseStorage)]
65 pub trait DefDatabase: InternDatabase + AstDatabase + Upcast<dyn AstDatabase> {
66     #[salsa::input]
67     fn enable_proc_attr_macros(&self) -> bool;
68
69     #[salsa::invoke(ItemTree::file_item_tree_query)]
70     fn file_item_tree(&self, file_id: HirFileId) -> Arc<ItemTree>;
71
72     #[salsa::invoke(crate_def_map_wait)]
73     #[salsa::transparent]
74     fn crate_def_map(&self, krate: CrateId) -> Arc<DefMap>;
75
76     #[salsa::invoke(DefMap::crate_def_map_query)]
77     fn crate_def_map_query(&self, krate: CrateId) -> Arc<DefMap>;
78
79     /// Computes the block-level `DefMap`, returning `None` when `block` doesn't contain any inner
80     /// items directly.
81     ///
82     /// For example:
83     ///
84     /// ```
85     /// fn f() { // (0)
86     ///     { // (1)
87     ///         fn inner() {}
88     ///     }
89     /// }
90     /// ```
91     ///
92     /// The `block_def_map` for block 0 would return `None`, while `block_def_map` of block 1 would
93     /// return a `DefMap` containing `inner`.
94     #[salsa::invoke(DefMap::block_def_map_query)]
95     fn block_def_map(&self, block: BlockId) -> Option<Arc<DefMap>>;
96
97     #[salsa::invoke(StructData::struct_data_query)]
98     fn struct_data(&self, id: StructId) -> Arc<StructData>;
99
100     #[salsa::invoke(StructData::union_data_query)]
101     fn union_data(&self, id: UnionId) -> Arc<StructData>;
102
103     #[salsa::invoke(EnumData::enum_data_query)]
104     fn enum_data(&self, e: EnumId) -> Arc<EnumData>;
105
106     #[salsa::invoke(ImplData::impl_data_query)]
107     fn impl_data(&self, e: ImplId) -> Arc<ImplData>;
108
109     #[salsa::invoke(TraitData::trait_data_query)]
110     fn trait_data(&self, e: TraitId) -> Arc<TraitData>;
111
112     #[salsa::invoke(TypeAliasData::type_alias_data_query)]
113     fn type_alias_data(&self, e: TypeAliasId) -> Arc<TypeAliasData>;
114
115     #[salsa::invoke(FunctionData::fn_data_query)]
116     fn function_data(&self, func: FunctionId) -> Arc<FunctionData>;
117
118     #[salsa::invoke(ConstData::const_data_query)]
119     fn const_data(&self, konst: ConstId) -> Arc<ConstData>;
120
121     #[salsa::invoke(StaticData::static_data_query)]
122     fn static_data(&self, konst: StaticId) -> Arc<StaticData>;
123
124     #[salsa::invoke(Macro2Data::macro2_data_query)]
125     fn macro2_data(&self, makro: Macro2Id) -> Arc<Macro2Data>;
126
127     #[salsa::invoke(MacroRulesData::macro_rules_data_query)]
128     fn macro_rules_data(&self, makro: MacroRulesId) -> Arc<MacroRulesData>;
129
130     #[salsa::invoke(ProcMacroData::proc_macro_data_query)]
131     fn proc_macro_data(&self, makro: ProcMacroId) -> Arc<ProcMacroData>;
132
133     #[salsa::invoke(Body::body_with_source_map_query)]
134     fn body_with_source_map(&self, def: DefWithBodyId) -> (Arc<Body>, Arc<BodySourceMap>);
135
136     #[salsa::invoke(Body::body_query)]
137     fn body(&self, def: DefWithBodyId) -> Arc<Body>;
138
139     #[salsa::invoke(ExprScopes::expr_scopes_query)]
140     fn expr_scopes(&self, def: DefWithBodyId) -> Arc<ExprScopes>;
141
142     #[salsa::invoke(GenericParams::generic_params_query)]
143     fn generic_params(&self, def: GenericDefId) -> Interned<GenericParams>;
144
145     #[salsa::invoke(Attrs::variants_attrs_query)]
146     fn variants_attrs(&self, def: EnumId) -> Arc<ArenaMap<LocalEnumVariantId, Attrs>>;
147
148     #[salsa::invoke(Attrs::fields_attrs_query)]
149     fn fields_attrs(&self, def: VariantId) -> Arc<ArenaMap<LocalFieldId, Attrs>>;
150
151     #[salsa::invoke(crate::attr::variants_attrs_source_map)]
152     fn variants_attrs_source_map(
153         &self,
154         def: EnumId,
155     ) -> Arc<ArenaMap<LocalEnumVariantId, AstPtr<ast::Variant>>>;
156
157     #[salsa::invoke(crate::attr::fields_attrs_source_map)]
158     fn fields_attrs_source_map(
159         &self,
160         def: VariantId,
161     ) -> Arc<ArenaMap<LocalFieldId, Either<AstPtr<ast::TupleField>, AstPtr<ast::RecordField>>>>;
162
163     #[salsa::invoke(AttrsWithOwner::attrs_query)]
164     fn attrs(&self, def: AttrDefId) -> AttrsWithOwner;
165
166     #[salsa::invoke(LangItems::crate_lang_items_query)]
167     fn crate_lang_items(&self, krate: CrateId) -> Arc<LangItems>;
168
169     #[salsa::invoke(LangItems::lang_item_query)]
170     fn lang_item(&self, start_crate: CrateId, item: SmolStr) -> Option<LangItemTarget>;
171
172     #[salsa::invoke(ImportMap::import_map_query)]
173     fn import_map(&self, krate: CrateId) -> Arc<ImportMap>;
174
175     #[salsa::invoke(visibility::field_visibilities_query)]
176     fn field_visibilities(&self, var: VariantId) -> Arc<ArenaMap<LocalFieldId, Visibility>>;
177
178     // FIXME: unify function_visibility and const_visibility?
179     #[salsa::invoke(visibility::function_visibility_query)]
180     fn function_visibility(&self, def: FunctionId) -> Visibility;
181
182     #[salsa::invoke(visibility::const_visibility_query)]
183     fn const_visibility(&self, def: ConstId) -> Visibility;
184
185     #[salsa::transparent]
186     fn crate_limits(&self, crate_id: CrateId) -> CrateLimits;
187 }
188
189 fn crate_def_map_wait(db: &dyn DefDatabase, krate: CrateId) -> Arc<DefMap> {
190     let _p = profile::span("crate_def_map:wait");
191     db.crate_def_map_query(krate)
192 }
193
194 pub struct CrateLimits {
195     /// The maximum depth for potentially infinitely-recursive compile-time operations like macro expansion or auto-dereference.
196     pub recursion_limit: u32,
197 }
198
199 fn crate_limits(db: &dyn DefDatabase, crate_id: CrateId) -> CrateLimits {
200     let def_map = db.crate_def_map(crate_id);
201
202     CrateLimits {
203         // 128 is the default in rustc.
204         recursion_limit: def_map.recursion_limit().unwrap_or(128),
205     }
206 }