rustc_passes/
diagnostic_items.rs
1use rustc_hir::diagnostic_items::DiagnosticItems;
13use rustc_hir::{Attribute, OwnerId};
14use rustc_middle::query::{LocalCrate, Providers};
15use rustc_middle::ty::TyCtxt;
16use rustc_span::def_id::{DefId, LOCAL_CRATE};
17use rustc_span::{Symbol, sym};
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) = extract(attrs) {
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 extract(attrs: &[Attribute]) -> Option<Symbol> {
58 attrs.iter().find_map(|attr| {
59 if attr.has_name(sym::rustc_diagnostic_item) { attr.value_str() } else { None }
60 })
61}
62
63fn diagnostic_items(tcx: TyCtxt<'_>, _: LocalCrate) -> DiagnosticItems {
65 let mut diagnostic_items = DiagnosticItems::default();
67
68 let crate_items = tcx.hir_crate_items(());
70 for id in crate_items.owners() {
71 observe_item(tcx, &mut diagnostic_items, id);
72 }
73
74 diagnostic_items
75}
76
77fn all_diagnostic_items(tcx: TyCtxt<'_>, (): ()) -> DiagnosticItems {
79 let mut items = DiagnosticItems::default();
81
82 for cnum in tcx
84 .crates(())
85 .iter()
86 .copied()
87 .filter(|cnum| tcx.is_user_visible_dep(*cnum))
88 .chain(std::iter::once(LOCAL_CRATE))
89 {
90 for (&name, &def_id) in &tcx.diagnostic_items(cnum).name_to_id {
91 collect_item(tcx, &mut items, name, def_id);
92 }
93 }
94
95 items
96}
97
98pub(crate) fn provide(providers: &mut Providers) {
99 providers.diagnostic_items = diagnostic_items;
100 providers.all_diagnostic_items = all_diagnostic_items;
101}