1 //! Collects lang items: items marked with `#[lang = "..."]` attribute.
3 //! This attribute to tell the compiler about semi built-in std library
4 //! features, such as Fn family of traits.
7 use ra_syntax::SmolStr;
8 use rustc_hash::FxHashMap;
11 db::DefDatabase, AdtId, AttrDefId, CrateId, EnumId, FunctionId, ImplId, ModuleDefId, ModuleId,
12 StaticId, StructId, TraitId,
15 #[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
16 pub enum LangItemTarget {
18 FunctionId(FunctionId),
26 pub fn as_enum(self) -> Option<EnumId> {
28 LangItemTarget::EnumId(id) => Some(id),
33 pub fn as_function(self) -> Option<FunctionId> {
35 LangItemTarget::FunctionId(id) => Some(id),
40 pub fn as_impl_block(self) -> Option<ImplId> {
42 LangItemTarget::ImplBlockId(id) => Some(id),
47 pub fn as_static(self) -> Option<StaticId> {
49 LangItemTarget::StaticId(id) => Some(id),
54 pub fn as_struct(self) -> Option<StructId> {
56 LangItemTarget::StructId(id) => Some(id),
61 pub fn as_trait(self) -> Option<TraitId> {
63 LangItemTarget::TraitId(id) => Some(id),
69 #[derive(Default, Debug, Clone, PartialEq, Eq)]
70 pub struct LangItems {
71 items: FxHashMap<SmolStr, LangItemTarget>,
75 pub fn target<'a>(&'a self, item: &str) -> Option<&'a LangItemTarget> {
79 /// Salsa query. This will look for lang items in a specific crate.
80 pub(crate) fn crate_lang_items_query(db: &impl DefDatabase, krate: CrateId) -> Arc<LangItems> {
81 let mut lang_items = LangItems::default();
83 let crate_def_map = db.crate_def_map(krate);
88 .filter_map(|(local_id, _)| db.module_lang_items(ModuleId { krate, local_id }))
89 .for_each(|it| lang_items.items.extend(it.items.iter().map(|(k, v)| (k.clone(), *v))));
94 pub(crate) fn module_lang_items_query(
95 db: &impl DefDatabase,
97 ) -> Option<Arc<LangItems>> {
98 let mut lang_items = LangItems::default();
99 lang_items.collect_lang_items(db, module);
100 if lang_items.items.is_empty() {
103 Some(Arc::new(lang_items))
107 /// Salsa query. Look for a lang item, starting from the specified crate and recursively
108 /// traversing its dependencies.
109 pub(crate) fn lang_item_query(
110 db: &impl DefDatabase,
111 start_crate: CrateId,
113 ) -> Option<LangItemTarget> {
114 let lang_items = db.crate_lang_items(start_crate);
115 let start_crate_target = lang_items.items.get(&item);
116 if let Some(target) = start_crate_target {
117 return Some(*target);
120 .dependencies(start_crate)
121 .find_map(|dep| db.lang_item(dep.crate_id, item.clone()))
124 fn collect_lang_items(&mut self, db: &impl DefDatabase, module: ModuleId) {
125 // Look for impl targets
126 let def_map = db.crate_def_map(module.krate);
127 let module_data = &def_map[module.local_id];
128 for impl_block in module_data.scope.impls() {
129 self.collect_lang_item(db, impl_block, LangItemTarget::ImplBlockId)
132 for def in module_data.scope.declarations() {
134 ModuleDefId::TraitId(trait_) => {
135 self.collect_lang_item(db, trait_, LangItemTarget::TraitId)
137 ModuleDefId::AdtId(AdtId::EnumId(e)) => {
138 self.collect_lang_item(db, e, LangItemTarget::EnumId)
140 ModuleDefId::AdtId(AdtId::StructId(s)) => {
141 self.collect_lang_item(db, s, LangItemTarget::StructId)
143 ModuleDefId::FunctionId(f) => {
144 self.collect_lang_item(db, f, LangItemTarget::FunctionId)
146 ModuleDefId::StaticId(s) => self.collect_lang_item(db, s, LangItemTarget::StaticId),
152 fn collect_lang_item<T>(
154 db: &impl DefDatabase,
156 constructor: fn(T) -> LangItemTarget,
158 T: Into<AttrDefId> + Copy,
160 let attrs = db.attrs(item.into());
161 if let Some(lang_item_name) = attrs.by_key("lang").string_value() {
162 self.items.entry(lang_item_name.clone()).or_insert_with(|| constructor(item));