rust-analyzer/crates/hir_def/src/lang_item.rs

175 lines
5.3 KiB
Rust
Raw Normal View History

2019-11-23 09:58:01 +00:00
//! Collects lang items: items marked with `#[lang = "..."]` attribute.
//!
//! This attribute to tell the compiler about semi built-in std library
//! features, such as Fn family of traits.
use std::sync::Arc;
use rustc_hash::FxHashMap;
2020-08-12 16:26:51 +00:00
use syntax::SmolStr;
2019-11-23 09:58:01 +00:00
use crate::{
2019-11-23 11:44:43 +00:00
db::DefDatabase, AdtId, AttrDefId, CrateId, EnumId, FunctionId, ImplId, ModuleDefId, ModuleId,
2019-11-23 09:58:01 +00:00
StaticId, StructId, TraitId,
};
#[derive(Debug, Clone, Copy, PartialEq, Eq, Hash)]
pub enum LangItemTarget {
EnumId(EnumId),
FunctionId(FunctionId),
2020-02-29 20:24:40 +00:00
ImplDefId(ImplId),
2019-11-23 09:58:01 +00:00
StaticId(StaticId),
StructId(StructId),
TraitId(TraitId),
}
2019-12-29 16:38:37 +00:00
impl LangItemTarget {
pub fn as_enum(self) -> Option<EnumId> {
match self {
LangItemTarget::EnumId(id) => Some(id),
_ => None,
}
}
pub fn as_function(self) -> Option<FunctionId> {
match self {
LangItemTarget::FunctionId(id) => Some(id),
_ => None,
}
}
2020-02-29 20:24:40 +00:00
pub fn as_impl_def(self) -> Option<ImplId> {
2019-12-29 16:38:37 +00:00
match self {
2020-02-29 20:24:40 +00:00
LangItemTarget::ImplDefId(id) => Some(id),
2019-12-29 16:38:37 +00:00
_ => None,
}
}
pub fn as_static(self) -> Option<StaticId> {
match self {
LangItemTarget::StaticId(id) => Some(id),
_ => None,
}
}
pub fn as_struct(self) -> Option<StructId> {
match self {
LangItemTarget::StructId(id) => Some(id),
_ => None,
}
}
pub fn as_trait(self) -> Option<TraitId> {
match self {
LangItemTarget::TraitId(id) => Some(id),
_ => None,
}
}
}
2019-11-23 09:58:01 +00:00
#[derive(Default, Debug, Clone, PartialEq, Eq)]
pub struct LangItems {
items: FxHashMap<SmolStr, LangItemTarget>,
}
impl LangItems {
2020-05-22 13:55:15 +00:00
pub fn target(&self, item: &str) -> Option<LangItemTarget> {
self.items.get(item).copied()
2019-11-23 09:58:01 +00:00
}
/// Salsa query. This will look for lang items in a specific crate.
pub(crate) fn crate_lang_items_query(db: &dyn DefDatabase, krate: CrateId) -> Arc<LangItems> {
2020-08-12 14:32:36 +00:00
let _p = profile::span("crate_lang_items_query");
2020-04-10 18:25:55 +00:00
2019-11-23 09:58:01 +00:00
let mut lang_items = LangItems::default();
let crate_def_map = db.crate_def_map(krate);
crate_def_map
2019-11-24 15:05:12 +00:00
.modules
.iter()
2019-11-27 18:31:51 +00:00
.filter_map(|(local_id, _)| db.module_lang_items(ModuleId { krate, local_id }))
2019-11-23 09:58:01 +00:00
.for_each(|it| lang_items.items.extend(it.items.iter().map(|(k, v)| (k.clone(), *v))));
Arc::new(lang_items)
}
pub(crate) fn module_lang_items_query(
db: &dyn DefDatabase,
2019-11-23 09:58:01 +00:00
module: ModuleId,
) -> Option<Arc<LangItems>> {
2020-08-12 14:32:36 +00:00
let _p = profile::span("module_lang_items_query");
2019-11-23 09:58:01 +00:00
let mut lang_items = LangItems::default();
lang_items.collect_lang_items(db, module);
if lang_items.items.is_empty() {
None
} else {
Some(Arc::new(lang_items))
}
}
/// Salsa query. Look for a lang item, starting from the specified crate and recursively
/// traversing its dependencies.
pub(crate) fn lang_item_query(
db: &dyn DefDatabase,
2019-11-23 09:58:01 +00:00
start_crate: CrateId,
item: SmolStr,
) -> Option<LangItemTarget> {
2020-08-12 14:32:36 +00:00
let _p = profile::span("lang_item_query");
2019-11-23 09:58:01 +00:00
let lang_items = db.crate_lang_items(start_crate);
let start_crate_target = lang_items.items.get(&item);
if let Some(target) = start_crate_target {
return Some(*target);
}
2020-03-09 10:11:59 +00:00
db.crate_graph()[start_crate]
2020-03-09 09:26:46 +00:00
.dependencies
.iter()
2019-11-23 09:58:01 +00:00
.find_map(|dep| db.lang_item(dep.crate_id, item.clone()))
}
fn collect_lang_items(&mut self, db: &dyn DefDatabase, module: ModuleId) {
2019-11-23 09:58:01 +00:00
// Look for impl targets
let def_map = db.crate_def_map(module.krate);
2019-11-27 18:31:51 +00:00
let module_data = &def_map[module.local_id];
2020-02-29 20:24:40 +00:00
for impl_def in module_data.scope.impls() {
self.collect_lang_item(db, impl_def, LangItemTarget::ImplDefId)
2019-11-23 09:58:01 +00:00
}
for def in module_data.scope.declarations() {
match def {
ModuleDefId::TraitId(trait_) => {
self.collect_lang_item(db, trait_, LangItemTarget::TraitId)
}
ModuleDefId::AdtId(AdtId::EnumId(e)) => {
self.collect_lang_item(db, e, LangItemTarget::EnumId)
}
ModuleDefId::AdtId(AdtId::StructId(s)) => {
self.collect_lang_item(db, s, LangItemTarget::StructId)
}
ModuleDefId::FunctionId(f) => {
self.collect_lang_item(db, f, LangItemTarget::FunctionId)
}
ModuleDefId::StaticId(s) => self.collect_lang_item(db, s, LangItemTarget::StaticId),
_ => {}
}
}
}
fn collect_lang_item<T>(
&mut self,
db: &dyn DefDatabase,
2019-11-23 09:58:01 +00:00
item: T,
constructor: fn(T) -> LangItemTarget,
) where
T: Into<AttrDefId> + Copy,
{
2020-05-22 13:55:15 +00:00
if let Some(lang_item_name) = lang_attr(db, item) {
2020-05-25 17:35:52 +00:00
self.items.entry(lang_item_name).or_insert_with(|| constructor(item));
2019-11-23 09:58:01 +00:00
}
}
}
2020-05-22 13:55:15 +00:00
pub fn lang_attr(db: &dyn DefDatabase, item: impl Into<AttrDefId> + Copy) -> Option<SmolStr> {
let attrs = db.attrs(item.into());
attrs.by_key("lang").string_value().cloned()
}