aboutsummaryrefslogtreecommitdiff
path: root/crates/ra_hir_def/src/lang_item.rs
diff options
context:
space:
mode:
Diffstat (limited to 'crates/ra_hir_def/src/lang_item.rs')
-rw-r--r--crates/ra_hir_def/src/lang_item.rs175
1 files changed, 0 insertions, 175 deletions
diff --git a/crates/ra_hir_def/src/lang_item.rs b/crates/ra_hir_def/src/lang_item.rs
deleted file mode 100644
index 3516784b8..000000000
--- a/crates/ra_hir_def/src/lang_item.rs
+++ /dev/null
@@ -1,175 +0,0 @@
1//! Collects lang items: items marked with `#[lang = "..."]` attribute.
2//!
3//! This attribute to tell the compiler about semi built-in std library
4//! features, such as Fn family of traits.
5use std::sync::Arc;
6
7use ra_prof::profile;
8use ra_syntax::SmolStr;
9use rustc_hash::FxHashMap;
10
11use crate::{
12 db::DefDatabase, AdtId, AttrDefId, CrateId, EnumId, FunctionId, ImplId, ModuleDefId, ModuleId,
13 StaticId, StructId, TraitId,
14};
15
16#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
17pub enum LangItemTarget {
18 EnumId(EnumId),
19 FunctionId(FunctionId),
20 ImplDefId(ImplId),
21 StaticId(StaticId),
22 StructId(StructId),
23 TraitId(TraitId),
24}
25
26impl LangItemTarget {
27 pub fn as_enum(self) -> Option<EnumId> {
28 match self {
29 LangItemTarget::EnumId(id) => Some(id),
30 _ => None,
31 }
32 }
33
34 pub fn as_function(self) -> Option<FunctionId> {
35 match self {
36 LangItemTarget::FunctionId(id) => Some(id),
37 _ => None,
38 }
39 }
40
41 pub fn as_impl_def(self) -> Option<ImplId> {
42 match self {
43 LangItemTarget::ImplDefId(id) => Some(id),
44 _ => None,
45 }
46 }
47
48 pub fn as_static(self) -> Option<StaticId> {
49 match self {
50 LangItemTarget::StaticId(id) => Some(id),
51 _ => None,
52 }
53 }
54
55 pub fn as_struct(self) -> Option<StructId> {
56 match self {
57 LangItemTarget::StructId(id) => Some(id),
58 _ => None,
59 }
60 }
61
62 pub fn as_trait(self) -> Option<TraitId> {
63 match self {
64 LangItemTarget::TraitId(id) => Some(id),
65 _ => None,
66 }
67 }
68}
69
70#[derive(Default, Debug, Clone, PartialEq, Eq)]
71pub struct LangItems {
72 items: FxHashMap<SmolStr, LangItemTarget>,
73}
74
75impl LangItems {
76 pub fn target(&self, item: &str) -> Option<LangItemTarget> {
77 self.items.get(item).copied()
78 }
79
80 /// Salsa query. This will look for lang items in a specific crate.
81 pub(crate) fn crate_lang_items_query(db: &dyn DefDatabase, krate: CrateId) -> Arc<LangItems> {
82 let _p = profile("crate_lang_items_query");
83
84 let mut lang_items = LangItems::default();
85
86 let crate_def_map = db.crate_def_map(krate);
87
88 crate_def_map
89 .modules
90 .iter()
91 .filter_map(|(local_id, _)| db.module_lang_items(ModuleId { krate, local_id }))
92 .for_each(|it| lang_items.items.extend(it.items.iter().map(|(k, v)| (k.clone(), *v))));
93
94 Arc::new(lang_items)
95 }
96
97 pub(crate) fn module_lang_items_query(
98 db: &dyn DefDatabase,
99 module: ModuleId,
100 ) -> Option<Arc<LangItems>> {
101 let _p = profile("module_lang_items_query");
102 let mut lang_items = LangItems::default();
103 lang_items.collect_lang_items(db, module);
104 if lang_items.items.is_empty() {
105 None
106 } else {
107 Some(Arc::new(lang_items))
108 }
109 }
110
111 /// Salsa query. Look for a lang item, starting from the specified crate and recursively
112 /// traversing its dependencies.
113 pub(crate) fn lang_item_query(
114 db: &dyn DefDatabase,
115 start_crate: CrateId,
116 item: SmolStr,
117 ) -> Option<LangItemTarget> {
118 let _p = profile("lang_item_query");
119 let lang_items = db.crate_lang_items(start_crate);
120 let start_crate_target = lang_items.items.get(&item);
121 if let Some(target) = start_crate_target {
122 return Some(*target);
123 }
124 db.crate_graph()[start_crate]
125 .dependencies
126 .iter()
127 .find_map(|dep| db.lang_item(dep.crate_id, item.clone()))
128 }
129
130 fn collect_lang_items(&mut self, db: &dyn DefDatabase, module: ModuleId) {
131 // Look for impl targets
132 let def_map = db.crate_def_map(module.krate);
133 let module_data = &def_map[module.local_id];
134 for impl_def in module_data.scope.impls() {
135 self.collect_lang_item(db, impl_def, LangItemTarget::ImplDefId)
136 }
137
138 for def in module_data.scope.declarations() {
139 match def {
140 ModuleDefId::TraitId(trait_) => {
141 self.collect_lang_item(db, trait_, LangItemTarget::TraitId)
142 }
143 ModuleDefId::AdtId(AdtId::EnumId(e)) => {
144 self.collect_lang_item(db, e, LangItemTarget::EnumId)
145 }
146 ModuleDefId::AdtId(AdtId::StructId(s)) => {
147 self.collect_lang_item(db, s, LangItemTarget::StructId)
148 }
149 ModuleDefId::FunctionId(f) => {
150 self.collect_lang_item(db, f, LangItemTarget::FunctionId)
151 }
152 ModuleDefId::StaticId(s) => self.collect_lang_item(db, s, LangItemTarget::StaticId),
153 _ => {}
154 }
155 }
156 }
157
158 fn collect_lang_item<T>(
159 &mut self,
160 db: &dyn DefDatabase,
161 item: T,
162 constructor: fn(T) -> LangItemTarget,
163 ) where
164 T: Into<AttrDefId> + Copy,
165 {
166 if let Some(lang_item_name) = lang_attr(db, item) {
167 self.items.entry(lang_item_name).or_insert_with(|| constructor(item));
168 }
169 }
170}
171
172pub fn lang_attr(db: &dyn DefDatabase, item: impl Into<AttrDefId> + Copy) -> Option<SmolStr> {
173 let attrs = db.attrs(item.into());
174 attrs.by_key("lang").string_value().cloned()
175}