1//! Helper functions corresponding to lifetime errors due to
2//! anonymous regions.
34use rustc_hiras hir;
5use rustc_hir::def_id::{DefId, LocalDefId};
6use rustc_middle::ty::{self, Binder, Region, Ty, TyCtxt, TypeFoldable, fold_regions};
7use rustc_span::Span;
8use tracing::instrument;
910use crate::error_reporting::infer::nice_region_error::NiceRegionError;
1112/// Information about the anonymous region we are searching for.
13#[derive(#[automatically_derived]
impl<'tcx> ::core::fmt::Debug for AnonymousParamInfo<'tcx> {
#[inline]
fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
::core::fmt::Formatter::debug_struct_field5_finish(f,
"AnonymousParamInfo", "param", &self.param, "param_ty",
&self.param_ty, "kind", &self.kind, "param_ty_span",
&self.param_ty_span, "is_first", &&self.is_first)
}
}Debug)]
14pub struct AnonymousParamInfo<'tcx> {
15/// The parameter corresponding to the anonymous region.
16pub param: &'tcx hir::Param<'tcx>,
17/// The type corresponding to the anonymous region parameter.
18pub param_ty: Ty<'tcx>,
19/// The `ty::LateParamRegionKind` corresponding to the anonymous region.
20pub kind: ty::LateParamRegionKind,
21/// The `Span` of the parameter type.
22pub param_ty_span: Span,
23/// Signals that the argument is the first parameter in the declaration.
24pub is_first: bool,
25}
2627// This method walks the Type of the function body parameters using
28// `fold_regions()` function and returns the
29// &hir::Param of the function parameter corresponding to the anonymous
30// region and the Ty corresponding to the named region.
31// Currently only the case where the function declaration consists of
32// one named region and one anonymous region is handled.
33// Consider the example `fn foo<'a>(x: &'a i32, y: &i32) -> &'a i32`
34// Here, we would return the hir::Param for y, we return the type &'a
35// i32, which is the type of y but with the anonymous region replaced
36// with 'a, the corresponding bound region and is_first which is true if
37// the hir::Param is the first parameter in the function declaration.
38#[allow(clippy :: suspicious_else_formatting)]
{
let __tracing_attr_span;
let __tracing_attr_guard;
if ::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() ||
{ false } {
__tracing_attr_span =
{
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("find_param_with_region",
"rustc_trait_selection::error_reporting::infer::nice_region_error::util",
::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_trait_selection/src/error_reporting/infer/nice_region_error/util.rs"),
::tracing_core::__macro_support::Option::Some(38u32),
::tracing_core::__macro_support::Option::Some("rustc_trait_selection::error_reporting::infer::nice_region_error::util"),
::tracing_core::field::FieldSet::new(&["generic_param_scope",
"anon_region", "replace_region"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::SPAN)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let mut interest = ::tracing::subscriber::Interest::never();
if ::tracing::Level::DEBUG <=
::tracing::level_filters::STATIC_MAX_LEVEL &&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() &&
{ interest = __CALLSITE.interest(); !interest.is_never() }
&&
::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
interest) {
let meta = __CALLSITE.metadata();
::tracing::Span::new(meta,
&{
#[allow(unused_imports)]
use ::tracing::field::{debug, display, Value};
let mut iter = meta.fields().iter();
meta.fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&::tracing::field::debug(&generic_param_scope)
as &dyn Value)),
(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&::tracing::field::debug(&anon_region)
as &dyn Value)),
(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&::tracing::field::debug(&replace_region)
as &dyn Value))])
})
} else {
let span =
::tracing::__macro_support::__disabled_span(__CALLSITE.metadata());
{};
span
}
};
__tracing_attr_guard = __tracing_attr_span.enter();
}
#[warn(clippy :: suspicious_else_formatting)]
{
#[allow(unknown_lints, unreachable_code, clippy ::
diverging_sub_expression, clippy :: empty_loop, clippy ::
let_unit_value, clippy :: let_with_type_underscore, clippy ::
needless_return, clippy :: unreachable)]
if false {
let __tracing_attr_fake_return: Option<AnonymousParamInfo<'tcx>> =
loop {};
return __tracing_attr_fake_return;
}
{
let (id, kind) =
match anon_region.kind() {
ty::ReLateParam(late_param) =>
(late_param.scope, late_param.kind),
ty::ReEarlyParam(ebr) => {
let region_def =
tcx.generics_of(generic_param_scope).region_param(ebr,
tcx).def_id;
(tcx.parent(region_def),
ty::LateParamRegionKind::Named(region_def))
}
_ => return None,
};
let def_id = id.as_local()?;
match tcx.hir_node_by_def_id(generic_param_scope) {
hir::Node::Expr(&hir::Expr {
kind: hir::ExprKind::Closure { .. }, .. }) => {
return None;
}
_ => {}
}
let body = tcx.hir_maybe_body_owned_by(def_id)?;
let owner_id = tcx.hir_body_owner(body.id());
let fn_decl = tcx.hir_fn_decl_by_hir_id(owner_id)?;
let poly_fn_sig = tcx.fn_sig(id).instantiate_identity();
let fn_sig = tcx.liberate_late_bound_regions(id, poly_fn_sig);
body.params.iter().take(if fn_sig.c_variadic {
fn_sig.inputs().len()
} else {
match (&fn_sig.inputs().len(), &body.params.len()) {
(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);
}
}
};
body.params.len()
}).enumerate().find_map(|(index, param)|
{
let ty = fn_sig.inputs()[index];
let mut found_anon_region = false;
let new_param_ty =
fold_regions(tcx, ty,
|r, _|
{
if r == anon_region {
found_anon_region = true;
replace_region
} else { r }
});
found_anon_region.then(||
{
let ty_hir_id = fn_decl.inputs[index].hir_id;
let param_ty_span = tcx.hir_span(ty_hir_id);
let is_first = index == 0;
AnonymousParamInfo {
param,
param_ty: new_param_ty,
param_ty_span,
kind,
is_first,
}
})
})
}
}
}#[instrument(skip(tcx), level = "debug")]39pub fn find_param_with_region<'tcx>(
40 tcx: TyCtxt<'tcx>,
41 generic_param_scope: LocalDefId,
42 anon_region: Region<'tcx>,
43 replace_region: Region<'tcx>,
44) -> Option<AnonymousParamInfo<'tcx>> {
45let (id, kind) = match anon_region.kind() {
46 ty::ReLateParam(late_param) => (late_param.scope, late_param.kind),
47 ty::ReEarlyParam(ebr) => {
48let region_def = tcx.generics_of(generic_param_scope).region_param(ebr, tcx).def_id;
49 (tcx.parent(region_def), ty::LateParamRegionKind::Named(region_def))
50 }
51_ => return None, // not a free region
52};
5354let def_id = id.as_local()?;
5556// FIXME: use def_kind
57 // Don't perform this on closures
58match tcx.hir_node_by_def_id(generic_param_scope) {
59 hir::Node::Expr(&hir::Expr { kind: hir::ExprKind::Closure { .. }, .. }) => {
60return None;
61 }
62_ => {}
63 }
6465let body = tcx.hir_maybe_body_owned_by(def_id)?;
6667let owner_id = tcx.hir_body_owner(body.id());
68let fn_decl = tcx.hir_fn_decl_by_hir_id(owner_id)?;
69let poly_fn_sig = tcx.fn_sig(id).instantiate_identity();
7071let fn_sig = tcx.liberate_late_bound_regions(id, poly_fn_sig);
72 body.params
73 .iter()
74 .take(if fn_sig.c_variadic {
75 fn_sig.inputs().len()
76 } else {
77assert_eq!(fn_sig.inputs().len(), body.params.len());
78 body.params.len()
79 })
80 .enumerate()
81 .find_map(|(index, param)| {
82// May return None; sometimes the tables are not yet populated.
83let ty = fn_sig.inputs()[index];
84let mut found_anon_region = false;
85let new_param_ty = fold_regions(tcx, ty, |r, _| {
86if r == anon_region {
87 found_anon_region = true;
88 replace_region
89 } else {
90 r
91 }
92 });
93 found_anon_region.then(|| {
94let ty_hir_id = fn_decl.inputs[index].hir_id;
95let param_ty_span = tcx.hir_span(ty_hir_id);
96let is_first = index == 0;
97 AnonymousParamInfo { param, param_ty: new_param_ty, param_ty_span, kind, is_first }
98 })
99 })
100}
101102impl<'a, 'tcx> NiceRegionError<'a, 'tcx> {
103pub(super) fn find_param_with_region(
104&self,
105 anon_region: Region<'tcx>,
106 replace_region: Region<'tcx>,
107 ) -> Option<AnonymousParamInfo<'tcx>> {
108find_param_with_region(self.tcx(), self.generic_param_scope, anon_region, replace_region)
109 }
110111// Here, we check for the case where the anonymous region
112 // is in the return type as written by the user.
113 // FIXME(#42703) - Need to handle certain cases here.
114pub(super) fn is_return_type_anon(
115&self,
116 scope_def_id: LocalDefId,
117 region_def_id: DefId,
118 hir_sig: &hir::FnSig<'_>,
119 ) -> Option<Span> {
120let fn_ty = self.tcx().type_of(scope_def_id).instantiate_identity();
121if let ty::FnDef(_, _) = fn_ty.kind() {
122let ret_ty = fn_ty.fn_sig(self.tcx()).output();
123let span = hir_sig.decl.output.span();
124let future_output = if hir_sig.header.is_async() {
125ret_ty.map_bound(|ty| self.cx.get_impl_future_output_ty(ty)).transpose()
126 } else {
127None128 };
129return match future_output {
130Some(output) if self.includes_region(output, region_def_id) => Some(span),
131Noneif self.includes_region(ret_ty, region_def_id) => Some(span),
132_ => None,
133 };
134 }
135None136 }
137138fn includes_region(
139&self,
140 ty: Binder<'tcx, impl TypeFoldable<TyCtxt<'tcx>>>,
141 region_def_id: DefId,
142 ) -> bool {
143let late_bound_regions = self.tcx().collect_referenced_late_bound_regions(ty);
144// We are only checking is any region meets the condition so order doesn't matter
145#[allow(rustc::potential_query_instability)]
146late_bound_regions.iter().any(|r| match *r {
147 ty::BoundRegionKind::Named(def_id) => def_id == region_def_id,
148_ => false,
149 })
150 }
151152// Here we check for the case where anonymous region
153 // corresponds to self and if yes, we display E0312.
154 // FIXME(#42700) - Need to format self properly to
155 // enable E0621 for it.
156pub(super) fn is_self_anon(&self, is_first: bool, scope_def_id: LocalDefId) -> bool {
157is_first158 && self159 .tcx()
160 .opt_associated_item(scope_def_id.to_def_id())
161 .is_some_and(|i| i.is_method())
162 }
163}