1use rustc_abi::ExternAbi;
6use rustc_ast::visit::{VisitorResult, walk_list};
7use rustc_data_structures::fingerprint::Fingerprint;
8use rustc_data_structures::stable_hash::{StableHash, StableHasher};
9use rustc_data_structures::steal::Steal;
10use rustc_data_structures::svh::Svh;
11use rustc_data_structures::sync::{DynSend, DynSync, par_for_each_in, spawn, try_par_for_each_in};
12use rustc_hir::def::{DefKind, Res};
13use rustc_hir::def_id::{DefId, LOCAL_CRATE, LocalDefId, LocalModDefId};
14use rustc_hir::definitions::{DefKey, DefPath, DefPathHash};
15use rustc_hir::intravisit::Visitor;
16use rustc_hir::lints::DelayedLints;
17use rustc_hir::*;
18use rustc_hir_pretty as pprust_hir;
19use rustc_span::def_id::StableCrateId;
20use rustc_span::{ErrorGuaranteed, Ident, Span, Symbol, kw, with_metavar_spans};
21
22use crate::hir::{ModuleItems, ProjectedMaybeOwner, nested_filter};
23use crate::middle::debugger_visualizer::DebuggerVisualizerFile;
24use crate::query::{IntoQueryKey, LocalCrate};
25use crate::ty::TyCtxt;
26
27struct ParentHirIterator<'tcx> {
30 current_id: HirId,
31 tcx: TyCtxt<'tcx>,
32 current_owner_nodes: Option<&'tcx OwnerNodes<'tcx>>,
35}
36
37impl<'tcx> ParentHirIterator<'tcx> {
38 fn new(tcx: TyCtxt<'tcx>, current_id: HirId) -> ParentHirIterator<'tcx> {
39 ParentHirIterator { current_id, tcx, current_owner_nodes: None }
40 }
41}
42
43impl<'tcx> Iterator for ParentHirIterator<'tcx> {
44 type Item = HirId;
45
46 fn next(&mut self) -> Option<Self::Item> {
47 if self.current_id == CRATE_HIR_ID {
48 return None;
49 }
50
51 let HirId { owner, local_id } = self.current_id;
52
53 let parent_id = if local_id == ItemLocalId::ZERO {
54 self.current_owner_nodes = None;
56 self.tcx.hir_owner_parent(owner)
57 } else {
58 let owner_nodes =
59 self.current_owner_nodes.get_or_insert_with(|| self.tcx.hir_owner_nodes(owner));
60 let parent_local_id = owner_nodes.nodes[local_id].parent;
61 if true {
match (&parent_local_id, &local_id) {
(left_val, right_val) => {
if *left_val == *right_val {
let kind = ::core::panicking::AssertKind::Ne;
::core::panicking::assert_failed(kind, &*left_val,
&*right_val, ::core::option::Option::None);
}
}
};
};debug_assert_ne!(parent_local_id, local_id);
63 HirId { owner, local_id: parent_local_id }
64 };
65
66 if true {
match (&parent_id, &self.current_id) {
(left_val, right_val) => {
if *left_val == *right_val {
let kind = ::core::panicking::AssertKind::Ne;
::core::panicking::assert_failed(kind, &*left_val,
&*right_val, ::core::option::Option::None);
}
}
};
};debug_assert_ne!(parent_id, self.current_id);
67
68 self.current_id = parent_id;
69 Some(parent_id)
70 }
71}
72
73pub struct ParentOwnerIterator<'tcx> {
76 current_id: HirId,
77 tcx: TyCtxt<'tcx>,
78}
79
80impl<'tcx> Iterator for ParentOwnerIterator<'tcx> {
81 type Item = (OwnerId, OwnerNode<'tcx>);
82
83 fn next(&mut self) -> Option<Self::Item> {
84 if self.current_id.local_id.index() != 0 {
85 self.current_id.local_id = ItemLocalId::ZERO;
86 let node = self.tcx.hir_owner_node(self.current_id.owner);
87 return Some((self.current_id.owner, node));
88 }
89 if self.current_id == CRATE_HIR_ID {
90 return None;
91 }
92
93 let parent_id = self.tcx.hir_def_key(self.current_id.owner.def_id).parent;
94 let parent_id = parent_id.map_or(CRATE_OWNER_ID, |local_def_index| {
95 let def_id = LocalDefId { local_def_index };
96 self.tcx.local_def_id_to_hir_id(def_id).owner
97 });
98 self.current_id = HirId::make_owner(parent_id.def_id);
99
100 let node = self.tcx.hir_owner_node(self.current_id.owner);
101 Some((self.current_id.owner, node))
102 }
103}
104
105impl<'tcx> TyCtxt<'tcx> {
106 #[inline]
107 pub fn local_def_id_to_hir_id(self, def_id: impl IntoQueryKey<LocalDefId>) -> HirId {
108 let def_id = def_id.into_query_key();
109 match self.hir_owner(def_id) {
110 ProjectedMaybeOwner::Owner(_) => HirId::make_owner(def_id),
111 ProjectedMaybeOwner::NonOwner(hir_id) => hir_id,
112 }
113 }
114
115 #[inline]
119 pub fn opt_ast_lowering_delayed_lints(self, id: OwnerId) -> Option<&'tcx Steal<DelayedLints>> {
120 self.dep_graph.assert_eval_always();
121 self.hir_owner(id.def_id).as_owner().map(|o| o.delayed_lints)
122 }
123
124 #[inline]
125 pub fn in_scope_traits_map(
126 self,
127 id: OwnerId,
128 ) -> Option<&'tcx ItemLocalMap<&'tcx [TraitCandidate<'tcx>]>> {
129 self.hir_owner(id.def_id).as_owner().map(|o| o.trait_map)
130 }
131
132 #[inline]
133 pub fn opt_hir_owner_nodes(self, def_id: LocalDefId) -> Option<&'tcx OwnerNodes<'tcx>> {
134 self.hir_owner(def_id).as_owner().map(|o| o.nodes)
135 }
136
137 #[inline]
138 fn expect_hir_owner_nodes(self, def_id: LocalDefId) -> &'tcx OwnerNodes<'tcx> {
139 self.opt_hir_owner_nodes(def_id)
140 .unwrap_or_else(|| crate::util::bug::span_bug_fmt(self.def_span(def_id),
format_args!("{0:?} is not an owner", def_id))span_bug!(self.def_span(def_id), "{def_id:?} is not an owner"))
141 }
142
143 #[inline]
144 pub fn hir_owner_nodes(self, owner_id: OwnerId) -> &'tcx OwnerNodes<'tcx> {
145 self.expect_hir_owner_nodes(owner_id.def_id)
146 }
147
148 #[inline]
149 fn opt_hir_owner_node(self, def_id: LocalDefId) -> Option<OwnerNode<'tcx>> {
150 self.opt_hir_owner_nodes(def_id).map(|nodes| nodes.node())
151 }
152
153 #[inline]
154 pub fn expect_hir_owner_node(self, def_id: LocalDefId) -> OwnerNode<'tcx> {
155 self.expect_hir_owner_nodes(def_id).node()
156 }
157
158 #[inline]
159 pub fn hir_owner_node(self, owner_id: OwnerId) -> OwnerNode<'tcx> {
160 self.hir_owner_nodes(owner_id).node()
161 }
162
163 pub fn hir_node(self, id: HirId) -> Node<'tcx> {
165 self.hir_owner_nodes(id.owner).nodes[id.local_id].node
166 }
167
168 #[inline]
170 pub fn hir_node_by_def_id(self, id: LocalDefId) -> Node<'tcx> {
171 self.hir_node(self.local_def_id_to_hir_id(id))
172 }
173
174 pub fn parent_hir_id(self, hir_id: HirId) -> HirId {
179 let HirId { owner, local_id } = hir_id;
180 if local_id == ItemLocalId::ZERO {
181 self.hir_owner_parent(owner)
182 } else {
183 let parent_local_id = self.hir_owner_nodes(owner).nodes[local_id].parent;
184 if true {
match (&parent_local_id, &local_id) {
(left_val, right_val) => {
if *left_val == *right_val {
let kind = ::core::panicking::AssertKind::Ne;
::core::panicking::assert_failed(kind, &*left_val,
&*right_val, ::core::option::Option::None);
}
}
};
};debug_assert_ne!(parent_local_id, local_id);
186 HirId { owner, local_id: parent_local_id }
187 }
188 }
189
190 pub fn parent_hir_node(self, hir_id: HirId) -> Node<'tcx> {
193 self.hir_node(self.parent_hir_id(hir_id))
194 }
195
196 #[inline]
197 pub fn hir_root_module(self) -> &'tcx Mod<'tcx> {
198 match self.hir_owner_node(CRATE_OWNER_ID) {
199 OwnerNode::Crate(item) => item,
200 _ => crate::util::bug::bug_fmt(format_args!("impossible case reached"))bug!(),
201 }
202 }
203
204 #[inline]
205 pub fn hir_free_items(self) -> impl Iterator<Item = ItemId> {
206 self.hir_crate_items(()).free_items.iter().copied()
207 }
208
209 #[inline]
210 pub fn hir_module_free_items(self, module: LocalModDefId) -> impl Iterator<Item = ItemId> {
211 self.hir_module_items(module).free_items()
212 }
213
214 pub fn hir_def_key(self, def_id: LocalDefId) -> DefKey {
215 self.definitions_untracked().def_key(def_id)
217 }
218
219 pub fn hir_def_path(self, def_id: LocalDefId) -> DefPath {
220 self.definitions_untracked().def_path(def_id)
222 }
223
224 #[inline]
225 pub fn hir_def_path_hash(self, def_id: LocalDefId) -> DefPathHash {
226 self.definitions_untracked().def_path_hash(def_id)
228 }
229
230 pub fn hir_get_if_local(self, id: DefId) -> Option<Node<'tcx>> {
231 id.as_local().map(|id| self.hir_node_by_def_id(id))
232 }
233
234 pub fn hir_get_generics(self, id: LocalDefId) -> Option<&'tcx Generics<'tcx>> {
235 self.opt_hir_owner_node(id)?.generics()
236 }
237
238 pub fn hir_item(self, id: ItemId) -> &'tcx Item<'tcx> {
239 self.hir_owner_node(id.owner_id).expect_item()
240 }
241
242 pub fn hir_trait_item(self, id: TraitItemId) -> &'tcx TraitItem<'tcx> {
243 self.hir_owner_node(id.owner_id).expect_trait_item()
244 }
245
246 pub fn hir_impl_item(self, id: ImplItemId) -> &'tcx ImplItem<'tcx> {
247 self.hir_owner_node(id.owner_id).expect_impl_item()
248 }
249
250 pub fn hir_foreign_item(self, id: ForeignItemId) -> &'tcx ForeignItem<'tcx> {
251 self.hir_owner_node(id.owner_id).expect_foreign_item()
252 }
253
254 pub fn hir_body(self, id: BodyId) -> &'tcx Body<'tcx> {
255 self.hir_owner_nodes(id.hir_id.owner).bodies[&id.hir_id.local_id]
256 }
257
258 #[track_caller]
259 pub fn hir_fn_decl_by_hir_id(self, hir_id: HirId) -> Option<&'tcx FnDecl<'tcx>> {
260 self.hir_node(hir_id).fn_decl()
261 }
262
263 #[track_caller]
264 pub fn hir_fn_sig_by_hir_id(self, hir_id: HirId) -> Option<&'tcx FnSig<'tcx>> {
265 self.hir_node(hir_id).fn_sig()
266 }
267
268 #[track_caller]
269 pub fn hir_enclosing_body_owner(self, hir_id: HirId) -> LocalDefId {
270 for (_, node) in self.hir_parent_iter(hir_id) {
271 if let Some((def_id, _)) = node.associated_body() {
272 return def_id;
273 }
274 }
275
276 crate::util::bug::bug_fmt(format_args!("no `hir_enclosing_body_owner` for hir_id `{0}`",
hir_id));bug!("no `hir_enclosing_body_owner` for hir_id `{}`", hir_id);
277 }
278
279 pub fn hir_body_owner(self, BodyId { hir_id }: BodyId) -> HirId {
283 let parent = self.parent_hir_id(hir_id);
284 match (&self.hir_node(parent).body_id().unwrap().hir_id, &hir_id) {
(left_val, right_val) => {
if !(*left_val == *right_val) {
let kind = ::core::panicking::AssertKind::Eq;
::core::panicking::assert_failed(kind, &*left_val, &*right_val,
::core::option::Option::Some(format_args!("{0:?}", hir_id)));
}
}
};assert_eq!(self.hir_node(parent).body_id().unwrap().hir_id, hir_id, "{hir_id:?}");
285 parent
286 }
287
288 pub fn hir_body_owner_def_id(self, BodyId { hir_id }: BodyId) -> LocalDefId {
289 self.parent_hir_node(hir_id).associated_body().unwrap().0
290 }
291
292 pub fn hir_maybe_body_owned_by(self, id: LocalDefId) -> Option<&'tcx Body<'tcx>> {
295 Some(self.hir_body(self.hir_node_by_def_id(id).body_id()?))
296 }
297
298 #[track_caller]
300 pub fn hir_body_owned_by(self, id: LocalDefId) -> &'tcx Body<'tcx> {
301 self.hir_maybe_body_owned_by(id).unwrap_or_else(|| {
302 let hir_id = self.local_def_id_to_hir_id(id);
303 crate::util::bug::span_bug_fmt(self.hir_span(hir_id),
format_args!("body_owned_by: {0} has no associated body",
self.hir_id_to_string(hir_id)));span_bug!(
304 self.hir_span(hir_id),
305 "body_owned_by: {} has no associated body",
306 self.hir_id_to_string(hir_id)
307 );
308 })
309 }
310
311 pub fn hir_body_param_idents(self, id: BodyId) -> impl Iterator<Item = Option<Ident>> {
312 self.hir_body(id).params.iter().map(|param| match param.pat.kind {
313 PatKind::Binding(_, _, ident, _) => Some(ident),
314 PatKind::Wild => Some(Ident::new(kw::Underscore, param.pat.span)),
315 _ => None,
316 })
317 }
318
319 pub fn hir_body_owner_kind(self, def_id: impl Into<DefId>) -> BodyOwnerKind {
323 let def_id = def_id.into();
324 match self.def_kind(def_id) {
325 DefKind::Const { .. } | DefKind::AssocConst { .. } | DefKind::AnonConst => {
326 BodyOwnerKind::Const { inline: false }
327 }
328 DefKind::InlineConst => BodyOwnerKind::Const { inline: true },
329 DefKind::Ctor(..) | DefKind::Fn | DefKind::AssocFn => BodyOwnerKind::Fn,
330 DefKind::Closure | DefKind::SyntheticCoroutineBody => BodyOwnerKind::Closure,
331 DefKind::Static { safety: _, mutability, nested: false } => {
332 BodyOwnerKind::Static(mutability)
333 }
334 DefKind::GlobalAsm => BodyOwnerKind::GlobalAsm,
335 dk => crate::util::bug::bug_fmt(format_args!("{0:?} is not a body node: {1:?}",
def_id, dk))bug!("{:?} is not a body node: {:?}", def_id, dk),
336 }
337 }
338
339 pub fn hir_body_const_context(self, local_def_id: LocalDefId) -> Option<ConstContext> {
347 let def_id = local_def_id.into();
348 let ccx = match self.hir_body_owner_kind(def_id) {
349 BodyOwnerKind::Const { inline } => ConstContext::Const { inline },
350 BodyOwnerKind::Static(mutability) => ConstContext::Static(mutability),
351
352 BodyOwnerKind::Fn if self.is_constructor(def_id) => return None,
353 BodyOwnerKind::Fn | BodyOwnerKind::Closure if self.is_const_fn(def_id) => {
354 ConstContext::ConstFn
355 }
356 BodyOwnerKind::Fn | BodyOwnerKind::Closure | BodyOwnerKind::GlobalAsm => return None,
357 };
358
359 Some(ccx)
360 }
361
362 #[inline]
365 pub fn hir_body_owners(self) -> impl Iterator<Item = LocalDefId> {
366 self.hir_crate_items(()).body_owners.iter().copied()
367 }
368
369 #[inline]
370 pub fn par_hir_body_owners(self, f: impl Fn(LocalDefId) + DynSend + DynSync) {
371 par_for_each_in(&self.hir_crate_items(()).body_owners[..], |&&def_id| f(def_id));
372 }
373
374 pub fn hir_ty_param_owner(self, def_id: LocalDefId) -> LocalDefId {
375 let def_kind = self.def_kind(def_id);
376 match def_kind {
377 DefKind::Trait | DefKind::TraitAlias => def_id,
378 DefKind::LifetimeParam | DefKind::TyParam | DefKind::ConstParam => {
379 self.local_parent(def_id)
380 }
381 _ => crate::util::bug::bug_fmt(format_args!("ty_param_owner: {0:?} is a {1:?} not a type parameter",
def_id, def_kind))bug!("ty_param_owner: {:?} is a {:?} not a type parameter", def_id, def_kind),
382 }
383 }
384
385 pub fn hir_ty_param_name(self, def_id: LocalDefId) -> Symbol {
386 let def_kind = self.def_kind(def_id);
387 match def_kind {
388 DefKind::Trait | DefKind::TraitAlias => kw::SelfUpper,
389 DefKind::LifetimeParam | DefKind::TyParam | DefKind::ConstParam => {
390 self.item_name(def_id.to_def_id())
391 }
392 _ => crate::util::bug::bug_fmt(format_args!("ty_param_name: {0:?} is a {1:?} not a type parameter",
def_id, def_kind))bug!("ty_param_name: {:?} is a {:?} not a type parameter", def_id, def_kind),
393 }
394 }
395
396 pub fn hir_krate_attrs(self) -> &'tcx [Attribute] {
400 self.hir_attrs(CRATE_HIR_ID)
401 }
402
403 pub fn hir_rustc_coherence_is_core(self) -> bool {
404 {
{
'done:
{
for i in self.hir_krate_attrs() {
#[allow(unused_imports)]
use rustc_hir::attrs::AttributeKind::*;
let i: &rustc_hir::Attribute = i;
match i {
rustc_hir::Attribute::Parsed(RustcCoherenceIsCore) => {
break 'done Some(());
}
rustc_hir::Attribute::Unparsed(..) =>
{}
#[deny(unreachable_patterns)]
_ => {}
}
}
None
}
}.is_some()
}find_attr!(self.hir_krate_attrs(), RustcCoherenceIsCore)
405 }
406
407 pub fn hir_get_module(self, module: LocalModDefId) -> (&'tcx Mod<'tcx>, Span, HirId) {
408 let hir_id = HirId::make_owner(module.to_local_def_id());
409 match self.hir_owner_node(hir_id.owner) {
410 OwnerNode::Item(&Item { span, kind: ItemKind::Mod(_, m), .. }) => (m, span, hir_id),
411 OwnerNode::Crate(item) => (item, item.spans.inner_span, hir_id),
412 node => { ::core::panicking::panic_fmt(format_args!("not a module: {0:?}", node)); }panic!("not a module: {node:?}"),
413 }
414 }
415
416 pub fn hir_walk_toplevel_module<V>(self, visitor: &mut V) -> V::Result
418 where
419 V: Visitor<'tcx>,
420 {
421 let (top_mod, span, hir_id) = self.hir_get_module(LocalModDefId::CRATE_DEF_ID);
422 visitor.visit_mod(top_mod, span, hir_id)
423 }
424
425 pub fn hir_walk_attributes<V>(self, visitor: &mut V) -> V::Result
427 where
428 V: Visitor<'tcx>,
429 {
430 let krate = self.hir_crate_items(());
431 for owner in krate.owners() {
432 let attrs = self.hir_attr_map(owner);
433 for attrs in attrs.map.values() {
434 for elem in *attrs {
match ::rustc_ast_ir::visit::VisitorResult::branch(visitor.visit_attribute(elem))
{
core::ops::ControlFlow::Continue(()) =>
(),
#[allow(unreachable_code)]
core::ops::ControlFlow::Break(r) => {
return ::rustc_ast_ir::visit::VisitorResult::from_residual(r);
}
};
};walk_list!(visitor, visit_attribute, *attrs);
435 }
436 }
437 V::Result::output()
438 }
439
440 pub fn hir_visit_all_item_likes_in_crate<V>(self, visitor: &mut V) -> V::Result
451 where
452 V: Visitor<'tcx>,
453 {
454 let krate = self.hir_crate_items(());
455 for elem in krate.free_items().map(|id| self.hir_item(id)) {
match ::rustc_ast_ir::visit::VisitorResult::branch(visitor.visit_item(elem))
{
core::ops::ControlFlow::Continue(()) =>
(),
#[allow(unreachable_code)]
core::ops::ControlFlow::Break(r) => {
return ::rustc_ast_ir::visit::VisitorResult::from_residual(r);
}
};
};walk_list!(visitor, visit_item, krate.free_items().map(|id| self.hir_item(id)));
456 for elem in krate.trait_items().map(|id| self.hir_trait_item(id)) {
match ::rustc_ast_ir::visit::VisitorResult::branch(visitor.visit_trait_item(elem))
{
core::ops::ControlFlow::Continue(()) =>
(),
#[allow(unreachable_code)]
core::ops::ControlFlow::Break(r) => {
return ::rustc_ast_ir::visit::VisitorResult::from_residual(r);
}
};
};walk_list!(
457 visitor,
458 visit_trait_item,
459 krate.trait_items().map(|id| self.hir_trait_item(id))
460 );
461 for elem in krate.impl_items().map(|id| self.hir_impl_item(id)) {
match ::rustc_ast_ir::visit::VisitorResult::branch(visitor.visit_impl_item(elem))
{
core::ops::ControlFlow::Continue(()) =>
(),
#[allow(unreachable_code)]
core::ops::ControlFlow::Break(r) => {
return ::rustc_ast_ir::visit::VisitorResult::from_residual(r);
}
};
};walk_list!(visitor, visit_impl_item, krate.impl_items().map(|id| self.hir_impl_item(id)));
462 for elem in krate.foreign_items().map(|id| self.hir_foreign_item(id)) {
match ::rustc_ast_ir::visit::VisitorResult::branch(visitor.visit_foreign_item(elem))
{
core::ops::ControlFlow::Continue(()) =>
(),
#[allow(unreachable_code)]
core::ops::ControlFlow::Break(r) => {
return ::rustc_ast_ir::visit::VisitorResult::from_residual(r);
}
};
};walk_list!(
463 visitor,
464 visit_foreign_item,
465 krate.foreign_items().map(|id| self.hir_foreign_item(id))
466 );
467 V::Result::output()
468 }
469
470 pub fn hir_visit_item_likes_in_module<V>(
473 self,
474 module: LocalModDefId,
475 visitor: &mut V,
476 ) -> V::Result
477 where
478 V: Visitor<'tcx>,
479 {
480 let module = self.hir_module_items(module);
481 for elem in module.free_items().map(|id| self.hir_item(id)) {
match ::rustc_ast_ir::visit::VisitorResult::branch(visitor.visit_item(elem))
{
core::ops::ControlFlow::Continue(()) =>
(),
#[allow(unreachable_code)]
core::ops::ControlFlow::Break(r) => {
return ::rustc_ast_ir::visit::VisitorResult::from_residual(r);
}
};
};walk_list!(visitor, visit_item, module.free_items().map(|id| self.hir_item(id)));
482 for elem in module.trait_items().map(|id| self.hir_trait_item(id)) {
match ::rustc_ast_ir::visit::VisitorResult::branch(visitor.visit_trait_item(elem))
{
core::ops::ControlFlow::Continue(()) =>
(),
#[allow(unreachable_code)]
core::ops::ControlFlow::Break(r) => {
return ::rustc_ast_ir::visit::VisitorResult::from_residual(r);
}
};
};walk_list!(
483 visitor,
484 visit_trait_item,
485 module.trait_items().map(|id| self.hir_trait_item(id))
486 );
487 for elem in module.impl_items().map(|id| self.hir_impl_item(id)) {
match ::rustc_ast_ir::visit::VisitorResult::branch(visitor.visit_impl_item(elem))
{
core::ops::ControlFlow::Continue(()) =>
(),
#[allow(unreachable_code)]
core::ops::ControlFlow::Break(r) => {
return ::rustc_ast_ir::visit::VisitorResult::from_residual(r);
}
};
};walk_list!(visitor, visit_impl_item, module.impl_items().map(|id| self.hir_impl_item(id)));
488 for elem in module.foreign_items().map(|id| self.hir_foreign_item(id)) {
match ::rustc_ast_ir::visit::VisitorResult::branch(visitor.visit_foreign_item(elem))
{
core::ops::ControlFlow::Continue(()) =>
(),
#[allow(unreachable_code)]
core::ops::ControlFlow::Break(r) => {
return ::rustc_ast_ir::visit::VisitorResult::from_residual(r);
}
};
};walk_list!(
489 visitor,
490 visit_foreign_item,
491 module.foreign_items().map(|id| self.hir_foreign_item(id))
492 );
493 V::Result::output()
494 }
495
496 pub fn hir_for_each_module(self, mut f: impl FnMut(LocalModDefId)) {
497 let crate_items = self.hir_crate_items(());
498 for module in crate_items.submodules.iter() {
499 f(LocalModDefId::new_unchecked(module.def_id))
500 }
501 }
502
503 #[inline]
504 pub fn par_hir_for_each_module(self, f: impl Fn(LocalModDefId) + DynSend + DynSync) {
505 let crate_items = self.hir_crate_items(());
506 par_for_each_in(&crate_items.submodules[..], |module| {
507 f(LocalModDefId::new_unchecked(module.def_id))
508 })
509 }
510
511 #[inline]
512 pub fn try_par_hir_for_each_module(
513 self,
514 f: impl Fn(LocalModDefId) -> Result<(), ErrorGuaranteed> + DynSend + DynSync,
515 ) -> Result<(), ErrorGuaranteed> {
516 let crate_items = self.hir_crate_items(());
517 try_par_for_each_in(&crate_items.submodules[..], |module| {
518 f(LocalModDefId::new_unchecked(module.def_id))
519 })
520 }
521
522 #[inline]
525 pub fn hir_parent_id_iter(self, current_id: HirId) -> impl Iterator<Item = HirId> {
526 ParentHirIterator::new(self, current_id)
527 }
528
529 #[inline]
532 pub fn hir_parent_iter(self, current_id: HirId) -> impl Iterator<Item = (HirId, Node<'tcx>)> {
533 self.hir_parent_id_iter(current_id).map(move |id| (id, self.hir_node(id)))
534 }
535
536 #[inline]
539 pub fn hir_parent_owner_iter(self, current_id: HirId) -> ParentOwnerIterator<'tcx> {
540 ParentOwnerIterator { current_id, tcx: self }
541 }
542
543 pub fn hir_is_lhs(self, id: HirId) -> bool {
545 match self.parent_hir_node(id) {
546 Node::Expr(expr) => match expr.kind {
547 ExprKind::Assign(lhs, _rhs, _span) => lhs.hir_id == id,
548 _ => false,
549 },
550 _ => false,
551 }
552 }
553
554 pub fn hir_is_inside_const_context(self, hir_id: HirId) -> bool {
557 self.hir_body_const_context(self.hir_enclosing_body_owner(hir_id)).is_some()
558 }
559
560 pub fn hir_get_fn_id_for_return_block(self, id: HirId) -> Option<HirId> {
587 let enclosing_body_owner = self.local_def_id_to_hir_id(self.hir_enclosing_body_owner(id));
588
589 let mut iter = [id].into_iter().chain(self.hir_parent_id_iter(id)).peekable();
591 while let Some(cur_id) = iter.next() {
592 if enclosing_body_owner == cur_id {
593 break;
594 }
595
596 if let Node::Expr(Expr { kind: ExprKind::Ret(_), .. }) = self.hir_node(cur_id) {
599 break;
600 }
601
602 if let Some(&parent_id) = iter.peek() {
605 match self.hir_node(parent_id) {
606 Node::Block(Block { expr: Some(e), .. }) if cur_id != e.hir_id => return None,
609 Node::Block(Block { expr: Some(e), .. })
610 if #[allow(non_exhaustive_omitted_patterns)] match e.kind {
ExprKind::If(_, _, None) => true,
_ => false,
}matches!(e.kind, ExprKind::If(_, _, None)) =>
611 {
612 return None;
613 }
614
615 Node::Block(Block { expr: None, .. })
618 | Node::Expr(Expr { kind: ExprKind::Loop(..), .. })
619 | Node::LetStmt(..) => return None,
620
621 _ => {}
622 }
623 }
624 }
625
626 Some(enclosing_body_owner)
627 }
628
629 pub fn hir_get_parent_item(self, hir_id: HirId) -> OwnerId {
634 if hir_id.local_id != ItemLocalId::ZERO {
635 hir_id.owner
637 } else if let Some((def_id, _node)) = self.hir_parent_owner_iter(hir_id).next() {
638 def_id
639 } else {
640 CRATE_OWNER_ID
641 }
642 }
643
644 pub fn hir_get_if_cause(self, hir_id: HirId) -> Option<&'tcx Expr<'tcx>> {
650 for (_, node) in self.hir_parent_iter(hir_id) {
651 match node {
652 Node::Item(_)
653 | Node::ForeignItem(_)
654 | Node::TraitItem(_)
655 | Node::ImplItem(_)
656 | Node::Stmt(Stmt { kind: StmtKind::Let(_), .. }) => break,
657 Node::Expr(expr @ Expr { kind: ExprKind::If(..) | ExprKind::Match(..), .. }) => {
658 return Some(expr);
659 }
660 _ => {}
661 }
662 }
663 None
664 }
665
666 pub fn hir_get_enclosing_scope(self, hir_id: HirId) -> Option<HirId> {
668 for (hir_id, node) in self.hir_parent_iter(hir_id) {
669 if let Node::Item(Item {
670 kind:
671 ItemKind::Fn { .. }
672 | ItemKind::Const(..)
673 | ItemKind::Static(..)
674 | ItemKind::Mod(..)
675 | ItemKind::Enum(..)
676 | ItemKind::Struct(..)
677 | ItemKind::Union(..)
678 | ItemKind::Trait { .. }
679 | ItemKind::Impl { .. },
680 ..
681 })
682 | Node::ForeignItem(ForeignItem { kind: ForeignItemKind::Fn(..), .. })
683 | Node::TraitItem(TraitItem { kind: TraitItemKind::Fn(..), .. })
684 | Node::ImplItem(ImplItem { kind: ImplItemKind::Fn(..), .. })
685 | Node::Block(_) = node
686 {
687 return Some(hir_id);
688 }
689 }
690 None
691 }
692
693 pub fn hir_get_defining_scope(self, id: HirId) -> HirId {
695 let mut scope = id;
696 loop {
697 scope = self.hir_get_enclosing_scope(scope).unwrap_or(CRATE_HIR_ID);
698 if scope == CRATE_HIR_ID || !#[allow(non_exhaustive_omitted_patterns)] match self.hir_node(scope) {
Node::Block(_) => true,
_ => false,
}matches!(self.hir_node(scope), Node::Block(_)) {
699 return scope;
700 }
701 }
702 }
703
704 pub fn hir_id_to_string(self, id: HirId) -> String {
707 let path_str = |def_id: LocalDefId| self.def_path_str(def_id);
708
709 let span_str =
710 || self.sess.source_map().span_to_snippet(self.hir_span(id)).unwrap_or_default();
711 let node_str = |prefix| ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{1} ({2} `{0}`)", span_str(), id,
prefix))
})format!("{id} ({prefix} `{}`)", span_str());
712
713 match self.hir_node(id) {
714 Node::Item(item) => {
715 let item_str = match item.kind {
716 ItemKind::ExternCrate(..) => "extern crate",
717 ItemKind::Use(..) => "use",
718 ItemKind::Static(..) => "static",
719 ItemKind::Const(..) => "const",
720 ItemKind::Fn { .. } => "fn",
721 ItemKind::Macro(..) => "macro",
722 ItemKind::Mod(..) => "mod",
723 ItemKind::ForeignMod { .. } => "foreign mod",
724 ItemKind::GlobalAsm { .. } => "global asm",
725 ItemKind::TyAlias(..) => "ty",
726 ItemKind::Enum(..) => "enum",
727 ItemKind::Struct(..) => "struct",
728 ItemKind::Union(..) => "union",
729 ItemKind::Trait { .. } => "trait",
730 ItemKind::TraitAlias(..) => "trait alias",
731 ItemKind::Impl { .. } => "impl",
732 };
733 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{1} ({2} {0})",
path_str(item.owner_id.def_id), id, item_str))
})format!("{id} ({item_str} {})", path_str(item.owner_id.def_id))
734 }
735 Node::ForeignItem(item) => {
736 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{1} (foreign item {0})",
path_str(item.owner_id.def_id), id))
})format!("{id} (foreign item {})", path_str(item.owner_id.def_id))
737 }
738 Node::ImplItem(ii) => {
739 let kind = match ii.kind {
740 ImplItemKind::Const(..) => "associated constant",
741 ImplItemKind::Fn(fn_sig, _) => match fn_sig.decl.implicit_self() {
742 ImplicitSelfKind::None => "associated function",
743 _ => "method",
744 },
745 ImplItemKind::Type(_) => "associated type",
746 };
747 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{2} ({3} `{0}` in {1})", ii.ident,
path_str(ii.owner_id.def_id), id, kind))
})format!("{id} ({kind} `{}` in {})", ii.ident, path_str(ii.owner_id.def_id))
748 }
749 Node::TraitItem(ti) => {
750 let kind = match ti.kind {
751 TraitItemKind::Const(..) => "associated constant",
752 TraitItemKind::Fn(fn_sig, _) => match fn_sig.decl.implicit_self() {
753 ImplicitSelfKind::None => "associated function",
754 _ => "trait method",
755 },
756 TraitItemKind::Type(..) => "associated type",
757 };
758
759 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{2} ({3} `{0}` in {1})", ti.ident,
path_str(ti.owner_id.def_id), id, kind))
})format!("{id} ({kind} `{}` in {})", ti.ident, path_str(ti.owner_id.def_id))
760 }
761 Node::Variant(variant) => {
762 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{2} (variant `{0}` in {1})",
variant.ident, path_str(variant.def_id), id))
})format!("{id} (variant `{}` in {})", variant.ident, path_str(variant.def_id))
763 }
764 Node::Field(field) => {
765 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{2} (field `{0}` in {1})",
field.ident, path_str(field.def_id), id))
})format!("{id} (field `{}` in {})", field.ident, path_str(field.def_id))
766 }
767 Node::AnonConst(_) => node_str("const"),
768 Node::ConstBlock(_) => node_str("const"),
769 Node::ConstArg(_) => node_str("const"),
770 Node::Expr(_) => node_str("expr"),
771 Node::ExprField(_) => node_str("expr field"),
772 Node::ConstArgExprField(_) => node_str("const arg expr field"),
773 Node::Stmt(_) => node_str("stmt"),
774 Node::PathSegment(_) => node_str("path segment"),
775 Node::Ty(_) => node_str("type"),
776 Node::AssocItemConstraint(_) => node_str("assoc item constraint"),
777 Node::TraitRef(_) => node_str("trait ref"),
778 Node::OpaqueTy(_) => node_str("opaque type"),
779 Node::Pat(_) => node_str("pat"),
780 Node::TyPat(_) => node_str("pat ty"),
781 Node::PatField(_) => node_str("pattern field"),
782 Node::PatExpr(_) => node_str("pattern literal"),
783 Node::Param(_) => node_str("param"),
784 Node::Arm(_) => node_str("arm"),
785 Node::Block(_) => node_str("block"),
786 Node::Infer(_) => node_str("infer"),
787 Node::LetStmt(_) => node_str("local"),
788 Node::Ctor(ctor) => ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{1} (ctor {0})",
ctor.ctor_def_id().map_or("<missing path>".into(),
|def_id| path_str(def_id)), id))
})format!(
789 "{id} (ctor {})",
790 ctor.ctor_def_id().map_or("<missing path>".into(), |def_id| path_str(def_id)),
791 ),
792 Node::Lifetime(_) => node_str("lifetime"),
793 Node::GenericParam(param) => {
794 ::alloc::__export::must_use({
::alloc::fmt::format(format_args!("{1} (generic_param {0})",
path_str(param.def_id), id))
})format!("{id} (generic_param {})", path_str(param.def_id))
795 }
796 Node::Crate(..) => String::from("(root_crate)"),
797 Node::WherePredicate(_) => node_str("where predicate"),
798 Node::Synthetic => ::core::panicking::panic("internal error: entered unreachable code")unreachable!(),
799 Node::Err(_) => node_str("error"),
800 Node::PreciseCapturingNonLifetimeArg(_param) => node_str("parameter"),
801 }
802 }
803
804 pub fn hir_get_foreign_abi(self, hir_id: HirId) -> ExternAbi {
805 let parent = self.hir_get_parent_item(hir_id);
806 if let OwnerNode::Item(Item { kind: ItemKind::ForeignMod { abi, .. }, .. }) =
807 self.hir_owner_node(parent)
808 {
809 return *abi;
810 }
811 crate::util::bug::bug_fmt(format_args!("expected foreign mod or inlined parent, found {0}",
self.hir_id_to_string(HirId::make_owner(parent.def_id))))bug!(
812 "expected foreign mod or inlined parent, found {}",
813 self.hir_id_to_string(HirId::make_owner(parent.def_id))
814 )
815 }
816
817 pub fn hir_expect_item(self, id: LocalDefId) -> &'tcx Item<'tcx> {
818 match self.expect_hir_owner_node(id) {
819 OwnerNode::Item(item) => item,
820 _ => crate::util::bug::bug_fmt(format_args!("expected item, found {0}",
self.hir_id_to_string(HirId::make_owner(id))))bug!("expected item, found {}", self.hir_id_to_string(HirId::make_owner(id))),
821 }
822 }
823
824 pub fn hir_expect_impl_item(self, id: LocalDefId) -> &'tcx ImplItem<'tcx> {
825 match self.expect_hir_owner_node(id) {
826 OwnerNode::ImplItem(item) => item,
827 _ => crate::util::bug::bug_fmt(format_args!("expected impl item, found {0}",
self.hir_id_to_string(HirId::make_owner(id))))bug!("expected impl item, found {}", self.hir_id_to_string(HirId::make_owner(id))),
828 }
829 }
830
831 pub fn hir_expect_trait_item(self, id: LocalDefId) -> &'tcx TraitItem<'tcx> {
832 match self.expect_hir_owner_node(id) {
833 OwnerNode::TraitItem(item) => item,
834 _ => {
835 crate::util::bug::bug_fmt(format_args!("expected trait item, found {0}",
self.hir_id_to_string(HirId::make_owner(id))))bug!("expected trait item, found {}", self.hir_id_to_string(HirId::make_owner(id)))
836 }
837 }
838 }
839
840 pub fn hir_get_fn_output(self, def_id: LocalDefId) -> Option<&'tcx FnRetTy<'tcx>> {
841 Some(&self.opt_hir_owner_node(def_id)?.fn_decl()?.output)
842 }
843
844 #[track_caller]
845 pub fn hir_expect_opaque_ty(self, id: LocalDefId) -> &'tcx OpaqueTy<'tcx> {
846 match self.hir_node_by_def_id(id) {
847 Node::OpaqueTy(opaq) => opaq,
848 _ => {
849 crate::util::bug::bug_fmt(format_args!("expected opaque type definition, found {0}",
self.hir_id_to_string(self.local_def_id_to_hir_id(id))))bug!(
850 "expected opaque type definition, found {}",
851 self.hir_id_to_string(self.local_def_id_to_hir_id(id))
852 )
853 }
854 }
855 }
856
857 pub fn hir_expect_expr(self, id: HirId) -> &'tcx Expr<'tcx> {
858 match self.hir_node(id) {
859 Node::Expr(expr) => expr,
860 _ => crate::util::bug::bug_fmt(format_args!("expected expr, found {0}",
self.hir_id_to_string(id)))bug!("expected expr, found {}", self.hir_id_to_string(id)),
861 }
862 }
863
864 pub fn hir_opt_delegation_sig_id(self, def_id: LocalDefId) -> Option<DefId> {
865 self.opt_hir_owner_node(def_id)?.fn_decl()?.opt_delegation_sig_id()
866 }
867
868 #[inline]
869 fn hir_opt_ident(self, id: HirId) -> Option<Ident> {
870 match self.hir_node(id) {
871 Node::Pat(&Pat { kind: PatKind::Binding(_, _, ident, _), .. }) => Some(ident),
872 Node::Ctor(..) => match self.parent_hir_node(id) {
875 Node::Item(item) => Some(item.kind.ident().unwrap()),
876 Node::Variant(variant) => Some(variant.ident),
877 _ => ::core::panicking::panic("internal error: entered unreachable code")unreachable!(),
878 },
879 node => node.ident(),
880 }
881 }
882
883 #[inline]
884 pub(super) fn hir_opt_ident_span(self, id: HirId) -> Option<Span> {
885 self.hir_opt_ident(id).map(|ident| ident.span)
886 }
887
888 #[inline]
889 pub fn hir_ident(self, id: HirId) -> Ident {
890 self.hir_opt_ident(id).unwrap()
891 }
892
893 #[inline]
894 pub fn hir_opt_name(self, id: HirId) -> Option<Symbol> {
895 self.hir_opt_ident(id).map(|ident| ident.name)
896 }
897
898 pub fn hir_name(self, id: HirId) -> Symbol {
899 self.hir_opt_name(id).unwrap_or_else(|| crate::util::bug::bug_fmt(format_args!("no name for {0}",
self.hir_id_to_string(id)))bug!("no name for {}", self.hir_id_to_string(id)))
900 }
901
902 pub fn hir_attrs(self, id: HirId) -> &'tcx [Attribute] {
905 self.hir_attr_map(id.owner).get(id.local_id)
906 }
907
908 pub fn hir_span(self, hir_id: HirId) -> Span {
911 fn until_within(outer: Span, end: Span) -> Span {
912 if let Some(end) = end.find_ancestor_inside(outer) {
913 outer.with_hi(end.hi())
914 } else {
915 outer
916 }
917 }
918
919 fn named_span(item_span: Span, ident: Ident, generics: Option<&Generics<'_>>) -> Span {
920 let mut span = until_within(item_span, ident.span);
921 if let Some(g) = generics
922 && !g.span.is_dummy()
923 && let Some(g_span) = g.span.find_ancestor_inside(item_span)
924 {
925 span = span.to(g_span);
926 }
927 span
928 }
929
930 let span = match self.hir_node(hir_id) {
931 Node::Item(Item { kind: ItemKind::Fn { sig, .. }, span: outer_span, .. })
933 | Node::TraitItem(TraitItem {
934 kind: TraitItemKind::Fn(sig, ..),
935 span: outer_span,
936 ..
937 })
938 | Node::ImplItem(ImplItem {
939 kind: ImplItemKind::Fn(sig, ..), span: outer_span, ..
940 })
941 | Node::ForeignItem(ForeignItem {
942 kind: ForeignItemKind::Fn(sig, ..),
943 span: outer_span,
944 ..
945 }) => {
946 sig.span.find_ancestor_in_same_ctxt(*outer_span).unwrap_or(*outer_span)
949 }
950 Node::Item(Item {
952 kind: ItemKind::Impl(Impl { generics, .. }),
953 span: outer_span,
954 ..
955 }) => until_within(*outer_span, generics.where_clause_span),
956 Node::Item(Item {
958 kind: ItemKind::Const(_, _, ty, _) | ItemKind::Static(_, _, ty, _),
959 span: outer_span,
960 ..
961 })
962 | Node::TraitItem(TraitItem {
963 kind: TraitItemKind::Const(ty, ..),
964 span: outer_span,
965 ..
966 })
967 | Node::ImplItem(ImplItem {
968 kind: ImplItemKind::Const(ty, ..),
969 span: outer_span,
970 ..
971 })
972 | Node::ForeignItem(ForeignItem {
973 kind: ForeignItemKind::Static(ty, ..),
974 span: outer_span,
975 ..
976 }) => until_within(*outer_span, ty.span),
977 Node::Item(Item {
979 kind: ItemKind::Trait { generics, bounds, .. },
980 span: outer_span,
981 ..
982 })
983 | Node::TraitItem(TraitItem {
984 kind: TraitItemKind::Type(bounds, _),
985 generics,
986 span: outer_span,
987 ..
988 }) => {
989 let end = if let Some(b) = bounds.last() { b.span() } else { generics.span };
990 until_within(*outer_span, end)
991 }
992 Node::Item(item) => match &item.kind {
994 ItemKind::Use(path, _) => {
995 path.span.find_ancestor_in_same_ctxt(item.span).unwrap_or(item.span)
998 }
999 _ => {
1000 if let Some(ident) = item.kind.ident() {
1001 named_span(item.span, ident, item.kind.generics())
1002 } else {
1003 item.span
1004 }
1005 }
1006 },
1007 Node::Variant(variant) => named_span(variant.span, variant.ident, None),
1008 Node::ImplItem(item) => named_span(item.span, item.ident, Some(item.generics)),
1009 Node::ForeignItem(item) => named_span(item.span, item.ident, None),
1010 Node::Ctor(_) => return self.hir_span(self.parent_hir_id(hir_id)),
1011 Node::Expr(Expr {
1012 kind: ExprKind::Closure(Closure { fn_decl_span, .. }),
1013 span,
1014 ..
1015 }) => {
1016 fn_decl_span.find_ancestor_inside_same_ctxt(*span).unwrap_or(*span)
1018 }
1019 _ => self.hir_span_with_body(hir_id),
1020 };
1021 if true {
match (&span.ctxt(), &self.hir_span_with_body(hir_id).ctxt()) {
(left_val, right_val) => {
if !(*left_val == *right_val) {
let kind = ::core::panicking::AssertKind::Eq;
::core::panicking::assert_failed(kind, &*left_val,
&*right_val, ::core::option::Option::None);
}
}
};
};debug_assert_eq!(span.ctxt(), self.hir_span_with_body(hir_id).ctxt());
1022 span
1023 }
1024
1025 pub fn hir_span_with_body(self, hir_id: HirId) -> Span {
1028 match self.hir_node(hir_id) {
1029 Node::Param(param) => param.span,
1030 Node::Item(item) => item.span,
1031 Node::ForeignItem(foreign_item) => foreign_item.span,
1032 Node::TraitItem(trait_item) => trait_item.span,
1033 Node::ImplItem(impl_item) => impl_item.span,
1034 Node::Variant(variant) => variant.span,
1035 Node::Field(field) => field.span,
1036 Node::AnonConst(constant) => constant.span,
1037 Node::ConstBlock(constant) => self.hir_body(constant.body).value.span,
1038 Node::ConstArg(const_arg) => const_arg.span,
1039 Node::Expr(expr) => expr.span,
1040 Node::ExprField(field) => field.span,
1041 Node::ConstArgExprField(field) => field.span,
1042 Node::Stmt(stmt) => stmt.span,
1043 Node::PathSegment(seg) => {
1044 let ident_span = seg.ident.span;
1045 ident_span
1046 .with_hi(seg.args.map_or_else(|| ident_span.hi(), |args| args.span_ext.hi()))
1047 }
1048 Node::Ty(ty) => ty.span,
1049 Node::AssocItemConstraint(constraint) => constraint.span,
1050 Node::TraitRef(tr) => tr.path.span,
1051 Node::OpaqueTy(op) => op.span,
1052 Node::Pat(pat) => pat.span,
1053 Node::TyPat(pat) => pat.span,
1054 Node::PatField(field) => field.span,
1055 Node::PatExpr(lit) => lit.span,
1056 Node::Arm(arm) => arm.span,
1057 Node::Block(block) => block.span,
1058 Node::Ctor(..) => self.hir_span_with_body(self.parent_hir_id(hir_id)),
1059 Node::Lifetime(lifetime) => lifetime.ident.span,
1060 Node::GenericParam(param) => param.span,
1061 Node::Infer(i) => i.span,
1062 Node::LetStmt(local) => local.span,
1063 Node::Crate(item) => item.spans.inner_span,
1064 Node::WherePredicate(pred) => pred.span,
1065 Node::PreciseCapturingNonLifetimeArg(param) => param.ident.span,
1066 Node::Synthetic => ::core::panicking::panic("internal error: entered unreachable code")unreachable!(),
1067 Node::Err(span) => span,
1068 }
1069 }
1070
1071 pub fn hir_span_if_local(self, id: DefId) -> Option<Span> {
1072 id.is_local().then(|| self.def_span(id))
1073 }
1074
1075 pub fn hir_res_span(self, res: Res) -> Option<Span> {
1076 match res {
1077 Res::Err => None,
1078 Res::Local(id) => Some(self.hir_span(id)),
1079 res => self.hir_span_if_local(res.opt_def_id()?),
1080 }
1081 }
1082
1083 pub fn hir_opt_const_param_default_param_def_id(self, anon_const: HirId) -> Option<LocalDefId> {
1086 let const_arg = self.parent_hir_id(anon_const);
1087 match self.parent_hir_node(const_arg) {
1088 Node::GenericParam(GenericParam {
1089 def_id: param_id,
1090 kind: GenericParamKind::Const { .. },
1091 ..
1092 }) => Some(*param_id),
1093 _ => None,
1094 }
1095 }
1096
1097 pub fn hir_maybe_get_struct_pattern_shorthand_field(self, expr: &Expr<'_>) -> Option<Symbol> {
1098 let local = match expr {
1099 Expr {
1100 kind:
1101 ExprKind::Path(QPath::Resolved(
1102 None,
1103 Path {
1104 res: def::Res::Local(_), segments: [PathSegment { ident, .. }], ..
1105 },
1106 )),
1107 ..
1108 } => Some(ident),
1109 _ => None,
1110 }?;
1111
1112 match self.parent_hir_node(expr.hir_id) {
1113 Node::ExprField(field) => {
1114 if field.ident.name == local.name && field.is_shorthand {
1115 return Some(local.name);
1116 }
1117 }
1118 _ => {}
1119 }
1120
1121 None
1122 }
1123}
1124
1125impl<'tcx> intravisit::HirTyCtxt<'tcx> for TyCtxt<'tcx> {
1126 fn hir_node(&self, hir_id: HirId) -> Node<'tcx> {
1127 (*self).hir_node(hir_id)
1128 }
1129
1130 fn hir_body(&self, id: BodyId) -> &'tcx Body<'tcx> {
1131 (*self).hir_body(id)
1132 }
1133
1134 fn hir_item(&self, id: ItemId) -> &'tcx Item<'tcx> {
1135 (*self).hir_item(id)
1136 }
1137
1138 fn hir_trait_item(&self, id: TraitItemId) -> &'tcx TraitItem<'tcx> {
1139 (*self).hir_trait_item(id)
1140 }
1141
1142 fn hir_impl_item(&self, id: ImplItemId) -> &'tcx ImplItem<'tcx> {
1143 (*self).hir_impl_item(id)
1144 }
1145
1146 fn hir_foreign_item(&self, id: ForeignItemId) -> &'tcx ForeignItem<'tcx> {
1147 (*self).hir_foreign_item(id)
1148 }
1149}
1150
1151impl<'tcx> pprust_hir::PpAnn for TyCtxt<'tcx> {
1152 fn nested(&self, state: &mut pprust_hir::State<'_>, nested: pprust_hir::Nested) {
1153 pprust_hir::PpAnn::nested(&(self as &dyn intravisit::HirTyCtxt<'_>), state, nested)
1154 }
1155}
1156
1157pub(super) fn crate_hash(tcx: TyCtxt<'_>, _: LocalCrate) -> Svh {
1158 let krate = tcx.hir_crate(());
1159 let hir_body_hash = krate.opt_hir_hash.expect("HIR hash missing while computing crate hash");
1160
1161 let upstream_crates = upstream_crates(tcx);
1162
1163 let resolutions = tcx.resolutions(());
1164
1165 let mut source_file_names: Vec<_> = tcx
1171 .sess
1172 .source_map()
1173 .files()
1174 .iter()
1175 .filter(|source_file| source_file.cnum == LOCAL_CRATE)
1176 .map(|source_file| source_file.stable_id)
1177 .collect();
1178
1179 source_file_names.sort_unstable();
1180
1181 let debugger_visualizers: Vec<_> = tcx
1187 .debugger_visualizers(LOCAL_CRATE)
1188 .iter()
1189 .map(DebuggerVisualizerFile::path_erased)
1193 .collect();
1194
1195 let crate_hash: Fingerprint = tcx.with_stable_hashing_context(|mut hcx| {
1196 let mut stable_hasher = StableHasher::new();
1197 hir_body_hash.stable_hash(&mut hcx, &mut stable_hasher);
1198 upstream_crates.stable_hash(&mut hcx, &mut stable_hasher);
1199 source_file_names.stable_hash(&mut hcx, &mut stable_hasher);
1200 debugger_visualizers.stable_hash(&mut hcx, &mut stable_hasher);
1201 if tcx.sess.opts.incremental.is_some() {
1202 let definitions = tcx.untracked().definitions.freeze();
1203 let mut owner_spans: Vec<_> = tcx
1204 .hir_crate_items(())
1205 .definitions()
1206 .map(|def_id| {
1207 let def_path_hash = definitions.def_path_hash(def_id);
1208 let span = tcx.source_span(def_id);
1209 if true {
match (&span.parent(), &None) {
(left_val, right_val) => {
if !(*left_val == *right_val) {
let kind = ::core::panicking::AssertKind::Eq;
::core::panicking::assert_failed(kind, &*left_val,
&*right_val, ::core::option::Option::None);
}
}
};
};debug_assert_eq!(span.parent(), None);
1210 (def_path_hash, span)
1211 })
1212 .collect();
1213 owner_spans.sort_unstable_by_key(|bn| bn.0);
1214 owner_spans.stable_hash(&mut hcx, &mut stable_hasher);
1215 }
1216 tcx.sess.opts.dep_tracking_hash(true).stable_hash(&mut hcx, &mut stable_hasher);
1217 tcx.stable_crate_id(LOCAL_CRATE).stable_hash(&mut hcx, &mut stable_hasher);
1218 resolutions.visibilities_for_hashing.stable_hash(&mut hcx, &mut stable_hasher);
1223 with_metavar_spans(|mspans| {
1224 mspans.freeze_and_get_read_spans().stable_hash(&mut hcx, &mut stable_hasher);
1225 });
1226 stable_hasher.finish()
1227 });
1228
1229 Svh::new(crate_hash)
1230}
1231
1232fn upstream_crates(tcx: TyCtxt<'_>) -> Vec<(StableCrateId, Svh)> {
1233 let mut upstream_crates: Vec<_> = tcx
1234 .crates(())
1235 .iter()
1236 .map(|&cnum| {
1237 let stable_crate_id = tcx.stable_crate_id(cnum);
1238 let hash = tcx.crate_hash(cnum);
1239 (stable_crate_id, hash)
1240 })
1241 .collect();
1242 upstream_crates.sort_unstable_by_key(|&(stable_crate_id, _)| stable_crate_id);
1243 upstream_crates
1244}
1245
1246pub(super) fn hir_module_items(tcx: TyCtxt<'_>, module_id: LocalModDefId) -> ModuleItems {
1247 let mut collector = ItemCollector::new(tcx, false);
1248
1249 let (hir_mod, span, hir_id) = tcx.hir_get_module(module_id);
1250 collector.visit_mod(hir_mod, span, hir_id);
1251
1252 let ItemCollector {
1253 submodules,
1254 items,
1255 trait_items,
1256 impl_items,
1257 foreign_items,
1258 body_owners,
1259 opaques,
1260 nested_bodies,
1261 eiis,
1262 ..
1263 } = collector;
1264 ModuleItems {
1265 add_root: false,
1266 submodules: submodules.into_boxed_slice(),
1267 free_items: items.into_boxed_slice(),
1268 trait_items: trait_items.into_boxed_slice(),
1269 impl_items: impl_items.into_boxed_slice(),
1270 foreign_items: foreign_items.into_boxed_slice(),
1271 body_owners: body_owners.into_boxed_slice(),
1272 opaques: opaques.into_boxed_slice(),
1273 nested_bodies: nested_bodies.into_boxed_slice(),
1274 delayed_lint_items: Box::new([]),
1275 eiis: eiis.into_boxed_slice(),
1276 }
1277}
1278
1279fn force_delayed_owners_lowering(tcx: TyCtxt<'_>) {
1280 let krate = tcx.hir_crate(());
1281 for &id in &krate.delayed_ids {
1282 tcx.ensure_done().lower_delayed_owner(id);
1283 }
1284
1285 let (_, krate) = krate.delayed_resolver.steal();
1286 let prof = tcx.sess.prof.clone();
1287
1288 spawn(move || {
1290 let _timer = prof.verbose_generic_activity("drop_ast");
1291 drop(krate);
1292 });
1293}
1294
1295pub(crate) fn hir_crate_items(tcx: TyCtxt<'_>, _: ()) -> ModuleItems {
1296 force_delayed_owners_lowering(tcx);
1297
1298 let mut collector = ItemCollector::new(tcx, true);
1299
1300 collector.submodules.push(CRATE_OWNER_ID);
1304 tcx.hir_walk_toplevel_module(&mut collector);
1305
1306 let ItemCollector {
1307 submodules,
1308 items,
1309 trait_items,
1310 impl_items,
1311 foreign_items,
1312 body_owners,
1313 opaques,
1314 nested_bodies,
1315 mut delayed_lint_items,
1316 eiis,
1317 ..
1318 } = collector;
1319
1320 delayed_lint_items.push(CRATE_OWNER_ID);
1327
1328 ModuleItems {
1329 add_root: true,
1330 submodules: submodules.into_boxed_slice(),
1331 free_items: items.into_boxed_slice(),
1332 trait_items: trait_items.into_boxed_slice(),
1333 impl_items: impl_items.into_boxed_slice(),
1334 foreign_items: foreign_items.into_boxed_slice(),
1335 body_owners: body_owners.into_boxed_slice(),
1336 opaques: opaques.into_boxed_slice(),
1337 nested_bodies: nested_bodies.into_boxed_slice(),
1338 delayed_lint_items: delayed_lint_items.into_boxed_slice(),
1339 eiis: eiis.into_boxed_slice(),
1340 }
1341}
1342
1343struct ItemCollector<'tcx> {
1344 crate_collector: bool,
1347 tcx: TyCtxt<'tcx>,
1348 submodules: Vec<OwnerId>,
1349 items: Vec<ItemId>,
1350 trait_items: Vec<TraitItemId>,
1351 impl_items: Vec<ImplItemId>,
1352 foreign_items: Vec<ForeignItemId>,
1353 body_owners: Vec<LocalDefId>,
1354 opaques: Vec<LocalDefId>,
1355 nested_bodies: Vec<LocalDefId>,
1356 delayed_lint_items: Vec<OwnerId>,
1357 eiis: Vec<LocalDefId>,
1358}
1359
1360impl<'tcx> ItemCollector<'tcx> {
1361 fn new(tcx: TyCtxt<'tcx>, crate_collector: bool) -> ItemCollector<'tcx> {
1362 ItemCollector {
1363 crate_collector,
1364 tcx,
1365 submodules: Vec::default(),
1366 items: Vec::default(),
1367 trait_items: Vec::default(),
1368 impl_items: Vec::default(),
1369 foreign_items: Vec::default(),
1370 body_owners: Vec::default(),
1371 opaques: Vec::default(),
1372 nested_bodies: Vec::default(),
1373 delayed_lint_items: Vec::default(),
1374 eiis: Vec::default(),
1375 }
1376 }
1377}
1378
1379impl<'hir> Visitor<'hir> for ItemCollector<'hir> {
1380 type NestedFilter = nested_filter::All;
1381
1382 fn maybe_tcx(&mut self) -> Self::MaybeTyCtxt {
1383 self.tcx
1384 }
1385
1386 fn visit_item(&mut self, item: &'hir Item<'hir>) {
1387 if Node::Item(item).associated_body().is_some() {
1388 self.body_owners.push(item.owner_id.def_id);
1389 }
1390
1391 self.items.push(item.item_id());
1392 if self.crate_collector && item.has_delayed_lints {
1393 self.delayed_lint_items.push(item.item_id().owner_id);
1394 }
1395
1396 if let ItemKind::Static(..) | ItemKind::Fn { .. } | ItemKind::Macro(..) = &item.kind
1397 && item.eii
1398 {
1399 self.eiis.push(item.owner_id.def_id)
1400 }
1401
1402 if let ItemKind::Mod(_, module) = &item.kind {
1404 self.submodules.push(item.owner_id);
1405 if self.crate_collector {
1407 intravisit::walk_mod(self, module);
1408 }
1409 } else {
1410 intravisit::walk_item(self, item)
1411 }
1412 }
1413
1414 fn visit_foreign_item(&mut self, item: &'hir ForeignItem<'hir>) {
1415 self.foreign_items.push(item.foreign_item_id());
1416 if self.crate_collector && item.has_delayed_lints {
1417 self.delayed_lint_items.push(item.foreign_item_id().owner_id);
1418 }
1419 intravisit::walk_foreign_item(self, item)
1420 }
1421
1422 fn visit_anon_const(&mut self, c: &'hir AnonConst) {
1423 self.body_owners.push(c.def_id);
1424 intravisit::walk_anon_const(self, c)
1425 }
1426
1427 fn visit_inline_const(&mut self, c: &'hir ConstBlock) {
1428 self.body_owners.push(c.def_id);
1429 self.nested_bodies.push(c.def_id);
1430 intravisit::walk_inline_const(self, c)
1431 }
1432
1433 fn visit_opaque_ty(&mut self, o: &'hir OpaqueTy<'hir>) {
1434 self.opaques.push(o.def_id);
1435 intravisit::walk_opaque_ty(self, o)
1436 }
1437
1438 fn visit_expr(&mut self, ex: &'hir Expr<'hir>) {
1439 if let ExprKind::Closure(closure) = ex.kind {
1440 self.body_owners.push(closure.def_id);
1441 self.nested_bodies.push(closure.def_id);
1442 }
1443 intravisit::walk_expr(self, ex)
1444 }
1445
1446 fn visit_trait_item(&mut self, item: &'hir TraitItem<'hir>) {
1447 if Node::TraitItem(item).associated_body().is_some() {
1448 self.body_owners.push(item.owner_id.def_id);
1449 }
1450
1451 self.trait_items.push(item.trait_item_id());
1452 if self.crate_collector && item.has_delayed_lints {
1453 self.delayed_lint_items.push(item.trait_item_id().owner_id);
1454 }
1455
1456 intravisit::walk_trait_item(self, item)
1457 }
1458
1459 fn visit_impl_item(&mut self, item: &'hir ImplItem<'hir>) {
1460 if Node::ImplItem(item).associated_body().is_some() {
1461 self.body_owners.push(item.owner_id.def_id);
1462 }
1463
1464 self.impl_items.push(item.impl_item_id());
1465 if self.crate_collector && item.has_delayed_lints {
1466 self.delayed_lint_items.push(item.impl_item_id().owner_id);
1467 }
1468
1469 intravisit::walk_impl_item(self, item)
1470 }
1471}