rustc_passes/
diagnostic_items.rs1use rustc_hir::diagnostic_items::DiagnosticItems;
13use rustc_hir::{CRATE_OWNER_ID, OwnerId, find_attr};
14use rustc_middle::query::{LocalCrate, Providers};
15use rustc_middle::ty::TyCtxt;
16use rustc_span::Symbol;
17use rustc_span::def_id::{DefId, LOCAL_CRATE};
18
19use crate::errors::DuplicateDiagnosticItemInCrate;
20
21fn observe_item<'tcx>(tcx: TyCtxt<'tcx>, diagnostic_items: &mut DiagnosticItems, owner: OwnerId) {
22 let attrs = tcx.hir_attrs(owner.into());
23 if let Some(name) = {
'done:
{
for i in attrs {
#[allow(unused_imports)]
use rustc_hir::attrs::AttributeKind::*;
let i: &rustc_hir::Attribute = i;
match i {
rustc_hir::Attribute::Parsed(RustcDiagnosticItem(name)) => {
break 'done Some(name);
}
rustc_hir::Attribute::Unparsed(..) =>
{}
#[deny(unreachable_patterns)]
_ => {}
}
}
None
}
}find_attr!(attrs, RustcDiagnosticItem(name) => name) {
24 collect_item(tcx, diagnostic_items, *name, owner.to_def_id());
26 }
27}
28
29fn collect_item(tcx: TyCtxt<'_>, items: &mut DiagnosticItems, name: Symbol, item_def_id: DefId) {
30 items.id_to_name.insert(item_def_id, name);
31 if let Some(original_def_id) = items.name_to_id.insert(name, item_def_id) {
32 if original_def_id != item_def_id {
33 report_duplicate_item(tcx, name, original_def_id, item_def_id);
34 }
35 }
36}
37
38fn report_duplicate_item(
39 tcx: TyCtxt<'_>,
40 name: Symbol,
41 original_def_id: DefId,
42 item_def_id: DefId,
43) {
44 let orig_span = tcx.hir_span_if_local(original_def_id);
45 let duplicate_span = tcx.hir_span_if_local(item_def_id);
46 tcx.dcx().emit_err(DuplicateDiagnosticItemInCrate {
47 duplicate_span,
48 orig_span,
49 crate_name: tcx.crate_name(item_def_id.krate),
50 orig_crate_name: tcx.crate_name(original_def_id.krate),
51 different_crates: (item_def_id.krate != original_def_id.krate),
52 name,
53 });
54}
55
56fn diagnostic_items(tcx: TyCtxt<'_>, _: LocalCrate) -> DiagnosticItems {
58 let mut diagnostic_items = DiagnosticItems::default();
60
61 let crate_items = tcx.hir_crate_items(());
63 for id in crate_items.owners().chain(std::iter::once(CRATE_OWNER_ID)) {
64 observe_item(tcx, &mut diagnostic_items, id);
65 }
66
67 diagnostic_items
68}
69
70fn all_diagnostic_items(tcx: TyCtxt<'_>, (): ()) -> DiagnosticItems {
72 let mut items = DiagnosticItems::default();
74
75 for cnum in tcx
77 .crates(())
78 .iter()
79 .copied()
80 .filter(|cnum| tcx.is_user_visible_dep(*cnum))
81 .chain(std::iter::once(LOCAL_CRATE))
82 {
83 for (&name, &def_id) in &tcx.diagnostic_items(cnum).name_to_id {
84 collect_item(tcx, &mut items, name, def_id);
85 }
86 }
87
88 items
89}
90
91pub(crate) fn provide(providers: &mut Providers) {
92 providers.diagnostic_items = diagnostic_items;
93 providers.all_diagnostic_items = all_diagnostic_items;
94}