1//! Deeply normalize types using the old trait solver.
23use rustc_data_structures::stack::ensure_sufficient_stack;
4use rustc_errors::msg;
5use rustc_infer::infer::at::At;
6use rustc_infer::infer::{InferCtxt, InferOk};
7use rustc_infer::traits::{
8FromSolverError, Normalized, Obligation, PredicateObligations, TraitEngine,
9};
10use rustc_macros::extension;
11use rustc_middle::span_bug;
12use rustc_middle::traits::{ObligationCause, ObligationCauseCode};
13use rustc_middle::ty::{
14self, AliasTerm, Term, Ty, TyCtxt, TypeFoldable, TypeFolder, TypeSuperFoldable, TypeVisitable,
15TypeVisitableExt, TypingMode, Unnormalized,
16};
17use tracing::{debug, instrument};
1819use super::{BoundVarReplacer, PlaceholderReplacer, SelectionContext, project};
20use crate::error_reporting::InferCtxtErrorExt;
21use crate::error_reporting::traits::OverflowCause;
22use crate::solve::NextSolverError;
2324impl<'tcx> NormalizeExt<'tcx> for At<'_, 'tcx> {
#[doc = " Normalize a value using the `AssocTypeNormalizer`."]
#[doc = ""]
#[doc =
" This normalization should be used when the type contains inference variables or the"]
#[doc = " projection may be fallible."]
fn normalize<T: TypeFoldable<TyCtxt<'tcx>>>(&self,
value: Unnormalized<'tcx, T>) -> InferOk<'tcx, T> {
if self.infcx.next_trait_solver() {
let Normalized { value, obligations } =
crate::solve::normalize(*self, value);
InferOk { value, obligations }
} else {
let value = value.skip_normalization();
let mut selcx = SelectionContext::new(self.infcx);
let Normalized { value, obligations } =
normalize_with_depth(&mut selcx, self.param_env,
self.cause.clone(), 0, value);
InferOk { value, obligations }
}
}
#[doc =
" Deeply normalizes `value`, replacing all aliases which can by normalized in"]
#[doc =
" the current environment. In the new solver this errors in case normalization"]
#[doc = " fails or is ambiguous."]
#[doc = ""]
#[doc =
" In the old solver this simply uses `normalizes` and adds the nested obligations"]
#[doc =
" to the `fulfill_cx`. This is necessary as we otherwise end up recomputing the"]
#[doc =
" same goals in both a temporary and the shared context which negatively impacts"]
#[doc = " performance as these don\'t share caching."]
#[doc = ""]
#[doc =
" FIXME(-Znext-solver=no): For performance reasons, we currently reuse an existing"]
#[doc =
" fulfillment context in the old solver. Once we have removed the old solver, we"]
#[doc = " can remove the `fulfill_cx` parameter on this function."]
fn deeply_normalize<T,
E>(self, value: Unnormalized<'tcx, T>,
fulfill_cx: &mut dyn TraitEngine<'tcx, E>) -> Result<T, Vec<E>> where
T: TypeFoldable<TyCtxt<'tcx>>,
E: FromSolverError<'tcx, NextSolverError<'tcx>> {
if self.infcx.next_trait_solver() {
crate::solve::deeply_normalize(self, value)
} else {
if fulfill_cx.has_pending_obligations() {
let pending_obligations = fulfill_cx.pending_obligations();
::rustc_middle::util::bug::span_bug_fmt(pending_obligations[0].cause.span,
format_args!("deeply_normalize should not be called with pending obligations: {0:#?}",
pending_obligations));
}
let value =
self.normalize(value).into_value_registering_obligations(self.infcx,
&mut *fulfill_cx);
let errors =
fulfill_cx.evaluate_obligations_error_on_ambiguity(self.infcx);
let value = self.infcx.resolve_vars_if_possible(value);
if errors.is_empty() {
Ok(value)
} else {
let _ = fulfill_cx.collect_remaining_errors(self.infcx);
Err(errors)
}
}
}
}#[extension(pub trait NormalizeExt<'tcx>)]25impl<'tcx> At<'_, 'tcx> {
26/// Normalize a value using the `AssocTypeNormalizer`.
27 ///
28 /// This normalization should be used when the type contains inference variables or the
29 /// projection may be fallible.
30fn normalize<T: TypeFoldable<TyCtxt<'tcx>>>(
31&self,
32 value: Unnormalized<'tcx, T>,
33 ) -> InferOk<'tcx, T> {
34if self.infcx.next_trait_solver() {
35let Normalized { value, obligations } = crate::solve::normalize(*self, value);
36InferOk { value, obligations }
37 } else {
38let value = value.skip_normalization();
39let mut selcx = SelectionContext::new(self.infcx);
40let Normalized { value, obligations } =
41normalize_with_depth(&mut selcx, self.param_env, self.cause.clone(), 0, value);
42InferOk { value, obligations }
43 }
44 }
4546/// Deeply normalizes `value`, replacing all aliases which can by normalized in
47 /// the current environment. In the new solver this errors in case normalization
48 /// fails or is ambiguous.
49 ///
50 /// In the old solver this simply uses `normalizes` and adds the nested obligations
51 /// to the `fulfill_cx`. This is necessary as we otherwise end up recomputing the
52 /// same goals in both a temporary and the shared context which negatively impacts
53 /// performance as these don't share caching.
54 ///
55 /// FIXME(-Znext-solver=no): For performance reasons, we currently reuse an existing
56 /// fulfillment context in the old solver. Once we have removed the old solver, we
57 /// can remove the `fulfill_cx` parameter on this function.
58fn deeply_normalize<T, E>(
59self,
60 value: Unnormalized<'tcx, T>,
61 fulfill_cx: &mut dyn TraitEngine<'tcx, E>,
62 ) -> Result<T, Vec<E>>
63where
64T: TypeFoldable<TyCtxt<'tcx>>,
65 E: FromSolverError<'tcx, NextSolverError<'tcx>>,
66 {
67if self.infcx.next_trait_solver() {
68crate::solve::deeply_normalize(self, value)
69 } else {
70if fulfill_cx.has_pending_obligations() {
71let pending_obligations = fulfill_cx.pending_obligations();
72span_bug!(
73 pending_obligations[0].cause.span,
74"deeply_normalize should not be called with pending obligations: \
75 {pending_obligations:#?}"
76);
77 }
78let value = self79 .normalize(value)
80 .into_value_registering_obligations(self.infcx, &mut *fulfill_cx);
81let errors = fulfill_cx.evaluate_obligations_error_on_ambiguity(self.infcx);
82let value = self.infcx.resolve_vars_if_possible(value);
83if errors.is_empty() {
84Ok(value)
85 } else {
86// Drop pending obligations, since deep normalization may happen
87 // in a loop and we don't want to trigger the assertion on the next
88 // iteration due to pending ambiguous obligations we've left over.
89let _ = fulfill_cx.collect_remaining_errors(self.infcx);
90Err(errors)
91 }
92 }
93 }
94}
9596/// As `normalize`, but with a custom depth.
97pub(crate) fn normalize_with_depth<'a, 'b, 'tcx, T>(
98 selcx: &'a mut SelectionContext<'b, 'tcx>,
99 param_env: ty::ParamEnv<'tcx>,
100 cause: ObligationCause<'tcx>,
101 depth: usize,
102 value: T,
103) -> Normalized<'tcx, T>
104where
105T: TypeFoldable<TyCtxt<'tcx>>,
106{
107let mut obligations = PredicateObligations::new();
108let value = normalize_with_depth_to(selcx, param_env, cause, depth, value, &mut obligations);
109Normalized { value, obligations }
110}
111112#[allow(clippy :: suspicious_else_formatting)]
{
let __tracing_attr_span;
let __tracing_attr_guard;
if ::tracing::Level::INFO <= ::tracing::level_filters::STATIC_MAX_LEVEL &&
::tracing::Level::INFO <=
::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("normalize_with_depth_to",
"rustc_trait_selection::traits::normalize",
::tracing::Level::INFO,
::tracing_core::__macro_support::Option::Some("compiler/rustc_trait_selection/src/traits/normalize.rs"),
::tracing_core::__macro_support::Option::Some(112u32),
::tracing_core::__macro_support::Option::Some("rustc_trait_selection::traits::normalize"),
::tracing_core::field::FieldSet::new(&["depth", "value"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::SPAN)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let mut interest = ::tracing::subscriber::Interest::never();
if ::tracing::Level::INFO <=
::tracing::level_filters::STATIC_MAX_LEVEL &&
::tracing::Level::INFO <=
::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(&depth 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(&value)
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: T = loop {};
return __tracing_attr_fake_return;
}
{
{
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_trait_selection/src/traits/normalize.rs:124",
"rustc_trait_selection::traits::normalize",
::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_trait_selection/src/traits/normalize.rs"),
::tracing_core::__macro_support::Option::Some(124u32),
::tracing_core::__macro_support::Option::Some("rustc_trait_selection::traits::normalize"),
::tracing_core::field::FieldSet::new(&["obligations.len"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::DEBUG <=
::tracing::level_filters::STATIC_MAX_LEVEL &&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() &&
{
let interest = __CALLSITE.interest();
!interest.is_never() &&
::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
interest)
};
if enabled {
(|value_set: ::tracing::field::ValueSet|
{
let meta = __CALLSITE.metadata();
::tracing::Event::dispatch(meta, &value_set);
;
})({
#[allow(unused_imports)]
use ::tracing::field::{debug, display, Value};
let mut iter = __CALLSITE.metadata().fields().iter();
__CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&obligations.len()
as &dyn Value))])
});
} else { ; }
};
let mut normalizer =
AssocTypeNormalizer::new(selcx, param_env, cause, depth,
obligations);
let result =
ensure_sufficient_stack(||
AssocTypeNormalizer::fold(&mut normalizer, value));
{
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_trait_selection/src/traits/normalize.rs:127",
"rustc_trait_selection::traits::normalize",
::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_trait_selection/src/traits/normalize.rs"),
::tracing_core::__macro_support::Option::Some(127u32),
::tracing_core::__macro_support::Option::Some("rustc_trait_selection::traits::normalize"),
::tracing_core::field::FieldSet::new(&["result",
"obligations.len"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::DEBUG <=
::tracing::level_filters::STATIC_MAX_LEVEL &&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() &&
{
let interest = __CALLSITE.interest();
!interest.is_never() &&
::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
interest)
};
if enabled {
(|value_set: ::tracing::field::ValueSet|
{
let meta = __CALLSITE.metadata();
::tracing::Event::dispatch(meta, &value_set);
;
})({
#[allow(unused_imports)]
use ::tracing::field::{debug, display, Value};
let mut iter = __CALLSITE.metadata().fields().iter();
__CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&debug(&result) as
&dyn Value)),
(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&normalizer.obligations.len()
as &dyn Value))])
});
} else { ; }
};
{
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_trait_selection/src/traits/normalize.rs:128",
"rustc_trait_selection::traits::normalize",
::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_trait_selection/src/traits/normalize.rs"),
::tracing_core::__macro_support::Option::Some(128u32),
::tracing_core::__macro_support::Option::Some("rustc_trait_selection::traits::normalize"),
::tracing_core::field::FieldSet::new(&["normalizer.obligations"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::DEBUG <=
::tracing::level_filters::STATIC_MAX_LEVEL &&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() &&
{
let interest = __CALLSITE.interest();
!interest.is_never() &&
::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
interest)
};
if enabled {
(|value_set: ::tracing::field::ValueSet|
{
let meta = __CALLSITE.metadata();
::tracing::Event::dispatch(meta, &value_set);
;
})({
#[allow(unused_imports)]
use ::tracing::field::{debug, display, Value};
let mut iter = __CALLSITE.metadata().fields().iter();
__CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&debug(&normalizer.obligations)
as &dyn Value))])
});
} else { ; }
};
result
}
}
}#[instrument(level = "info", skip(selcx, param_env, cause, obligations))]113pub(crate) fn normalize_with_depth_to<'a, 'b, 'tcx, T>(
114 selcx: &'a mut SelectionContext<'b, 'tcx>,
115 param_env: ty::ParamEnv<'tcx>,
116 cause: ObligationCause<'tcx>,
117 depth: usize,
118 value: T,
119 obligations: &mut PredicateObligations<'tcx>,
120) -> T
121where
122T: TypeFoldable<TyCtxt<'tcx>>,
123{
124debug!(obligations.len = obligations.len());
125let mut normalizer = AssocTypeNormalizer::new(selcx, param_env, cause, depth, obligations);
126let result = ensure_sufficient_stack(|| AssocTypeNormalizer::fold(&mut normalizer, value));
127debug!(?result, obligations.len = normalizer.obligations.len());
128debug!(?normalizer.obligations,);
129 result
130}
131132pub(super) fn needs_normalization<'tcx, T: TypeVisitable<TyCtxt<'tcx>>>(
133 infcx: &InferCtxt<'tcx>,
134 value: &T,
135) -> bool {
136let mut flags = ty::TypeFlags::HAS_ALIAS;
137138// Opaques are treated as rigid outside of `TypingMode::PostAnalysis`,
139 // so we can ignore those.
140match infcx.typing_mode_raw().assert_not_erased() {
141// FIXME(#132279): We likely want to reveal opaques during post borrowck analysis
142TypingMode::Coherence143 | TypingMode::Typeck { .. }
144 | TypingMode::PostTypeckUntilBorrowck { .. }
145 | TypingMode::PostBorrowck { .. } => flags.remove(ty::TypeFlags::HAS_TY_OPAQUE),
146TypingMode::PostAnalysis | TypingMode::Codegen => {}
147 }
148149value.has_type_flags(flags)
150}
151152struct AssocTypeNormalizer<'a, 'b, 'tcx> {
153 selcx: &'a mut SelectionContext<'b, 'tcx>,
154 param_env: ty::ParamEnv<'tcx>,
155 cause: ObligationCause<'tcx>,
156 obligations: &'a mut PredicateObligations<'tcx>,
157 depth: usize,
158 universes: Vec<Option<ty::UniverseIndex>>,
159}
160161impl<'a, 'b, 'tcx> AssocTypeNormalizer<'a, 'b, 'tcx> {
162fn new(
163 selcx: &'a mut SelectionContext<'b, 'tcx>,
164 param_env: ty::ParamEnv<'tcx>,
165 cause: ObligationCause<'tcx>,
166 depth: usize,
167 obligations: &'a mut PredicateObligations<'tcx>,
168 ) -> AssocTypeNormalizer<'a, 'b, 'tcx> {
169if true {
if !!selcx.infcx.next_trait_solver() {
::core::panicking::panic("assertion failed: !selcx.infcx.next_trait_solver()")
};
};debug_assert!(!selcx.infcx.next_trait_solver());
170AssocTypeNormalizer { selcx, param_env, cause, obligations, depth, universes: ::alloc::vec::Vec::new()vec![] }
171 }
172173fn fold<T: TypeFoldable<TyCtxt<'tcx>>>(&mut self, value: T) -> T {
174let value = self.selcx.infcx.resolve_vars_if_possible(value);
175{
use ::tracing::__macro_support::Callsite as _;
static __CALLSITE: ::tracing::callsite::DefaultCallsite =
{
static META: ::tracing::Metadata<'static> =
{
::tracing_core::metadata::Metadata::new("event compiler/rustc_trait_selection/src/traits/normalize.rs:175",
"rustc_trait_selection::traits::normalize",
::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_trait_selection/src/traits/normalize.rs"),
::tracing_core::__macro_support::Option::Some(175u32),
::tracing_core::__macro_support::Option::Some("rustc_trait_selection::traits::normalize"),
::tracing_core::field::FieldSet::new(&["value"],
::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::DEBUG <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::DEBUG <=
::tracing::level_filters::LevelFilter::current() &&
{
let interest = __CALLSITE.interest();
!interest.is_never() &&
::tracing::__macro_support::__is_enabled(__CALLSITE.metadata(),
interest)
};
if enabled {
(|value_set: ::tracing::field::ValueSet|
{
let meta = __CALLSITE.metadata();
::tracing::Event::dispatch(meta, &value_set);
;
})({
#[allow(unused_imports)]
use ::tracing::field::{debug, display, Value};
let mut iter = __CALLSITE.metadata().fields().iter();
__CALLSITE.metadata().fields().value_set(&[(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&debug(&value) as
&dyn Value))])
});
} else { ; }
};debug!(?value);
176177if !!value.has_escaping_bound_vars() {
{
::core::panicking::panic_fmt(format_args!("Normalizing {0:?} without wrapping in a `Binder`",
value));
}
};assert!(
178 !value.has_escaping_bound_vars(),
179"Normalizing {value:?} without wrapping in a `Binder`"
180);
181182if !needs_normalization(self.selcx.infcx, &value) { value } else { value.fold_with(self) }
183 }
184185// FIXME(mgca): While this supports constants, it is only used for types by default right now
186x;#[instrument(level = "debug", skip(self), ret)]187fn normalize_trait_projection(&mut self, proj: AliasTerm<'tcx>) -> Term<'tcx> {
188if !proj.has_escaping_bound_vars() {
189// When we don't have escaping bound vars we can normalize ambig aliases
190 // to inference variables (done in `normalize_projection_ty`). This would
191 // be wrong if there were escaping bound vars as even if we instantiated
192 // the bound vars with placeholders, we wouldn't be able to map them back
193 // after normalization succeeded.
194 //
195 // Also, as an optimization: when we don't have escaping bound vars, we don't
196 // need to replace them with placeholders (see branch below).
197let proj = proj.fold_with(self);
198 project::normalize_projection_term(
199self.selcx,
200self.param_env,
201 proj,
202self.cause.clone(),
203self.depth,
204self.obligations,
205 )
206 } else {
207// If there are escaping bound vars, we temporarily replace the
208 // bound vars with placeholders. Note though, that in the case
209 // that we still can't project for whatever reason (e.g. self
210 // type isn't known enough), we *can't* register an obligation
211 // and return an inference variable (since then that obligation
212 // would have bound vars and that's a can of worms). Instead,
213 // we just give up and fall back to pretending like we never tried!
214 //
215 // Note: this isn't necessarily the final approach here; we may
216 // want to figure out how to register obligations with escaping vars
217 // or handle this some other way.
218let infcx = self.selcx.infcx;
219let (proj, mapped_regions, mapped_types, mapped_consts) =
220 BoundVarReplacer::replace_bound_vars(infcx, &mut self.universes, proj);
221let proj = proj.fold_with(self);
222let normalized_term = project::opt_normalize_projection_term(
223self.selcx,
224self.param_env,
225 proj,
226self.cause.clone(),
227self.depth,
228self.obligations,
229 )
230 .ok()
231 .flatten()
232 .unwrap_or_else(|| proj.to_term(infcx.tcx));
233234 PlaceholderReplacer::replace_placeholders(
235 infcx,
236 mapped_regions,
237 mapped_types,
238 mapped_consts,
239&self.universes,
240 normalized_term,
241 )
242 }
243 }
244245// FIXME(mgca): While this supports constants, it is only used for types by default right now
246x;#[instrument(level = "debug", skip(self), ret)]247fn normalize_inherent_projection(&mut self, inherent: AliasTerm<'tcx>) -> Term<'tcx> {
248if !inherent.has_escaping_bound_vars() {
249// When we don't have escaping bound vars we can normalize ambig aliases
250 // to inference variables (done in `normalize_projection_ty`). This would
251 // be wrong if there were escaping bound vars as even if we instantiated
252 // the bound vars with placeholders, we wouldn't be able to map them back
253 // after normalization succeeded.
254 //
255 // Also, as an optimization: when we don't have escaping bound vars, we don't
256 // need to replace them with placeholders (see branch below).
257258let inherent = inherent.fold_with(self);
259 project::normalize_inherent_projection(
260self.selcx,
261self.param_env,
262 inherent,
263self.cause.clone(),
264self.depth,
265self.obligations,
266 )
267 } else {
268let infcx = self.selcx.infcx;
269let (inherent, mapped_regions, mapped_types, mapped_consts) =
270 BoundVarReplacer::replace_bound_vars(infcx, &mut self.universes, inherent);
271let inherent = inherent.fold_with(self);
272let inherent = project::normalize_inherent_projection(
273self.selcx,
274self.param_env,
275 inherent,
276self.cause.clone(),
277self.depth,
278self.obligations,
279 );
280281 PlaceholderReplacer::replace_placeholders(
282 infcx,
283 mapped_regions,
284 mapped_types,
285 mapped_consts,
286&self.universes,
287 inherent,
288 )
289 }
290 }
291292// FIXME(mgca): While this supports constants, it is only used for types by default right now
293x;#[instrument(level = "debug", skip(self), ret)]294fn normalize_free_alias(&mut self, free: AliasTerm<'tcx>) -> Term<'tcx> {
295let recursion_limit = self.cx().recursion_limit();
296if !recursion_limit.value_within_limit(self.depth) {
297self.selcx.infcx.err_ctxt().report_overflow_error(
298 OverflowCause::DeeplyNormalize(free),
299self.cause.span,
300false,
301 |diag| {
302 diag.note(msg!("in case this is a recursive type alias, consider using a struct, enum, or union instead"));
303 },
304 );
305 }
306307let def_id = free.expect_free_def_id();
308309// We don't replace bound vars in the generic arguments of the free alias with
310 // placeholders. This doesn't cause any issues as instantiating parameters with
311 // bound variables is special-cased to rewrite the debruijn index to be higher
312 // whenever we fold through a binder.
313 //
314 // However, we do replace any escaping bound vars in the resulting goals with
315 // placeholders as the trait solver does not expect to encounter escaping bound
316 // vars in obligations.
317 //
318 // FIXME(lazy_type_alias): Check how much this actually matters for perf before
319 // stabilization. This is a bit weird and generally not how we handle binders in
320 // the compiler so ideally we'd do the same boundvar->placeholder->boundvar dance
321 // that other kinds of normalization do.
322let infcx = self.selcx.infcx;
323self.obligations.extend(
324 infcx
325 .tcx
326 .predicates_of(def_id)
327 .instantiate_own(infcx.tcx, free.args)
328 .map(|(pred, span)| (pred.skip_norm_wip(), span))
329 .map(|(mut predicate, span)| {
330if free.has_escaping_bound_vars() {
331 (predicate, ..) = BoundVarReplacer::replace_bound_vars(
332 infcx,
333&mut self.universes,
334 predicate,
335 );
336 }
337let mut cause = self.cause.clone();
338 cause.map_code(|code| ObligationCauseCode::TypeAlias(code, span, def_id));
339 Obligation::new(infcx.tcx, cause, self.param_env, predicate)
340 }),
341 );
342self.depth += 1;
343let res: ty::Term<'tcx> = if free.kind.is_type() {
344 infcx
345 .tcx
346 .type_of(def_id)
347 .instantiate(infcx.tcx, free.args)
348 .skip_norm_wip()
349 .fold_with(self)
350 .into()
351 } else {
352 infcx
353 .tcx
354 .const_of_item(def_id)
355 .instantiate(infcx.tcx, free.args)
356 .skip_norm_wip()
357 .fold_with(self)
358 .into()
359 };
360// When normalizing a free const alias, register a `ConstArgHasType`
361 // obligation to ensure the const value's type matches the declared type.
362if let Some(ct) = res.as_const() {
363let expected_ty =
364 infcx.tcx.type_of(def_id).instantiate(infcx.tcx, free.args).skip_norm_wip();
365self.obligations.push(Obligation::with_depth(
366 infcx.tcx,
367self.cause.clone(),
368self.depth,
369self.param_env,
370 ty::ClauseKind::ConstArgHasType(ct, expected_ty),
371 ));
372 }
373self.depth -= 1;
374 res
375 }
376}
377378impl<'a, 'b, 'tcx> TypeFolder<TyCtxt<'tcx>> for AssocTypeNormalizer<'a, 'b, 'tcx> {
379fn cx(&self) -> TyCtxt<'tcx> {
380self.selcx.tcx()
381 }
382383fn fold_binder<T: TypeFoldable<TyCtxt<'tcx>>>(
384&mut self,
385 t: ty::Binder<'tcx, T>,
386 ) -> ty::Binder<'tcx, T> {
387self.universes.push(None);
388let t = t.super_fold_with(self);
389self.universes.pop();
390t391 }
392393fn fold_ty(&mut self, ty: Ty<'tcx>) -> Ty<'tcx> {
394if !needs_normalization(self.selcx.infcx, &ty) {
395return ty;
396 }
397398let ty::Alias(data) = *ty.kind() else { return ty.super_fold_with(self) };
399400// We try to be a little clever here as a performance optimization in
401 // cases where there are nested projections under binders.
402 // For example:
403 // ```
404 // for<'a> fn(<T as Foo>::One<'a, Box<dyn Bar<'a, Item=<T as Foo>::Two<'a>>>>)
405 // ```
406 // We normalize the args on the projection before the projecting, but
407 // if we're naive, we'll
408 // replace bound vars on inner, project inner, replace placeholders on inner,
409 // replace bound vars on outer, project outer, replace placeholders on outer
410 //
411 // However, if we're a bit more clever, we can replace the bound vars
412 // on the entire type before normalizing nested projections, meaning we
413 // replace bound vars on outer, project inner,
414 // project outer, replace placeholders on outer
415 //
416 // This is possible because the inner `'a` will already be a placeholder
417 // when we need to normalize the inner projection
418 //
419 // On the other hand, this does add a bit of complexity, since we only
420 // replace bound vars if the current type is a `Projection` and we need
421 // to make sure we don't forget to fold the args regardless.
422423match data.kind {
424 ty::Opaque { def_id } => {
425// Only normalize `impl Trait` outside of type inference, usually in codegen.
426match self.selcx.typing_mode() {
427// FIXME(#132279): We likely want to reveal opaques during post borrowck analysis
428TypingMode::Coherence429 | TypingMode::Typeck { .. }
430 | TypingMode::PostTypeckUntilBorrowck { .. }
431 | TypingMode::PostBorrowck { .. } => ty.super_fold_with(self),
432TypingMode::PostAnalysis | TypingMode::Codegen => {
433let recursion_limit = self.cx().recursion_limit();
434if !recursion_limit.value_within_limit(self.depth) {
435self.selcx.infcx.err_ctxt().report_overflow_error(
436 OverflowCause::DeeplyNormalize(data.into()),
437self.cause.span,
438true,
439 |_| {},
440 );
441 }
442443let args = data.args.fold_with(self);
444let generic_ty = self.cx().type_of(def_id);
445let concrete_ty = generic_ty.instantiate(self.cx(), args).skip_norm_wip();
446self.depth += 1;
447let folded_ty = self.fold_ty(concrete_ty);
448self.depth -= 1;
449folded_ty450 }
451 }
452 }
453454 ty::Projection { .. } => self.normalize_trait_projection(data.into()).expect_type(),
455 ty::Inherent { .. } => self.normalize_inherent_projection(data.into()).expect_type(),
456 ty::Free { .. } => self.normalize_free_alias(data.into()).expect_type(),
457 }
458 }
459460#[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("fold_const",
"rustc_trait_selection::traits::normalize",
::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_trait_selection/src/traits/normalize.rs"),
::tracing_core::__macro_support::Option::Some(460u32),
::tracing_core::__macro_support::Option::Some("rustc_trait_selection::traits::normalize"),
::tracing_core::field::FieldSet::new(&["ct"],
::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(&ct)
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: ty::Const<'tcx> = loop {};
return __tracing_attr_fake_return;
}
{
let tcx = self.selcx.tcx();
if tcx.features().generic_const_exprs() &&
!#[allow(non_exhaustive_omitted_patterns)] match ct.kind() {
ty::ConstKind::Unevaluated(uv) if uv.kind.is_type_const(tcx)
=> true,
_ => false,
} || !needs_normalization(self.selcx.infcx, &ct) {
return ct;
}
let uv =
match ct.kind() {
ty::ConstKind::Unevaluated(uv) => uv,
_ => return ct.super_fold_with(self),
};
let ct =
match uv.kind {
ty::UnevaluatedConstKind::Projection { .. } => {
self.normalize_trait_projection(uv.into()).expect_const()
}
ty::UnevaluatedConstKind::Inherent { .. } => {
self.normalize_inherent_projection(uv.into()).expect_const()
}
ty::UnevaluatedConstKind::Free { .. } => {
self.normalize_free_alias(uv.into()).expect_const()
}
ty::UnevaluatedConstKind::Anon { .. } => {
let ct = ct.super_fold_with(self);
super::with_replaced_escaping_bound_vars(self.selcx.infcx,
&mut self.universes, ct,
|ct|
super::evaluate_const(self.selcx.infcx, ct, self.param_env))
}
};
ct.super_fold_with(self)
}
}
}#[instrument(skip(self), level = "debug")]461fn fold_const(&mut self, ct: ty::Const<'tcx>) -> ty::Const<'tcx> {
462let tcx = self.selcx.tcx();
463464if tcx.features().generic_const_exprs()
465// Normalize type_const items even with feature `generic_const_exprs`.
466&& !matches!(ct.kind(), ty::ConstKind::Unevaluated(uv) if uv.kind.is_type_const(tcx))
467 || !needs_normalization(self.selcx.infcx, &ct)
468 {
469return ct;
470 }
471472let uv = match ct.kind() {
473 ty::ConstKind::Unevaluated(uv) => uv,
474_ => return ct.super_fold_with(self),
475 };
476477// Note that the Projection/Inherent/Free cases are unreachable on stable,
478 // unless a `min_generic_const_args` feature gate error has already
479 // been emitted earlier in compilation.
480 //
481 // That's because we can only end up with an Unevaluated ty::Const for a const item
482 // if it was marked with `type const`. Using this attribute without the mgca
483 // feature gate causes a parse error.
484let ct = match uv.kind {
485 ty::UnevaluatedConstKind::Projection { .. } => {
486self.normalize_trait_projection(uv.into()).expect_const()
487 }
488 ty::UnevaluatedConstKind::Inherent { .. } => {
489self.normalize_inherent_projection(uv.into()).expect_const()
490 }
491 ty::UnevaluatedConstKind::Free { .. } => {
492self.normalize_free_alias(uv.into()).expect_const()
493 }
494 ty::UnevaluatedConstKind::Anon { .. } => {
495let ct = ct.super_fold_with(self);
496super::with_replaced_escaping_bound_vars(
497self.selcx.infcx,
498&mut self.universes,
499 ct,
500 |ct| super::evaluate_const(self.selcx.infcx, ct, self.param_env),
501 )
502 }
503 };
504505// We re-fold the normalized const as the `ty` field on `ConstKind::Value` may be
506 // unnormalized after const evaluation returns.
507ct.super_fold_with(self)
508 }
509510#[inline]
511fn fold_predicate(&mut self, p: ty::Predicate<'tcx>) -> ty::Predicate<'tcx> {
512if p.allow_normalization() && needs_normalization(self.selcx.infcx, &p) {
513p.super_fold_with(self)
514 } else {
515p516 }
517 }
518}