Skip to main content

rustc_next_trait_solver/solve/assembly/
mod.rs

1//! Code shared by trait and projection goals for candidate assembly.
2
3pub(super) mod structural_traits;
4
5use std::cell::Cell;
6use std::ops::ControlFlow;
7
8use derive_where::derive_where;
9use rustc_type_ir::inherent::*;
10use rustc_type_ir::lang_items::SolverTraitLangItem;
11use rustc_type_ir::search_graph::CandidateHeadUsages;
12use rustc_type_ir::solve::{
13    AliasBoundKind, MaybeInfo, NoSolutionOrRerunNonErased, QueryResultOrRerunNonErased,
14    RerunNonErased, RerunReason, RerunResultExt, SizedTraitKind, StalledOnCoroutines,
15};
16use rustc_type_ir::{
17    self as ty, AliasTy, Interner, MayBeErased, TypeFlags, TypeFoldable, TypeFolder,
18    TypeSuperFoldable, TypeSuperVisitable, TypeVisitable, TypeVisitableExt, TypeVisitor,
19    TypingMode, Unnormalized, Upcast, elaborate,
20};
21use tracing::{debug, instrument};
22
23use super::trait_goals::TraitGoalProvenVia;
24use super::{has_only_region_constraints, inspect};
25use crate::delegate::SolverDelegate;
26use crate::solve::inspect::ProbeKind;
27use crate::solve::{
28    BuiltinImplSource, CandidateSource, CanonicalResponse, Certainty, EvalCtxt, Goal, GoalSource,
29    MaybeCause, NoSolution, OpaqueTypesJank, ParamEnvSource, QueryResult,
30    has_no_inference_or_external_constraints,
31};
32
33/// A candidate is a possible way to prove a goal.
34///
35/// It consists of both the `source`, which describes how that goal would be proven,
36/// and the `result` when using the given `source`.
37#[automatically_derived]
impl<I: Interner> ::core::fmt::Debug for Candidate<I> where I: Interner {
    fn fmt(&self, __f: &mut ::core::fmt::Formatter<'_>)
        -> ::core::fmt::Result {
        match self {
            Candidate {
                source: ref __field_source,
                result: ref __field_result,
                head_usages: ref __field_head_usages } => {
                let mut __builder =
                    ::core::fmt::Formatter::debug_struct(__f, "Candidate");
                ::core::fmt::DebugStruct::field(&mut __builder, "source",
                    __field_source);
                ::core::fmt::DebugStruct::field(&mut __builder, "result",
                    __field_result);
                ::core::fmt::DebugStruct::field(&mut __builder, "head_usages",
                    __field_head_usages);
                ::core::fmt::DebugStruct::finish(&mut __builder)
            }
        }
    }
}#[derive_where(Debug; I: Interner)]
38pub(super) struct Candidate<I: Interner> {
39    pub(super) source: CandidateSource<I>,
40    pub(super) result: CanonicalResponse<I>,
41    pub(super) head_usages: CandidateHeadUsages,
42}
43
44/// Methods used to assemble candidates for either trait or projection goals.
45pub(super) trait GoalKind<D, I = <D as SolverDelegate>::Interner>:
46    TypeFoldable<I> + Copy + Eq + std::fmt::Display
47where
48    D: SolverDelegate<Interner = I>,
49    I: Interner,
50{
51    fn self_ty(self) -> I::Ty;
52
53    fn trait_ref(self, cx: I) -> ty::TraitRef<I>;
54
55    fn with_replaced_self_ty(self, cx: I, self_ty: I::Ty) -> Self;
56
57    fn trait_def_id(self, cx: I) -> I::TraitId;
58
59    /// Consider a clause, which consists of a "assumption" and some "requirements",
60    /// to satisfy a goal. If the requirements hold, then attempt to satisfy our
61    /// goal by equating it with the assumption.
62    fn probe_and_consider_implied_clause(
63        ecx: &mut EvalCtxt<'_, D>,
64        parent_source: CandidateSource<I>,
65        goal: Goal<I, Self>,
66        assumption: I::Clause,
67        requirements: impl IntoIterator<Item = (GoalSource, Goal<I, I::Predicate>)>,
68    ) -> Result<Candidate<I>, NoSolutionOrRerunNonErased> {
69        Self::probe_and_match_goal_against_assumption(ecx, parent_source, goal, assumption, |ecx| {
70            for (nested_source, goal) in requirements {
71                ecx.add_goal(nested_source, goal);
72            }
73            ecx.evaluate_added_goals_and_make_canonical_response(Certainty::Yes)
74        })
75    }
76
77    /// Consider a clause specifically for a `dyn Trait` self type. This requires
78    /// additionally checking all of the supertraits and object bounds to hold,
79    /// since they're not implied by the well-formedness of the object type.
80    /// `NormalizesTo` overrides this to not check the supertraits for backwards
81    /// compatibility with the old solver. cc trait-system-refactor-initiative#245.
82    fn probe_and_consider_object_bound_candidate(
83        ecx: &mut EvalCtxt<'_, D>,
84        source: CandidateSource<I>,
85        goal: Goal<I, Self>,
86        assumption: I::Clause,
87    ) -> Result<Candidate<I>, NoSolutionOrRerunNonErased> {
88        Self::probe_and_match_goal_against_assumption(ecx, source, goal, assumption, |ecx| {
89            let cx = ecx.cx();
90            let ty::Dynamic(bounds, _) = goal.predicate.self_ty().kind() else {
91                {
    ::core::panicking::panic_fmt(format_args!("expected object type in `probe_and_consider_object_bound_candidate`"));
};panic!("expected object type in `probe_and_consider_object_bound_candidate`");
92            };
93
94            let trait_ref = assumption.kind().map_bound(|clause| match clause {
95                ty::ClauseKind::Trait(pred) => pred.trait_ref,
96                ty::ClauseKind::Projection(proj) => proj.projection_term.trait_ref(cx),
97
98                ty::ClauseKind::RegionOutlives(..)
99                | ty::ClauseKind::TypeOutlives(..)
100                | ty::ClauseKind::ConstArgHasType(..)
101                | ty::ClauseKind::WellFormed(..)
102                | ty::ClauseKind::ConstEvaluatable(..)
103                | ty::ClauseKind::HostEffect(..)
104                | ty::ClauseKind::UnstableFeature(..) => {
105                    {
    ::core::panicking::panic_fmt(format_args!("internal error: entered unreachable code: {0}",
            format_args!("expected trait or projection predicate as an assumption")));
}unreachable!("expected trait or projection predicate as an assumption")
106                }
107            });
108
109            match structural_traits::predicates_for_object_candidate(
110                ecx,
111                goal.param_env,
112                trait_ref,
113                bounds,
114            ) {
115                Ok(requirements) => {
116                    ecx.add_goals(GoalSource::ImplWhereBound, requirements);
117                    ecx.evaluate_added_goals_and_make_canonical_response(Certainty::Yes)
118                }
119                Err(_) => {
120                    ecx.evaluate_added_goals_and_make_canonical_response(Certainty::AMBIGUOUS)
121                }
122            }
123        })
124    }
125
126    /// Assemble additional assumptions for an alias that are not included
127    /// in the item bounds of the alias. For now, this is limited to the
128    /// `explicit_implied_const_bounds` for an associated type.
129    fn consider_additional_alias_assumptions(
130        ecx: &mut EvalCtxt<'_, D>,
131        goal: Goal<I, Self>,
132        alias_ty: ty::AliasTy<I>,
133    ) -> Vec<Candidate<I>>;
134
135    fn probe_and_consider_param_env_candidate(
136        ecx: &mut EvalCtxt<'_, D>,
137        goal: Goal<I, Self>,
138        assumption: I::Clause,
139    ) -> Result<Result<Candidate<I>, CandidateHeadUsages>, RerunNonErased> {
140        match Self::fast_reject_assumption(ecx, goal, assumption) {
141            Ok(()) => {}
142            Err(NoSolution) => return Ok(Err(CandidateHeadUsages::default())),
143        }
144
145        // Dealing with `ParamEnv` candidates is a bit of a mess as we need to lazily
146        // check whether the candidate is global while considering normalization.
147        //
148        // We need to write into `source` inside of `match_assumption`, but need to access it
149        // in `probe` even if the candidate does not apply before we get there. We handle this
150        // by using a `Cell` here. We only ever write into it inside of `match_assumption`.
151        let source = Cell::new(CandidateSource::ParamEnv(ParamEnvSource::Global));
152        let (result, head_usages) = ecx
153            .probe(|result: &QueryResult<I>| inspect::ProbeKind::TraitCandidate {
154                source: source.get(),
155                result: *result,
156            })
157            .enter_single_candidate(|ecx| {
158                Self::match_assumption(
159                    ecx,
160                    goal,
161                    assumption,
162                    |ecx| -> Result<_, NoSolutionOrRerunNonErased> {
163                        ecx.try_evaluate_added_goals()?;
164                        let (src, certainty) =
165                            ecx.characterize_param_env_assumption(goal.param_env, assumption)?;
166                        source.set(src);
167                        ecx.evaluate_added_goals_and_make_canonical_response(certainty)
168                    },
169                )
170                .map_err(Into::into)
171            });
172
173        Ok(match result.map_err_to_rerun()? {
174            Ok(result) => Ok(Candidate { source: source.get(), result, head_usages }),
175            Err(NoSolution) => Err(head_usages),
176        })
177    }
178
179    /// Try equating an assumption predicate against a goal's predicate. If it
180    /// holds, then execute the `then` callback, which should do any additional
181    /// work, then produce a response (typically by executing
182    /// [`EvalCtxt::evaluate_added_goals_and_make_canonical_response`]).
183    fn probe_and_match_goal_against_assumption(
184        ecx: &mut EvalCtxt<'_, D>,
185        source: CandidateSource<I>,
186        goal: Goal<I, Self>,
187        assumption: I::Clause,
188        then: impl FnOnce(&mut EvalCtxt<'_, D>) -> QueryResultOrRerunNonErased<I>,
189    ) -> Result<Candidate<I>, NoSolutionOrRerunNonErased> {
190        Self::fast_reject_assumption(ecx, goal, assumption)?;
191
192        ecx.probe_trait_candidate(source)
193            .enter(|ecx| Self::match_assumption(ecx, goal, assumption, then))
194    }
195
196    /// Try to reject the assumption based off of simple heuristics, such as [`ty::ClauseKind`]
197    /// and `DefId`.
198    fn fast_reject_assumption(
199        ecx: &mut EvalCtxt<'_, D>,
200        goal: Goal<I, Self>,
201        assumption: I::Clause,
202    ) -> Result<(), NoSolution>;
203
204    /// Relate the goal and assumption.
205    fn match_assumption(
206        ecx: &mut EvalCtxt<'_, D>,
207        goal: Goal<I, Self>,
208        assumption: I::Clause,
209        then: impl FnOnce(&mut EvalCtxt<'_, D>) -> QueryResultOrRerunNonErased<I>,
210    ) -> QueryResultOrRerunNonErased<I>;
211
212    fn consider_impl_candidate(
213        ecx: &mut EvalCtxt<'_, D>,
214        goal: Goal<I, Self>,
215        impl_def_id: I::ImplId,
216        then: impl FnOnce(&mut EvalCtxt<'_, D>, Certainty) -> QueryResultOrRerunNonErased<I>,
217    ) -> Result<Candidate<I>, NoSolutionOrRerunNonErased>;
218
219    /// If the predicate contained an error, we want to avoid emitting unnecessary trait
220    /// errors but still want to emit errors for other trait goals. We have some special
221    /// handling for this case.
222    ///
223    /// Trait goals always hold while projection goals never do. This is a bit arbitrary
224    /// but prevents incorrect normalization while hiding any trait errors.
225    fn consider_error_guaranteed_candidate(
226        ecx: &mut EvalCtxt<'_, D>,
227        goal: Goal<I, Self>,
228        guar: I::ErrorGuaranteed,
229    ) -> Result<Candidate<I>, NoSolutionOrRerunNonErased>;
230
231    /// A type implements an `auto trait` if its components do as well.
232    ///
233    /// These components are given by built-in rules from
234    /// [`structural_traits::instantiate_constituent_tys_for_auto_trait`].
235    fn consider_auto_trait_candidate(
236        ecx: &mut EvalCtxt<'_, D>,
237        goal: Goal<I, Self>,
238    ) -> Result<Candidate<I>, NoSolutionOrRerunNonErased>;
239
240    /// A trait alias holds if the RHS traits and `where` clauses hold.
241    fn consider_trait_alias_candidate(
242        ecx: &mut EvalCtxt<'_, D>,
243        goal: Goal<I, Self>,
244    ) -> Result<Candidate<I>, NoSolutionOrRerunNonErased>;
245
246    /// A type is `Sized` if its tail component is `Sized` and a type is `MetaSized` if its tail
247    /// component is `MetaSized`.
248    ///
249    /// These components are given by built-in rules from
250    /// [`structural_traits::instantiate_constituent_tys_for_sizedness_trait`].
251    fn consider_builtin_sizedness_candidates(
252        ecx: &mut EvalCtxt<'_, D>,
253        goal: Goal<I, Self>,
254        sizedness: SizedTraitKind,
255    ) -> Result<Candidate<I>, NoSolutionOrRerunNonErased>;
256
257    /// A type is `Copy` or `Clone` if its components are `Copy` or `Clone`.
258    ///
259    /// These components are given by built-in rules from
260    /// [`structural_traits::instantiate_constituent_tys_for_copy_clone_trait`].
261    fn consider_builtin_copy_clone_candidate(
262        ecx: &mut EvalCtxt<'_, D>,
263        goal: Goal<I, Self>,
264    ) -> Result<Candidate<I>, NoSolutionOrRerunNonErased>;
265
266    /// A type is a `FnPtr` if it is of `FnPtr` type.
267    fn consider_builtin_fn_ptr_trait_candidate(
268        ecx: &mut EvalCtxt<'_, D>,
269        goal: Goal<I, Self>,
270    ) -> Result<Candidate<I>, NoSolutionOrRerunNonErased>;
271
272    /// A callable type (a closure, fn def, or fn ptr) is known to implement the `Fn<A>`
273    /// family of traits where `A` is given by the signature of the type.
274    fn consider_builtin_fn_trait_candidates(
275        ecx: &mut EvalCtxt<'_, D>,
276        goal: Goal<I, Self>,
277        kind: ty::ClosureKind,
278    ) -> Result<Candidate<I>, NoSolutionOrRerunNonErased>;
279
280    /// An async closure is known to implement the `AsyncFn<A>` family of traits
281    /// where `A` is given by the signature of the type.
282    fn consider_builtin_async_fn_trait_candidates(
283        ecx: &mut EvalCtxt<'_, D>,
284        goal: Goal<I, Self>,
285        kind: ty::ClosureKind,
286    ) -> Result<Candidate<I>, NoSolutionOrRerunNonErased>;
287
288    /// Compute the built-in logic of the `AsyncFnKindHelper` helper trait, which
289    /// is used internally to delay computation for async closures until after
290    /// upvar analysis is performed in HIR typeck.
291    fn consider_builtin_async_fn_kind_helper_candidate(
292        ecx: &mut EvalCtxt<'_, D>,
293        goal: Goal<I, Self>,
294    ) -> Result<Candidate<I>, NoSolutionOrRerunNonErased>;
295
296    /// `Tuple` is implemented if the `Self` type is a tuple.
297    fn consider_builtin_tuple_candidate(
298        ecx: &mut EvalCtxt<'_, D>,
299        goal: Goal<I, Self>,
300    ) -> Result<Candidate<I>, NoSolutionOrRerunNonErased>;
301
302    /// `Pointee` is always implemented.
303    ///
304    /// See the projection implementation for the `Metadata` types for all of
305    /// the built-in types. For structs, the metadata type is given by the struct
306    /// tail.
307    fn consider_builtin_pointee_candidate(
308        ecx: &mut EvalCtxt<'_, D>,
309        goal: Goal<I, Self>,
310    ) -> Result<Candidate<I>, NoSolutionOrRerunNonErased>;
311
312    /// A coroutine (that comes from an `async` desugaring) is known to implement
313    /// `Future<Output = O>`, where `O` is given by the coroutine's return type
314    /// that was computed during type-checking.
315    fn consider_builtin_future_candidate(
316        ecx: &mut EvalCtxt<'_, D>,
317        goal: Goal<I, Self>,
318    ) -> Result<Candidate<I>, NoSolutionOrRerunNonErased>;
319
320    /// A coroutine (that comes from a `gen` desugaring) is known to implement
321    /// `Iterator<Item = O>`, where `O` is given by the generator's yield type
322    /// that was computed during type-checking.
323    fn consider_builtin_iterator_candidate(
324        ecx: &mut EvalCtxt<'_, D>,
325        goal: Goal<I, Self>,
326    ) -> Result<Candidate<I>, NoSolutionOrRerunNonErased>;
327
328    /// A coroutine (that comes from a `gen` desugaring) is known to implement
329    /// `FusedIterator`
330    fn consider_builtin_fused_iterator_candidate(
331        ecx: &mut EvalCtxt<'_, D>,
332        goal: Goal<I, Self>,
333    ) -> Result<Candidate<I>, NoSolutionOrRerunNonErased>;
334
335    fn consider_builtin_async_iterator_candidate(
336        ecx: &mut EvalCtxt<'_, D>,
337        goal: Goal<I, Self>,
338    ) -> Result<Candidate<I>, NoSolutionOrRerunNonErased>;
339
340    /// A coroutine (that doesn't come from an `async` or `gen` desugaring) is known to
341    /// implement `Coroutine<R, Yield = Y, Return = O>`, given the resume, yield,
342    /// and return types of the coroutine computed during type-checking.
343    fn consider_builtin_coroutine_candidate(
344        ecx: &mut EvalCtxt<'_, D>,
345        goal: Goal<I, Self>,
346    ) -> Result<Candidate<I>, NoSolutionOrRerunNonErased>;
347
348    fn consider_builtin_discriminant_kind_candidate(
349        ecx: &mut EvalCtxt<'_, D>,
350        goal: Goal<I, Self>,
351    ) -> Result<Candidate<I>, NoSolutionOrRerunNonErased>;
352
353    fn consider_builtin_destruct_candidate(
354        ecx: &mut EvalCtxt<'_, D>,
355        goal: Goal<I, Self>,
356    ) -> Result<Candidate<I>, NoSolutionOrRerunNonErased>;
357
358    fn consider_builtin_transmute_candidate(
359        ecx: &mut EvalCtxt<'_, D>,
360        goal: Goal<I, Self>,
361    ) -> Result<Candidate<I>, NoSolutionOrRerunNonErased>;
362
363    fn consider_builtin_bikeshed_guaranteed_no_drop_candidate(
364        ecx: &mut EvalCtxt<'_, D>,
365        goal: Goal<I, Self>,
366    ) -> Result<Candidate<I>, NoSolutionOrRerunNonErased>;
367
368    /// Consider (possibly several) candidates to upcast or unsize a type to another
369    /// type, excluding the coercion of a sized type into a `dyn Trait`.
370    ///
371    /// We return the `BuiltinImplSource` for each candidate as it is needed
372    /// for unsize coercion in hir typeck and because it is difficult to
373    /// otherwise recompute this for codegen. This is a bit of a mess but the
374    /// easiest way to maintain the existing behavior for now.
375    fn consider_structural_builtin_unsize_candidates(
376        ecx: &mut EvalCtxt<'_, D>,
377        goal: Goal<I, Self>,
378    ) -> Result<Vec<Candidate<I>>, RerunNonErased>;
379
380    fn consider_builtin_field_candidate(
381        ecx: &mut EvalCtxt<'_, D>,
382        goal: Goal<I, Self>,
383    ) -> Result<Candidate<I>, NoSolutionOrRerunNonErased>;
384}
385
386/// Allows callers of `assemble_and_evaluate_candidates` to choose whether to limit
387/// candidate assembly to param-env and alias-bound candidates.
388///
389/// On top of being a micro-optimization, as it avoids doing unnecessary work when
390/// a param-env trait bound candidate shadows impls for normalization, this is also
391/// required to prevent query cycles due to RPITIT inference. See the issue at:
392/// <https://github.com/rust-lang/trait-system-refactor-initiative/issues/173>.
393pub(super) enum AssembleCandidatesFrom {
394    All,
395    /// Only assemble candidates from the environment and alias bounds, ignoring
396    /// user-written and built-in impls. We only expect `ParamEnv` and `AliasBound`
397    /// candidates to be assembled.
398    EnvAndBounds,
399}
400
401impl AssembleCandidatesFrom {
402    fn should_assemble_impl_candidates(&self) -> bool {
403        match self {
404            AssembleCandidatesFrom::All => true,
405            AssembleCandidatesFrom::EnvAndBounds => false,
406        }
407    }
408}
409
410/// This is currently used to track the [CandidateHeadUsages] of all failed `ParamEnv`
411/// candidates. This is then used to ignore their head usages in case there's another
412/// always applicable `ParamEnv` candidate. Look at how `param_env_head_usages` is
413/// used in the code for more details.
414///
415/// We could easily extend this to also ignore head usages of other ignored candidates.
416/// However, we currently don't have any tests where this matters and the complexity of
417/// doing so does not feel worth it for now.
418#[derive(#[automatically_derived]
impl ::core::fmt::Debug for FailedCandidateInfo {
    #[inline]
    fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
        ::core::fmt::Formatter::debug_struct_field1_finish(f,
            "FailedCandidateInfo", "param_env_head_usages",
            &&self.param_env_head_usages)
    }
}Debug)]
419pub(super) struct FailedCandidateInfo {
420    pub param_env_head_usages: CandidateHeadUsages,
421}
422
423impl<D, I> EvalCtxt<'_, D>
424where
425    D: SolverDelegate<Interner = I>,
426    I: Interner,
427{
428    // FIXME(#155443): This function should only ever return an error
429    // as we want to force a rerun when accessing opaques. We should change
430    // this file to revert all the newly added places which return `NoSolution`.
431    pub(super) fn assemble_and_evaluate_candidates<G: GoalKind<D>>(
432        &mut self,
433        goal: Goal<I, G>,
434        assemble_from: AssembleCandidatesFrom,
435    ) -> Result<(Vec<Candidate<I>>, FailedCandidateInfo), RerunNonErased> {
436        let mut candidates = ::alloc::vec::Vec::new()vec![];
437        let mut failed_candidate_info =
438            FailedCandidateInfo { param_env_head_usages: CandidateHeadUsages::default() };
439        let Ok(normalized_self_ty) =
440            self.structurally_normalize_ty(goal.param_env, goal.predicate.self_ty())
441        else {
442            return Ok((candidates, failed_candidate_info));
443        };
444
445        let goal: Goal<I, G> = goal
446            .with(self.cx(), goal.predicate.with_replaced_self_ty(self.cx(), normalized_self_ty));
447
448        if normalized_self_ty.is_ty_var() {
449            {
    use ::tracing::__macro_support::Callsite as _;
    static __CALLSITE: ::tracing::callsite::DefaultCallsite =
        {
            static META: ::tracing::Metadata<'static> =
                {
                    ::tracing_core::metadata::Metadata::new("event compiler/rustc_next_trait_solver/src/solve/assembly/mod.rs:449",
                        "rustc_next_trait_solver::solve::assembly",
                        ::tracing::Level::DEBUG,
                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_next_trait_solver/src/solve/assembly/mod.rs"),
                        ::tracing_core::__macro_support::Option::Some(449u32),
                        ::tracing_core::__macro_support::Option::Some("rustc_next_trait_solver::solve::assembly"),
                        ::tracing_core::field::FieldSet::new(&["message"],
                            ::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(&format_args!("self type has been normalized to infer")
                                            as &dyn Value))])
            });
    } else { ; }
};debug!("self type has been normalized to infer");
450            self.try_assemble_bounds_via_registered_opaques(goal, assemble_from, &mut candidates)?;
451            return Ok((candidates, failed_candidate_info));
452        }
453
454        // Vars that show up in the rest of the goal substs may have been constrained by
455        // normalizing the self type as well, since type variables are not uniquified.
456        let goal = self.resolve_vars_if_possible(goal);
457
458        if self.typing_mode().is_coherence()
459            && let Ok(candidate) = self.consider_coherence_unknowable_candidate(goal)
460        {
461            candidates.push(candidate);
462            return Ok((candidates, failed_candidate_info));
463        }
464
465        self.assemble_alias_bound_candidates(goal, &mut candidates)?;
466        self.assemble_param_env_candidates(goal, &mut candidates, &mut failed_candidate_info)?;
467
468        match assemble_from {
469            AssembleCandidatesFrom::All => {
470                self.assemble_builtin_impl_candidates(goal, &mut candidates)?;
471                // For performance we only assemble impls if there are no candidates
472                // which would shadow them. This is necessary to avoid hangs in rayon,
473                // see trait-system-refactor-initiative#109 for more details.
474                //
475                // We always assemble builtin impls as trivial builtin impls have a higher
476                // priority than where-clauses.
477                //
478                // We only do this if any such candidate applies without any constraints
479                // as we may want to weaken inference guidance in the future and don't want
480                // to worry about causing major performance regressions when doing so.
481                // See trait-system-refactor-initiative#226 for some ideas here.
482                let assemble_impls = match self.typing_mode() {
483                    TypingMode::Coherence => true,
484                    TypingMode::Typeck { .. }
485                    | TypingMode::PostTypeckUntilBorrowck { .. }
486                    | TypingMode::PostBorrowck { .. }
487                    | TypingMode::PostAnalysis
488                    | TypingMode::Codegen
489                    | TypingMode::ErasedNotCoherence(MayBeErased) => !candidates.iter().any(|c| {
490                        #[allow(non_exhaustive_omitted_patterns)] match c.source {
    CandidateSource::ParamEnv(ParamEnvSource::NonGlobal) |
        CandidateSource::AliasBound(_) => true,
    _ => false,
}matches!(
491                            c.source,
492                            CandidateSource::ParamEnv(ParamEnvSource::NonGlobal)
493                                | CandidateSource::AliasBound(_)
494                        ) && has_no_inference_or_external_constraints(c.result)
495                    }),
496                };
497                if assemble_impls {
498                    self.assemble_impl_candidates(goal, &mut candidates)?;
499                    self.assemble_object_bound_candidates(goal, &mut candidates);
500                }
501            }
502            AssembleCandidatesFrom::EnvAndBounds => {
503                // This is somewhat inconsistent and may make #57893 slightly easier to exploit.
504                // However, it matches the behavior of the old solver. See
505                // `tests/ui/traits/next-solver/normalization-shadowing/use_object_if_empty_env.rs`.
506                if #[allow(non_exhaustive_omitted_patterns)] match normalized_self_ty.kind() {
    ty::Dynamic(..) => true,
    _ => false,
}matches!(normalized_self_ty.kind(), ty::Dynamic(..))
507                    && !candidates.iter().any(|c| #[allow(non_exhaustive_omitted_patterns)] match c.source {
    CandidateSource::ParamEnv(_) => true,
    _ => false,
}matches!(c.source, CandidateSource::ParamEnv(_)))
508                {
509                    self.assemble_object_bound_candidates(goal, &mut candidates);
510                }
511            }
512        }
513
514        Ok((candidates, failed_candidate_info))
515    }
516
517    pub(super) fn forced_ambiguity(
518        &mut self,
519        maybe: MaybeInfo,
520    ) -> Result<Candidate<I>, NoSolutionOrRerunNonErased> {
521        // This may fail if `try_evaluate_added_goals` overflows because it
522        // fails to reach a fixpoint but ends up getting an error after
523        // running for some additional step.
524        //
525        // FIXME(@lcnr): While I believe an error here to be possible, we
526        // currently don't have any test which actually triggers it. @lqd
527        // created a minimization for an ICE in typenum, but that one no
528        // longer fails here. cc trait-system-refactor-initiative#105.
529        let source = CandidateSource::BuiltinImpl(BuiltinImplSource::Misc);
530        let certainty = Certainty::Maybe(maybe);
531        self.probe_trait_candidate(source)
532            .enter(|this| this.evaluate_added_goals_and_make_canonical_response(certainty))
533    }
534
535    #[allow(clippy :: suspicious_else_formatting)]
{
    let __tracing_attr_span;
    let __tracing_attr_guard;
    if ::tracing::Level::TRACE <= ::tracing::level_filters::STATIC_MAX_LEVEL
                &&
                ::tracing::Level::TRACE <=
                    ::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("assemble_impl_candidates",
                                    "rustc_next_trait_solver::solve::assembly",
                                    ::tracing::Level::TRACE,
                                    ::tracing_core::__macro_support::Option::Some("compiler/rustc_next_trait_solver/src/solve/assembly/mod.rs"),
                                    ::tracing_core::__macro_support::Option::Some(535u32),
                                    ::tracing_core::__macro_support::Option::Some("rustc_next_trait_solver::solve::assembly"),
                                    ::tracing_core::field::FieldSet::new(&[],
                                        ::tracing_core::callsite::Identifier(&__CALLSITE)),
                                    ::tracing::metadata::Kind::SPAN)
                            };
                        ::tracing::callsite::DefaultCallsite::new(&META)
                    };
                let mut interest = ::tracing::subscriber::Interest::never();
                if ::tracing::Level::TRACE <=
                                    ::tracing::level_filters::STATIC_MAX_LEVEL &&
                                ::tracing::Level::TRACE <=
                                    ::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,
                        &{ meta.fields().value_set(&[]) })
                } 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: Result<(), RerunNonErased> =
                loop {};
            return __tracing_attr_fake_return;
        }
        {
            let cx = self.cx();
            cx.for_each_relevant_impl(goal.predicate.trait_def_id(cx),
                goal.predicate.self_ty(),
                |impl_def_id| -> Result<_, _>
                    {
                        if cx.impl_is_default(impl_def_id) { return Ok(()); }
                        match G::consider_impl_candidate(self, goal, impl_def_id,
                                        |ecx, certainty|
                                            {
                                                ecx.evaluate_added_goals_and_make_canonical_response(certainty)
                                            }).map_err_to_rerun()? {
                            Ok(candidate) => candidates.push(candidate),
                            Err(NoSolution) => {}
                        }
                        Ok(())
                    })
        }
    }
}#[instrument(level = "trace", skip_all)]
536    fn assemble_impl_candidates<G: GoalKind<D>>(
537        &mut self,
538        goal: Goal<I, G>,
539        candidates: &mut Vec<Candidate<I>>,
540    ) -> Result<(), RerunNonErased> {
541        let cx = self.cx();
542        cx.for_each_relevant_impl(
543            goal.predicate.trait_def_id(cx),
544            goal.predicate.self_ty(),
545            |impl_def_id| -> Result<_, _> {
546                // For every `default impl`, there's always a non-default `impl`
547                // that will *also* apply. There's no reason to register a candidate
548                // for this impl, since it is *not* proof that the trait goal holds.
549                if cx.impl_is_default(impl_def_id) {
550                    return Ok(());
551                }
552                match G::consider_impl_candidate(self, goal, impl_def_id, |ecx, certainty| {
553                    ecx.evaluate_added_goals_and_make_canonical_response(certainty)
554                })
555                .map_err_to_rerun()?
556                {
557                    Ok(candidate) => candidates.push(candidate),
558                    Err(NoSolution) => {}
559                }
560
561                Ok(())
562            },
563        )
564    }
565
566    #[allow(clippy :: suspicious_else_formatting)]
{
    let __tracing_attr_span;
    let __tracing_attr_guard;
    if ::tracing::Level::TRACE <= ::tracing::level_filters::STATIC_MAX_LEVEL
                &&
                ::tracing::Level::TRACE <=
                    ::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("assemble_builtin_impl_candidates",
                                    "rustc_next_trait_solver::solve::assembly",
                                    ::tracing::Level::TRACE,
                                    ::tracing_core::__macro_support::Option::Some("compiler/rustc_next_trait_solver/src/solve/assembly/mod.rs"),
                                    ::tracing_core::__macro_support::Option::Some(566u32),
                                    ::tracing_core::__macro_support::Option::Some("rustc_next_trait_solver::solve::assembly"),
                                    ::tracing_core::field::FieldSet::new(&[],
                                        ::tracing_core::callsite::Identifier(&__CALLSITE)),
                                    ::tracing::metadata::Kind::SPAN)
                            };
                        ::tracing::callsite::DefaultCallsite::new(&META)
                    };
                let mut interest = ::tracing::subscriber::Interest::never();
                if ::tracing::Level::TRACE <=
                                    ::tracing::level_filters::STATIC_MAX_LEVEL &&
                                ::tracing::Level::TRACE <=
                                    ::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,
                        &{ meta.fields().value_set(&[]) })
                } 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: Result<(), RerunNonErased> =
                loop {};
            return __tracing_attr_fake_return;
        }
        {
            let cx = self.cx();
            let trait_def_id = goal.predicate.trait_def_id(cx);
            let result =
                if let ty::Error(guar) = goal.predicate.self_ty().kind() {
                    G::consider_error_guaranteed_candidate(self, goal, guar)
                } else if cx.trait_is_auto(trait_def_id) {
                    G::consider_auto_trait_candidate(self, goal)
                } else if cx.trait_is_alias(trait_def_id) {
                    G::consider_trait_alias_candidate(self, goal)
                } else {
                    match cx.as_trait_lang_item(trait_def_id) {
                        Some(SolverTraitLangItem::Sized) => {
                            G::consider_builtin_sizedness_candidates(self, goal,
                                SizedTraitKind::Sized)
                        }
                        Some(SolverTraitLangItem::MetaSized) => {
                            G::consider_builtin_sizedness_candidates(self, goal,
                                SizedTraitKind::MetaSized)
                        }
                        Some(SolverTraitLangItem::PointeeSized) => {
                            {
                                ::core::panicking::panic_fmt(format_args!("internal error: entered unreachable code: {0}",
                                        format_args!("`PointeeSized` is removed during lowering")));
                            };
                        }
                        Some(SolverTraitLangItem::Copy | SolverTraitLangItem::Clone
                            | SolverTraitLangItem::TrivialClone) =>
                            G::consider_builtin_copy_clone_candidate(self, goal),
                        Some(SolverTraitLangItem::Fn) => {
                            G::consider_builtin_fn_trait_candidates(self, goal,
                                ty::ClosureKind::Fn)
                        }
                        Some(SolverTraitLangItem::FnMut) => {
                            G::consider_builtin_fn_trait_candidates(self, goal,
                                ty::ClosureKind::FnMut)
                        }
                        Some(SolverTraitLangItem::FnOnce) => {
                            G::consider_builtin_fn_trait_candidates(self, goal,
                                ty::ClosureKind::FnOnce)
                        }
                        Some(SolverTraitLangItem::AsyncFn) => {
                            G::consider_builtin_async_fn_trait_candidates(self, goal,
                                ty::ClosureKind::Fn)
                        }
                        Some(SolverTraitLangItem::AsyncFnMut) => {
                            G::consider_builtin_async_fn_trait_candidates(self, goal,
                                ty::ClosureKind::FnMut)
                        }
                        Some(SolverTraitLangItem::AsyncFnOnce) => {
                            G::consider_builtin_async_fn_trait_candidates(self, goal,
                                ty::ClosureKind::FnOnce)
                        }
                        Some(SolverTraitLangItem::FnPtrTrait) => {
                            G::consider_builtin_fn_ptr_trait_candidate(self, goal)
                        }
                        Some(SolverTraitLangItem::AsyncFnKindHelper) => {
                            G::consider_builtin_async_fn_kind_helper_candidate(self,
                                goal)
                        }
                        Some(SolverTraitLangItem::Tuple) =>
                            G::consider_builtin_tuple_candidate(self, goal),
                        Some(SolverTraitLangItem::PointeeTrait) => {
                            G::consider_builtin_pointee_candidate(self, goal)
                        }
                        Some(SolverTraitLangItem::Future) => {
                            G::consider_builtin_future_candidate(self, goal)
                        }
                        Some(SolverTraitLangItem::Iterator) => {
                            G::consider_builtin_iterator_candidate(self, goal)
                        }
                        Some(SolverTraitLangItem::FusedIterator) => {
                            G::consider_builtin_fused_iterator_candidate(self, goal)
                        }
                        Some(SolverTraitLangItem::AsyncIterator) => {
                            G::consider_builtin_async_iterator_candidate(self, goal)
                        }
                        Some(SolverTraitLangItem::Coroutine) => {
                            G::consider_builtin_coroutine_candidate(self, goal)
                        }
                        Some(SolverTraitLangItem::DiscriminantKind) => {
                            G::consider_builtin_discriminant_kind_candidate(self, goal)
                        }
                        Some(SolverTraitLangItem::Destruct) => {
                            G::consider_builtin_destruct_candidate(self, goal)
                        }
                        Some(SolverTraitLangItem::TransmuteTrait) => {
                            G::consider_builtin_transmute_candidate(self, goal)
                        }
                        Some(SolverTraitLangItem::BikeshedGuaranteedNoDrop) => {
                            G::consider_builtin_bikeshed_guaranteed_no_drop_candidate(self,
                                goal)
                        }
                        Some(SolverTraitLangItem::Field) =>
                            G::consider_builtin_field_candidate(self, goal),
                        _ => Err(NoSolution.into()),
                    }
                };
            candidates.extend(result);
            if cx.is_trait_lang_item(trait_def_id,
                    SolverTraitLangItem::Unsize) {
                candidates.extend(G::consider_structural_builtin_unsize_candidates(self,
                            goal)?);
            }
            Ok(())
        }
    }
}#[instrument(level = "trace", skip_all)]
567    fn assemble_builtin_impl_candidates<G: GoalKind<D>>(
568        &mut self,
569        goal: Goal<I, G>,
570        candidates: &mut Vec<Candidate<I>>,
571    ) -> Result<(), RerunNonErased> {
572        let cx = self.cx();
573        let trait_def_id = goal.predicate.trait_def_id(cx);
574
575        // N.B. When assembling built-in candidates for lang items that are also
576        // `auto` traits, then the auto trait candidate that is assembled in
577        // `consider_auto_trait_candidate` MUST be disqualified to remain sound.
578        //
579        // Instead of adding the logic here, it's a better idea to add it in
580        // `EvalCtxt::disqualify_auto_trait_candidate_due_to_possible_impl` in
581        // `solve::trait_goals` instead.
582        let result = if let ty::Error(guar) = goal.predicate.self_ty().kind() {
583            G::consider_error_guaranteed_candidate(self, goal, guar)
584        } else if cx.trait_is_auto(trait_def_id) {
585            G::consider_auto_trait_candidate(self, goal)
586        } else if cx.trait_is_alias(trait_def_id) {
587            G::consider_trait_alias_candidate(self, goal)
588        } else {
589            match cx.as_trait_lang_item(trait_def_id) {
590                Some(SolverTraitLangItem::Sized) => {
591                    G::consider_builtin_sizedness_candidates(self, goal, SizedTraitKind::Sized)
592                }
593                Some(SolverTraitLangItem::MetaSized) => {
594                    G::consider_builtin_sizedness_candidates(self, goal, SizedTraitKind::MetaSized)
595                }
596                Some(SolverTraitLangItem::PointeeSized) => {
597                    unreachable!("`PointeeSized` is removed during lowering");
598                }
599                Some(
600                    SolverTraitLangItem::Copy
601                    | SolverTraitLangItem::Clone
602                    | SolverTraitLangItem::TrivialClone,
603                ) => G::consider_builtin_copy_clone_candidate(self, goal),
604                Some(SolverTraitLangItem::Fn) => {
605                    G::consider_builtin_fn_trait_candidates(self, goal, ty::ClosureKind::Fn)
606                }
607                Some(SolverTraitLangItem::FnMut) => {
608                    G::consider_builtin_fn_trait_candidates(self, goal, ty::ClosureKind::FnMut)
609                }
610                Some(SolverTraitLangItem::FnOnce) => {
611                    G::consider_builtin_fn_trait_candidates(self, goal, ty::ClosureKind::FnOnce)
612                }
613                Some(SolverTraitLangItem::AsyncFn) => {
614                    G::consider_builtin_async_fn_trait_candidates(self, goal, ty::ClosureKind::Fn)
615                }
616                Some(SolverTraitLangItem::AsyncFnMut) => {
617                    G::consider_builtin_async_fn_trait_candidates(
618                        self,
619                        goal,
620                        ty::ClosureKind::FnMut,
621                    )
622                }
623                Some(SolverTraitLangItem::AsyncFnOnce) => {
624                    G::consider_builtin_async_fn_trait_candidates(
625                        self,
626                        goal,
627                        ty::ClosureKind::FnOnce,
628                    )
629                }
630                Some(SolverTraitLangItem::FnPtrTrait) => {
631                    G::consider_builtin_fn_ptr_trait_candidate(self, goal)
632                }
633                Some(SolverTraitLangItem::AsyncFnKindHelper) => {
634                    G::consider_builtin_async_fn_kind_helper_candidate(self, goal)
635                }
636                Some(SolverTraitLangItem::Tuple) => G::consider_builtin_tuple_candidate(self, goal),
637                Some(SolverTraitLangItem::PointeeTrait) => {
638                    G::consider_builtin_pointee_candidate(self, goal)
639                }
640                Some(SolverTraitLangItem::Future) => {
641                    G::consider_builtin_future_candidate(self, goal)
642                }
643                Some(SolverTraitLangItem::Iterator) => {
644                    G::consider_builtin_iterator_candidate(self, goal)
645                }
646                Some(SolverTraitLangItem::FusedIterator) => {
647                    G::consider_builtin_fused_iterator_candidate(self, goal)
648                }
649                Some(SolverTraitLangItem::AsyncIterator) => {
650                    G::consider_builtin_async_iterator_candidate(self, goal)
651                }
652                Some(SolverTraitLangItem::Coroutine) => {
653                    G::consider_builtin_coroutine_candidate(self, goal)
654                }
655                Some(SolverTraitLangItem::DiscriminantKind) => {
656                    G::consider_builtin_discriminant_kind_candidate(self, goal)
657                }
658                Some(SolverTraitLangItem::Destruct) => {
659                    G::consider_builtin_destruct_candidate(self, goal)
660                }
661                Some(SolverTraitLangItem::TransmuteTrait) => {
662                    G::consider_builtin_transmute_candidate(self, goal)
663                }
664                Some(SolverTraitLangItem::BikeshedGuaranteedNoDrop) => {
665                    G::consider_builtin_bikeshed_guaranteed_no_drop_candidate(self, goal)
666                }
667                Some(SolverTraitLangItem::Field) => G::consider_builtin_field_candidate(self, goal),
668                _ => Err(NoSolution.into()),
669            }
670        };
671
672        candidates.extend(result);
673
674        // There may be multiple unsize candidates for a trait with several supertraits:
675        // `trait Foo: Bar<A> + Bar<B>` and `dyn Foo: Unsize<dyn Bar<_>>`
676        if cx.is_trait_lang_item(trait_def_id, SolverTraitLangItem::Unsize) {
677            candidates.extend(G::consider_structural_builtin_unsize_candidates(self, goal)?);
678        }
679
680        Ok(())
681    }
682
683    #[allow(clippy :: suspicious_else_formatting)]
{
    let __tracing_attr_span;
    let __tracing_attr_guard;
    if ::tracing::Level::TRACE <= ::tracing::level_filters::STATIC_MAX_LEVEL
                &&
                ::tracing::Level::TRACE <=
                    ::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("assemble_param_env_candidates",
                                    "rustc_next_trait_solver::solve::assembly",
                                    ::tracing::Level::TRACE,
                                    ::tracing_core::__macro_support::Option::Some("compiler/rustc_next_trait_solver/src/solve/assembly/mod.rs"),
                                    ::tracing_core::__macro_support::Option::Some(683u32),
                                    ::tracing_core::__macro_support::Option::Some("rustc_next_trait_solver::solve::assembly"),
                                    ::tracing_core::field::FieldSet::new(&[],
                                        ::tracing_core::callsite::Identifier(&__CALLSITE)),
                                    ::tracing::metadata::Kind::SPAN)
                            };
                        ::tracing::callsite::DefaultCallsite::new(&META)
                    };
                let mut interest = ::tracing::subscriber::Interest::never();
                if ::tracing::Level::TRACE <=
                                    ::tracing::level_filters::STATIC_MAX_LEVEL &&
                                ::tracing::Level::TRACE <=
                                    ::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,
                        &{ meta.fields().value_set(&[]) })
                } 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: Result<(), RerunNonErased> =
                loop {};
            return __tracing_attr_fake_return;
        }
        {
            for assumption in goal.param_env.caller_bounds().iter() {
                match G::probe_and_consider_param_env_candidate(self, goal,
                            assumption)? {
                    Ok(candidate) => candidates.push(candidate),
                    Err(head_usages) => {
                        failed_candidate_info.param_env_head_usages.merge_usages(head_usages)
                    }
                }
            }
            Ok(())
        }
    }
}#[instrument(level = "trace", skip_all)]
684    fn assemble_param_env_candidates<G: GoalKind<D>>(
685        &mut self,
686        goal: Goal<I, G>,
687        candidates: &mut Vec<Candidate<I>>,
688        failed_candidate_info: &mut FailedCandidateInfo,
689    ) -> Result<(), RerunNonErased> {
690        for assumption in goal.param_env.caller_bounds().iter() {
691            match G::probe_and_consider_param_env_candidate(self, goal, assumption)? {
692                Ok(candidate) => candidates.push(candidate),
693                Err(head_usages) => {
694                    failed_candidate_info.param_env_head_usages.merge_usages(head_usages)
695                }
696            }
697        }
698
699        Ok(())
700    }
701
702    #[allow(clippy :: suspicious_else_formatting)]
{
    let __tracing_attr_span;
    let __tracing_attr_guard;
    if ::tracing::Level::TRACE <= ::tracing::level_filters::STATIC_MAX_LEVEL
                &&
                ::tracing::Level::TRACE <=
                    ::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("assemble_alias_bound_candidates",
                                    "rustc_next_trait_solver::solve::assembly",
                                    ::tracing::Level::TRACE,
                                    ::tracing_core::__macro_support::Option::Some("compiler/rustc_next_trait_solver/src/solve/assembly/mod.rs"),
                                    ::tracing_core::__macro_support::Option::Some(702u32),
                                    ::tracing_core::__macro_support::Option::Some("rustc_next_trait_solver::solve::assembly"),
                                    ::tracing_core::field::FieldSet::new(&[],
                                        ::tracing_core::callsite::Identifier(&__CALLSITE)),
                                    ::tracing::metadata::Kind::SPAN)
                            };
                        ::tracing::callsite::DefaultCallsite::new(&META)
                    };
                let mut interest = ::tracing::subscriber::Interest::never();
                if ::tracing::Level::TRACE <=
                                    ::tracing::level_filters::STATIC_MAX_LEVEL &&
                                ::tracing::Level::TRACE <=
                                    ::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,
                        &{ meta.fields().value_set(&[]) })
                } 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: Result<(), RerunNonErased> =
                loop {};
            return __tracing_attr_fake_return;
        }
        {
            let res =
                self.probe(|_|
                            ProbeKind::NormalizedSelfTyAssembly).enter(|ecx|
                        {
                            ecx.assemble_alias_bound_candidates_recur(goal.predicate.self_ty(),
                                    goal, candidates, AliasBoundKind::SelfBounds)?;
                            Ok(())
                        });
            match res {
                Ok(_) => Ok(()),
                Err(NoSolutionOrRerunNonErased::RerunNonErased(e)) => Err(e),
                Err(NoSolutionOrRerunNonErased::NoSolution(NoSolution)) => {
                    ::core::panicking::panic("internal error: entered unreachable code")
                }
            }
        }
    }
}#[instrument(level = "trace", skip_all)]
703    fn assemble_alias_bound_candidates<G: GoalKind<D>>(
704        &mut self,
705        goal: Goal<I, G>,
706        candidates: &mut Vec<Candidate<I>>,
707    ) -> Result<(), RerunNonErased> {
708        let res = self.probe(|_| ProbeKind::NormalizedSelfTyAssembly).enter(|ecx| {
709            ecx.assemble_alias_bound_candidates_recur(
710                goal.predicate.self_ty(),
711                goal,
712                candidates,
713                AliasBoundKind::SelfBounds,
714            )?;
715            Ok(())
716        });
717
718        // always returns Ok
719        match res {
720            Ok(_) => Ok(()),
721            Err(NoSolutionOrRerunNonErased::RerunNonErased(e)) => Err(e),
722            Err(NoSolutionOrRerunNonErased::NoSolution(NoSolution)) => {
723                unreachable!()
724            }
725        }
726    }
727
728    /// For some deeply nested `<T>::A::B::C::D` rigid associated type,
729    /// we should explore the item bounds for all levels, since the
730    /// `associated_type_bounds` feature means that a parent associated
731    /// type may carry bounds for a nested associated type.
732    ///
733    /// If we have a projection, check that its self type is a rigid projection.
734    /// If so, continue searching by recursively calling after normalization.
735    // FIXME: This may recurse infinitely, but I can't seem to trigger it without
736    // hitting another overflow error something. Add a depth parameter needed later.
737    fn assemble_alias_bound_candidates_recur<G: GoalKind<D>>(
738        &mut self,
739        self_ty: I::Ty,
740        goal: Goal<I, G>,
741        candidates: &mut Vec<Candidate<I>>,
742        consider_self_bounds: AliasBoundKind,
743    ) -> Result<(), RerunNonErased> {
744        let (alias_ty, def_id) = match self_ty.kind() {
745            ty::Bool
746            | ty::Char
747            | ty::Int(_)
748            | ty::Uint(_)
749            | ty::Float(_)
750            | ty::Adt(_, _)
751            | ty::Foreign(_)
752            | ty::Str
753            | ty::Array(_, _)
754            | ty::Pat(_, _)
755            | ty::Slice(_)
756            | ty::RawPtr(_, _)
757            | ty::Ref(_, _, _)
758            | ty::FnDef(_, _)
759            | ty::FnPtr(..)
760            | ty::UnsafeBinder(_)
761            | ty::Dynamic(..)
762            | ty::Closure(..)
763            | ty::CoroutineClosure(..)
764            | ty::Coroutine(..)
765            | ty::CoroutineWitness(..)
766            | ty::Never
767            | ty::Tuple(_)
768            | ty::Param(_)
769            | ty::Placeholder(..)
770            | ty::Infer(ty::IntVar(_) | ty::FloatVar(_))
771            | ty::Error(_) => return Ok(()),
772            ty::Infer(ty::FreshTy(_) | ty::FreshIntTy(_) | ty::FreshFloatTy(_)) | ty::Bound(..) => {
773                {
    ::core::panicking::panic_fmt(format_args!("unexpected self type for `{0:?}`",
            goal));
}panic!("unexpected self type for `{goal:?}`")
774            }
775
776            ty::Infer(ty::TyVar(_)) => {
777                // If we hit infer when normalizing the self type of an alias,
778                // then bail with ambiguity. We should never encounter this on
779                // the *first* iteration of this recursive function.
780                if let Ok(result) =
781                    self.evaluate_added_goals_and_make_canonical_response(Certainty::AMBIGUOUS)
782                {
783                    candidates.push(Candidate {
784                        source: CandidateSource::AliasBound(consider_self_bounds),
785                        result,
786                        head_usages: CandidateHeadUsages::default(),
787                    });
788                }
789                return Ok(());
790            }
791
792            ty::Alias(alias_ty @ AliasTy { kind: ty::Projection { def_id }, .. }) => {
793                (alias_ty, def_id.into())
794            }
795            ty::Alias(alias_ty @ AliasTy { kind: ty::Opaque { def_id }, .. }) => {
796                (alias_ty, def_id.into())
797            }
798            ty::Alias(AliasTy { kind: ty::Inherent { .. } | ty::Free { .. }, .. }) => {
799                self.cx().delay_bug(::alloc::__export::must_use({
        ::alloc::fmt::format(format_args!("could not normalize {0:?}, it is not WF",
                self_ty))
    })format!("could not normalize {self_ty:?}, it is not WF"));
800                return Ok(());
801            }
802        };
803
804        match consider_self_bounds {
805            AliasBoundKind::SelfBounds => {
806                for assumption in self
807                    .cx()
808                    .item_self_bounds(def_id)
809                    .iter_instantiated(self.cx(), alias_ty.args)
810                    .map(Unnormalized::skip_norm_wip)
811                {
812                    candidates.extend(G::probe_and_consider_implied_clause(
813                        self,
814                        CandidateSource::AliasBound(consider_self_bounds),
815                        goal,
816                        assumption,
817                        [],
818                    ));
819                }
820            }
821            AliasBoundKind::NonSelfBounds => {
822                for assumption in self
823                    .cx()
824                    .item_non_self_bounds(def_id)
825                    .iter_instantiated(self.cx(), alias_ty.args)
826                    .map(Unnormalized::skip_norm_wip)
827                {
828                    candidates.extend(G::probe_and_consider_implied_clause(
829                        self,
830                        CandidateSource::AliasBound(consider_self_bounds),
831                        goal,
832                        assumption,
833                        [],
834                    ));
835                }
836            }
837        }
838
839        candidates.extend(G::consider_additional_alias_assumptions(self, goal, alias_ty));
840
841        let Some(projection_ty) = alias_ty.try_to_projection() else {
842            return Ok(());
843        };
844
845        // Recurse on the self type of the projection.
846        match self.structurally_normalize_ty(goal.param_env, projection_ty.projection_self_ty()) {
847            Ok(next_self_ty) => self.assemble_alias_bound_candidates_recur(
848                next_self_ty,
849                goal,
850                candidates,
851                AliasBoundKind::NonSelfBounds,
852            ),
853            Err(NoSolutionOrRerunNonErased::NoSolution(NoSolution)) => Ok(()),
854            Err(NoSolutionOrRerunNonErased::RerunNonErased(e)) => Err(e),
855        }
856    }
857
858    #[allow(clippy :: suspicious_else_formatting)]
{
    let __tracing_attr_span;
    let __tracing_attr_guard;
    if ::tracing::Level::TRACE <= ::tracing::level_filters::STATIC_MAX_LEVEL
                &&
                ::tracing::Level::TRACE <=
                    ::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("assemble_object_bound_candidates",
                                    "rustc_next_trait_solver::solve::assembly",
                                    ::tracing::Level::TRACE,
                                    ::tracing_core::__macro_support::Option::Some("compiler/rustc_next_trait_solver/src/solve/assembly/mod.rs"),
                                    ::tracing_core::__macro_support::Option::Some(858u32),
                                    ::tracing_core::__macro_support::Option::Some("rustc_next_trait_solver::solve::assembly"),
                                    ::tracing_core::field::FieldSet::new(&[],
                                        ::tracing_core::callsite::Identifier(&__CALLSITE)),
                                    ::tracing::metadata::Kind::SPAN)
                            };
                        ::tracing::callsite::DefaultCallsite::new(&META)
                    };
                let mut interest = ::tracing::subscriber::Interest::never();
                if ::tracing::Level::TRACE <=
                                    ::tracing::level_filters::STATIC_MAX_LEVEL &&
                                ::tracing::Level::TRACE <=
                                    ::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,
                        &{ meta.fields().value_set(&[]) })
                } 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: () = loop {};
            return __tracing_attr_fake_return;
        }
        {
            let cx = self.cx();
            if cx.is_sizedness_trait(goal.predicate.trait_def_id(cx)) {
                return;
            }
            let self_ty = goal.predicate.self_ty();
            let bounds =
                match self_ty.kind() {
                    ty::Bool | ty::Char | ty::Int(_) | ty::Uint(_) |
                        ty::Float(_) | ty::Adt(_, _) | ty::Foreign(_) | ty::Str |
                        ty::Array(_, _) | ty::Pat(_, _) | ty::Slice(_) |
                        ty::RawPtr(_, _) | ty::Ref(_, _, _) | ty::FnDef(_, _) |
                        ty::FnPtr(..) | ty::UnsafeBinder(_) | ty::Alias(..) |
                        ty::Closure(..) | ty::CoroutineClosure(..) |
                        ty::Coroutine(..) | ty::CoroutineWitness(..) | ty::Never |
                        ty::Tuple(_) | ty::Param(_) | ty::Placeholder(..) |
                        ty::Infer(ty::IntVar(_) | ty::FloatVar(_)) | ty::Error(_) =>
                        return,
                    ty::Infer(ty::TyVar(_) | ty::FreshTy(_) | ty::FreshIntTy(_)
                        | ty::FreshFloatTy(_)) | ty::Bound(..) => {
                        ::core::panicking::panic_fmt(format_args!("unexpected self type for `{0:?}`",
                                goal));
                    }
                    ty::Dynamic(bounds, ..) => bounds,
                };
            if bounds.principal_def_id().is_some_and(|def_id|
                        !cx.trait_is_dyn_compatible(def_id)) {
                return;
            }
            for bound in bounds.iter() {
                match bound.skip_binder() {
                    ty::ExistentialPredicate::Trait(_) => {}
                    ty::ExistentialPredicate::Projection(_) |
                        ty::ExistentialPredicate::AutoTrait(_) => {
                        candidates.extend(G::probe_and_consider_object_bound_candidate(self,
                                CandidateSource::BuiltinImpl(BuiltinImplSource::Misc), goal,
                                bound.with_self_ty(cx, self_ty)));
                    }
                }
            }
            if let Some(principal) = bounds.principal() {
                let principal_trait_ref = principal.with_self_ty(cx, self_ty);
                for (idx, assumption) in
                    elaborate::supertraits(cx, principal_trait_ref).enumerate()
                    {
                    candidates.extend(G::probe_and_consider_object_bound_candidate(self,
                            CandidateSource::BuiltinImpl(BuiltinImplSource::Object(idx)),
                            goal, assumption.upcast(cx)));
                }
            }
        }
    }
}#[instrument(level = "trace", skip_all)]
859    fn assemble_object_bound_candidates<G: GoalKind<D>>(
860        &mut self,
861        goal: Goal<I, G>,
862        candidates: &mut Vec<Candidate<I>>,
863    ) {
864        let cx = self.cx();
865        if cx.is_sizedness_trait(goal.predicate.trait_def_id(cx)) {
866            // `dyn MetaSized` is valid, but should get its `MetaSized` impl from
867            // being `dyn` (SizedCandidate), not from the object candidate.
868            return;
869        }
870
871        let self_ty = goal.predicate.self_ty();
872        let bounds = match self_ty.kind() {
873            ty::Bool
874            | ty::Char
875            | ty::Int(_)
876            | ty::Uint(_)
877            | ty::Float(_)
878            | ty::Adt(_, _)
879            | ty::Foreign(_)
880            | ty::Str
881            | ty::Array(_, _)
882            | ty::Pat(_, _)
883            | ty::Slice(_)
884            | ty::RawPtr(_, _)
885            | ty::Ref(_, _, _)
886            | ty::FnDef(_, _)
887            | ty::FnPtr(..)
888            | ty::UnsafeBinder(_)
889            | ty::Alias(..)
890            | ty::Closure(..)
891            | ty::CoroutineClosure(..)
892            | ty::Coroutine(..)
893            | ty::CoroutineWitness(..)
894            | ty::Never
895            | ty::Tuple(_)
896            | ty::Param(_)
897            | ty::Placeholder(..)
898            | ty::Infer(ty::IntVar(_) | ty::FloatVar(_))
899            | ty::Error(_) => return,
900            ty::Infer(ty::TyVar(_) | ty::FreshTy(_) | ty::FreshIntTy(_) | ty::FreshFloatTy(_))
901            | ty::Bound(..) => panic!("unexpected self type for `{goal:?}`"),
902            ty::Dynamic(bounds, ..) => bounds,
903        };
904
905        // Do not consider built-in object impls for dyn-incompatible types.
906        if bounds.principal_def_id().is_some_and(|def_id| !cx.trait_is_dyn_compatible(def_id)) {
907            return;
908        }
909
910        // Consider all of the auto-trait and projection bounds, which don't
911        // need to be recorded as a `BuiltinImplSource::Object` since they don't
912        // really have a vtable base...
913        for bound in bounds.iter() {
914            match bound.skip_binder() {
915                ty::ExistentialPredicate::Trait(_) => {
916                    // Skip principal
917                }
918                ty::ExistentialPredicate::Projection(_)
919                | ty::ExistentialPredicate::AutoTrait(_) => {
920                    candidates.extend(G::probe_and_consider_object_bound_candidate(
921                        self,
922                        CandidateSource::BuiltinImpl(BuiltinImplSource::Misc),
923                        goal,
924                        bound.with_self_ty(cx, self_ty),
925                    ));
926                }
927            }
928        }
929
930        // FIXME: We only need to do *any* of this if we're considering a trait goal,
931        // since we don't need to look at any supertrait or anything if we are doing
932        // a projection goal.
933        if let Some(principal) = bounds.principal() {
934            let principal_trait_ref = principal.with_self_ty(cx, self_ty);
935            for (idx, assumption) in elaborate::supertraits(cx, principal_trait_ref).enumerate() {
936                candidates.extend(G::probe_and_consider_object_bound_candidate(
937                    self,
938                    CandidateSource::BuiltinImpl(BuiltinImplSource::Object(idx)),
939                    goal,
940                    assumption.upcast(cx),
941                ));
942            }
943        }
944    }
945
946    /// In coherence we have to not only care about all impls we know about, but
947    /// also consider impls which may get added in a downstream or sibling crate
948    /// or which an upstream impl may add in a minor release.
949    ///
950    /// To do so we return a single ambiguous candidate in case such an unknown
951    /// impl could apply to the current goal.
952    #[allow(clippy :: suspicious_else_formatting)]
{
    let __tracing_attr_span;
    let __tracing_attr_guard;
    if ::tracing::Level::TRACE <= ::tracing::level_filters::STATIC_MAX_LEVEL
                &&
                ::tracing::Level::TRACE <=
                    ::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("consider_coherence_unknowable_candidate",
                                    "rustc_next_trait_solver::solve::assembly",
                                    ::tracing::Level::TRACE,
                                    ::tracing_core::__macro_support::Option::Some("compiler/rustc_next_trait_solver/src/solve/assembly/mod.rs"),
                                    ::tracing_core::__macro_support::Option::Some(952u32),
                                    ::tracing_core::__macro_support::Option::Some("rustc_next_trait_solver::solve::assembly"),
                                    ::tracing_core::field::FieldSet::new(&[],
                                        ::tracing_core::callsite::Identifier(&__CALLSITE)),
                                    ::tracing::metadata::Kind::SPAN)
                            };
                        ::tracing::callsite::DefaultCallsite::new(&META)
                    };
                let mut interest = ::tracing::subscriber::Interest::never();
                if ::tracing::Level::TRACE <=
                                    ::tracing::level_filters::STATIC_MAX_LEVEL &&
                                ::tracing::Level::TRACE <=
                                    ::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,
                        &{ meta.fields().value_set(&[]) })
                } 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:
                    Result<Candidate<I>, NoSolutionOrRerunNonErased> = loop {};
            return __tracing_attr_fake_return;
        }
        {
            self.probe_trait_candidate(CandidateSource::CoherenceUnknowable).enter(|ecx|
                    {
                        let cx = ecx.cx();
                        let trait_ref = goal.predicate.trait_ref(cx);
                        if ecx.trait_ref_is_knowable(goal.param_env, trait_ref)? {
                            Err(NoSolution.into())
                        } else {
                            let predicate: I::Predicate = trait_ref.upcast(cx);
                            ecx.add_goals(GoalSource::Misc,
                                elaborate::elaborate(cx,
                                            [predicate]).skip(1).map(|predicate|
                                        goal.with(cx, predicate)));
                            ecx.evaluate_added_goals_and_make_canonical_response(Certainty::AMBIGUOUS)
                        }
                    })
        }
    }
}#[instrument(level = "trace", skip_all)]
953    fn consider_coherence_unknowable_candidate<G: GoalKind<D>>(
954        &mut self,
955        goal: Goal<I, G>,
956    ) -> Result<Candidate<I>, NoSolutionOrRerunNonErased> {
957        self.probe_trait_candidate(CandidateSource::CoherenceUnknowable).enter(|ecx| {
958            let cx = ecx.cx();
959            let trait_ref = goal.predicate.trait_ref(cx);
960            if ecx.trait_ref_is_knowable(goal.param_env, trait_ref)? {
961                Err(NoSolution.into())
962            } else {
963                // While the trait bound itself may be unknowable, we may be able to
964                // prove that a super trait is not implemented. For this, we recursively
965                // prove the super trait bounds of the current goal.
966                //
967                // We skip the goal itself as that one would cycle.
968                let predicate: I::Predicate = trait_ref.upcast(cx);
969                ecx.add_goals(
970                    GoalSource::Misc,
971                    elaborate::elaborate(cx, [predicate])
972                        .skip(1)
973                        .map(|predicate| goal.with(cx, predicate)),
974                );
975                ecx.evaluate_added_goals_and_make_canonical_response(Certainty::AMBIGUOUS)
976            }
977        })
978    }
979}
980
981pub(super) enum AllowInferenceConstraints {
982    Yes,
983    No,
984}
985
986impl<D, I> EvalCtxt<'_, D>
987where
988    D: SolverDelegate<Interner = I>,
989    I: Interner,
990{
991    /// Check whether we can ignore impl candidates due to specialization.
992    ///
993    /// This is only necessary for `feature(specialization)` and seems quite ugly.
994    pub(super) fn filter_specialized_impls(
995        &mut self,
996        allow_inference_constraints: AllowInferenceConstraints,
997        candidates: &mut Vec<Candidate<I>>,
998    ) {
999        if self.typing_mode().is_coherence() {
1000            return;
1001        }
1002
1003        let mut i = 0;
1004        'outer: while i < candidates.len() {
1005            let CandidateSource::Impl(victim_def_id) = candidates[i].source else {
1006                i += 1;
1007                continue;
1008            };
1009
1010            for (j, c) in candidates.iter().enumerate() {
1011                if i == j {
1012                    continue;
1013                }
1014
1015                let CandidateSource::Impl(other_def_id) = c.source else {
1016                    continue;
1017                };
1018
1019                // See if we can toss out `victim` based on specialization.
1020                //
1021                // While this requires us to know *for sure* that the `lhs` impl applies
1022                // we still use modulo regions here. This is fine as specialization currently
1023                // assumes that specializing impls have to be always applicable, meaning that
1024                // the only allowed region constraints may be constraints also present on the default impl.
1025                if #[allow(non_exhaustive_omitted_patterns)] match allow_inference_constraints {
    AllowInferenceConstraints::Yes => true,
    _ => false,
}matches!(allow_inference_constraints, AllowInferenceConstraints::Yes)
1026                    || has_only_region_constraints(c.result)
1027                {
1028                    if self.cx().impl_specializes(other_def_id, victim_def_id) {
1029                        candidates.remove(i);
1030                        continue 'outer;
1031                    }
1032                }
1033            }
1034
1035            i += 1;
1036        }
1037    }
1038
1039    /// If the self type is the hidden type of an opaque, try to assemble
1040    /// candidates for it by consider its item bounds and by using blanket
1041    /// impls. This is used to incompletely guide type inference when handling
1042    /// non-defining uses in the defining scope.
1043    ///
1044    /// We otherwise just fail fail with ambiguity. Even if we're using an
1045    /// opaque type item bound or a blank impls, we still force its certainty
1046    /// to be `Maybe` so that we properly prove this goal later.
1047    ///
1048    /// See <https://github.com/rust-lang/trait-system-refactor-initiative/issues/182>
1049    /// for why this is necessary.
1050    #[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("try_assemble_bounds_via_registered_opaques",
                                    "rustc_next_trait_solver::solve::assembly",
                                    ::tracing::Level::INFO,
                                    ::tracing_core::__macro_support::Option::Some("compiler/rustc_next_trait_solver/src/solve/assembly/mod.rs"),
                                    ::tracing_core::__macro_support::Option::Some(1050u32),
                                    ::tracing_core::__macro_support::Option::Some("rustc_next_trait_solver::solve::assembly"),
                                    ::tracing_core::field::FieldSet::new(&["goal",
                                                    "candidates"],
                                        ::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(&::tracing::field::debug(&goal)
                                                            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(&candidates)
                                                            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: Result<(), RerunNonErased> =
                loop {};
            return __tracing_attr_fake_return;
        }
        {
            let self_ty = goal.predicate.self_ty();
            let opaque_types =
                match self.typing_mode() {
                    TypingMode::Typeck { .. } =>
                        self.opaques_with_sub_unified_hidden_type(self_ty),
                    TypingMode::Coherence |
                        TypingMode::PostTypeckUntilBorrowck { .. } |
                        TypingMode::PostBorrowck { .. } | TypingMode::PostAnalysis |
                        TypingMode::Codegen => ::alloc::vec::Vec::new(),
                    TypingMode::ErasedNotCoherence(MayBeErased) => {
                        self.opaque_accesses.rerun_if_any_opaque_has_infer_as_hidden_type(RerunReason::SelfTyInfer)?;
                        Vec::new()
                    }
                };
            if opaque_types.is_empty() {
                candidates.extend(self.forced_ambiguity(MaybeInfo::AMBIGUOUS));
                return Ok(());
            }
            for &opaque_ty in &opaque_types {
                {
                    use ::tracing::__macro_support::Callsite as _;
                    static __CALLSITE: ::tracing::callsite::DefaultCallsite =
                        {
                            static META: ::tracing::Metadata<'static> =
                                {
                                    ::tracing_core::metadata::Metadata::new("event compiler/rustc_next_trait_solver/src/solve/assembly/mod.rs:1079",
                                        "rustc_next_trait_solver::solve::assembly",
                                        ::tracing::Level::DEBUG,
                                        ::tracing_core::__macro_support::Option::Some("compiler/rustc_next_trait_solver/src/solve/assembly/mod.rs"),
                                        ::tracing_core::__macro_support::Option::Some(1079u32),
                                        ::tracing_core::__macro_support::Option::Some("rustc_next_trait_solver::solve::assembly"),
                                        ::tracing_core::field::FieldSet::new(&["message"],
                                            ::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(&format_args!("self ty is sub unified with {0:?}",
                                                                    opaque_ty) as &dyn Value))])
                            });
                    } else { ; }
                };
                struct ReplaceOpaque<I: Interner> {
                    cx: I,
                    opaque_ty: ty::OpaqueAliasTy<I>,
                    self_ty: I::Ty,
                }
                impl<I: Interner> TypeFolder<I> for ReplaceOpaque<I> {
                    fn cx(&self) -> I { self.cx }
                    fn fold_ty(&mut self, ty: I::Ty) -> I::Ty {
                        if let ty::Alias(alias_ty) = ty.kind() &&
                                let Some(opaque_ty) = alias_ty.try_to_opaque() {
                            if opaque_ty == self.opaque_ty { return self.self_ty; }
                        }
                        ty.super_fold_with(self)
                    }
                }
                for item_bound in
                    self.cx().item_self_bounds(opaque_ty.kind.into()).iter_instantiated(self.cx(),
                            opaque_ty.args).map(Unnormalized::skip_norm_wip) {
                    let assumption =
                        item_bound.fold_with(&mut ReplaceOpaque {
                                    cx: self.cx(),
                                    opaque_ty,
                                    self_ty,
                                });
                    candidates.extend(G::probe_and_match_goal_against_assumption(self,
                            CandidateSource::AliasBound(AliasBoundKind::SelfBounds),
                            goal, assumption,
                            |ecx|
                                {
                                    ecx.evaluate_added_goals_and_make_canonical_response(Certainty::AMBIGUOUS)
                                }));
                }
            }
            if assemble_from.should_assemble_impl_candidates() {
                let cx = self.cx();
                cx.for_each_blanket_impl(goal.predicate.trait_def_id(cx),
                        |impl_def_id|
                            {
                                if cx.impl_is_default(impl_def_id) { return Ok(()); }
                                match G::consider_impl_candidate(self, goal, impl_def_id,
                                                |ecx, certainty|
                                                    {
                                                        if ecx.shallow_resolve(self_ty).is_ty_var() {
                                                            let certainty = certainty.and(Certainty::AMBIGUOUS);
                                                            ecx.evaluate_added_goals_and_make_canonical_response(certainty)
                                                        } else { Err(NoSolution.into()) }
                                                    }).map_err_to_rerun()? {
                                    Ok(candidate) => candidates.push(candidate),
                                    Err(NoSolution) => {}
                                }
                                Ok(())
                            })?;
            }
            if candidates.is_empty() {
                let source =
                    CandidateSource::BuiltinImpl(BuiltinImplSource::Misc);
                let certainty =
                    Certainty::Maybe(MaybeInfo {
                            cause: MaybeCause::Ambiguity,
                            opaque_types_jank: OpaqueTypesJank::ErrorIfRigidSelfTy,
                            stalled_on_coroutines: StalledOnCoroutines::No,
                        });
                candidates.extend(self.probe_trait_candidate(source).enter(|this|
                            {
                                this.evaluate_added_goals_and_make_canonical_response(certainty)
                            }));
            }
            Ok(())
        }
    }
}#[tracing::instrument(skip(self, assemble_from))]
1051    fn try_assemble_bounds_via_registered_opaques<G: GoalKind<D>>(
1052        &mut self,
1053        goal: Goal<I, G>,
1054        assemble_from: AssembleCandidatesFrom,
1055        candidates: &mut Vec<Candidate<I>>,
1056    ) -> Result<(), RerunNonErased> {
1057        let self_ty = goal.predicate.self_ty();
1058        // We only use this hack during HIR typeck.
1059        let opaque_types = match self.typing_mode() {
1060            TypingMode::Typeck { .. } => self.opaques_with_sub_unified_hidden_type(self_ty),
1061            TypingMode::Coherence
1062            | TypingMode::PostTypeckUntilBorrowck { .. }
1063            | TypingMode::PostBorrowck { .. }
1064            | TypingMode::PostAnalysis
1065            | TypingMode::Codegen => vec![],
1066            TypingMode::ErasedNotCoherence(MayBeErased) => {
1067                self.opaque_accesses
1068                    .rerun_if_any_opaque_has_infer_as_hidden_type(RerunReason::SelfTyInfer)?;
1069                Vec::new()
1070            }
1071        };
1072
1073        if opaque_types.is_empty() {
1074            candidates.extend(self.forced_ambiguity(MaybeInfo::AMBIGUOUS));
1075            return Ok(());
1076        }
1077
1078        for &opaque_ty in &opaque_types {
1079            debug!("self ty is sub unified with {opaque_ty:?}");
1080
1081            struct ReplaceOpaque<I: Interner> {
1082                cx: I,
1083                opaque_ty: ty::OpaqueAliasTy<I>,
1084                self_ty: I::Ty,
1085            }
1086            impl<I: Interner> TypeFolder<I> for ReplaceOpaque<I> {
1087                fn cx(&self) -> I {
1088                    self.cx
1089                }
1090                fn fold_ty(&mut self, ty: I::Ty) -> I::Ty {
1091                    if let ty::Alias(alias_ty) = ty.kind()
1092                        && let Some(opaque_ty) = alias_ty.try_to_opaque()
1093                    {
1094                        if opaque_ty == self.opaque_ty {
1095                            return self.self_ty;
1096                        }
1097                    }
1098                    ty.super_fold_with(self)
1099                }
1100            }
1101
1102            // We look at all item-bounds of the opaque, replacing the
1103            // opaque with the current self type before considering
1104            // them as a candidate. Imagine we've got `?x: Trait<?y>`
1105            // and `?x` has been sub-unified with the hidden type of
1106            // `impl Trait<u32>`, We take the item bound `opaque: Trait<u32>`
1107            // and replace all occurrences of `opaque` with `?x`. This results
1108            // in a `?x: Trait<u32>` alias-bound candidate.
1109            for item_bound in self
1110                .cx()
1111                .item_self_bounds(opaque_ty.kind.into())
1112                .iter_instantiated(self.cx(), opaque_ty.args)
1113                .map(Unnormalized::skip_norm_wip)
1114            {
1115                let assumption =
1116                    item_bound.fold_with(&mut ReplaceOpaque { cx: self.cx(), opaque_ty, self_ty });
1117                candidates.extend(G::probe_and_match_goal_against_assumption(
1118                    self,
1119                    CandidateSource::AliasBound(AliasBoundKind::SelfBounds),
1120                    goal,
1121                    assumption,
1122                    |ecx| {
1123                        // We want to reprove this goal once we've inferred the
1124                        // hidden type, so we force the certainty to `Maybe`.
1125                        ecx.evaluate_added_goals_and_make_canonical_response(Certainty::AMBIGUOUS)
1126                    },
1127                ));
1128            }
1129        }
1130
1131        // If the self type is sub unified with any opaque type, we also look at blanket
1132        // impls for it.
1133        //
1134        // See tests/ui/impl-trait/non-defining-uses/use-blanket-impl.rs for an example.
1135        if assemble_from.should_assemble_impl_candidates() {
1136            let cx = self.cx();
1137            cx.for_each_blanket_impl(goal.predicate.trait_def_id(cx), |impl_def_id| {
1138                // For every `default impl`, there's always a non-default `impl`
1139                // that will *also* apply. There's no reason to register a candidate
1140                // for this impl, since it is *not* proof that the trait goal holds.
1141                if cx.impl_is_default(impl_def_id) {
1142                    return Ok(());
1143                }
1144
1145                match G::consider_impl_candidate(self, goal, impl_def_id, |ecx, certainty| {
1146                    if ecx.shallow_resolve(self_ty).is_ty_var() {
1147                        // We force the certainty of impl candidates to be `Maybe`.
1148                        let certainty = certainty.and(Certainty::AMBIGUOUS);
1149                        ecx.evaluate_added_goals_and_make_canonical_response(certainty)
1150                    } else {
1151                        // We don't want to use impls if they constrain the opaque.
1152                        //
1153                        // FIXME(trait-system-refactor-initiative#229): This isn't
1154                        // perfect yet as it still allows us to incorrectly constrain
1155                        // other inference variables.
1156                        Err(NoSolution.into())
1157                    }
1158                })
1159                .map_err_to_rerun()?
1160                {
1161                    Ok(candidate) => candidates.push(candidate),
1162                    Err(NoSolution) => {}
1163                }
1164
1165                Ok(())
1166            })?;
1167        }
1168
1169        if candidates.is_empty() {
1170            let source = CandidateSource::BuiltinImpl(BuiltinImplSource::Misc);
1171            let certainty = Certainty::Maybe(MaybeInfo {
1172                cause: MaybeCause::Ambiguity,
1173                opaque_types_jank: OpaqueTypesJank::ErrorIfRigidSelfTy,
1174                stalled_on_coroutines: StalledOnCoroutines::No,
1175            });
1176            candidates
1177                .extend(self.probe_trait_candidate(source).enter(|this| {
1178                    this.evaluate_added_goals_and_make_canonical_response(certainty)
1179                }));
1180        }
1181
1182        Ok(())
1183    }
1184
1185    /// Assemble and merge candidates for goals which are related to an underlying trait
1186    /// goal. Right now, this is normalizes-to and host effect goals.
1187    ///
1188    /// We sadly can't simply take all possible candidates for normalization goals
1189    /// and check whether they result in the same constraints. We want to make sure
1190    /// that trying to normalize an alias doesn't result in constraints which aren't
1191    /// otherwise required.
1192    ///
1193    /// Most notably, when proving a trait goal by via a where-bound, we should not
1194    /// normalize via impls which have stricter region constraints than the where-bound:
1195    ///
1196    /// ```rust
1197    /// trait Trait<'a> {
1198    ///     type Assoc;
1199    /// }
1200    ///
1201    /// impl<'a, T: 'a> Trait<'a> for T {
1202    ///     type Assoc = u32;
1203    /// }
1204    ///
1205    /// fn with_bound<'a, T: Trait<'a>>(_value: T::Assoc) {}
1206    /// ```
1207    ///
1208    /// The where-bound of `with_bound` doesn't specify the associated type, so we would
1209    /// only be able to normalize `<T as Trait<'a>>::Assoc` by using the impl. This impl
1210    /// adds a `T: 'a` bound however, which would result in a region error. Given that the
1211    /// user explicitly wrote that `T: Trait<'a>` holds, this is undesirable and we instead
1212    /// treat the alias as rigid.
1213    ///
1214    /// See trait-system-refactor-initiative#124 for more details.
1215    x;#[instrument(level = "debug", skip_all, fields(proven_via, goal), ret)]
1216    pub(super) fn assemble_and_merge_candidates<G: GoalKind<D>>(
1217        &mut self,
1218        proven_via: Option<TraitGoalProvenVia>,
1219        goal: Goal<I, G>,
1220        inject_forced_ambiguity_candidate: impl FnOnce(
1221            &mut EvalCtxt<'_, D>,
1222        ) -> Option<
1223            Result<CanonicalResponse<I>, NoSolutionOrRerunNonErased>,
1224        >,
1225        inject_normalize_to_rigid_candidate: impl FnOnce(
1226            &mut EvalCtxt<'_, D>,
1227        ) -> Result<
1228            CanonicalResponse<I>,
1229            NoSolutionOrRerunNonErased,
1230        >,
1231    ) -> QueryResultOrRerunNonErased<I> {
1232        let Some(proven_via) = proven_via else {
1233            // We don't care about overflow. If proving the trait goal overflowed, then
1234            // it's enough to report an overflow error for that, we don't also have to
1235            // overflow during normalization.
1236            //
1237            // We use `forced_ambiguity` here over `make_ambiguous_response_no_constraints`
1238            // because the former will also record a built-in candidate in the inspector.
1239            return self.forced_ambiguity(MaybeInfo::AMBIGUOUS).map(|cand| cand.result);
1240        };
1241
1242        match proven_via {
1243            TraitGoalProvenVia::ParamEnv | TraitGoalProvenVia::AliasBound => {
1244                // Even when a trait bound has been proven using a where-bound, we
1245                // still need to consider alias-bounds for normalization, see
1246                // `tests/ui/next-solver/alias-bound-shadowed-by-env.rs`.
1247                let (mut candidates, _) = self
1248                    .assemble_and_evaluate_candidates(goal, AssembleCandidatesFrom::EnvAndBounds)?;
1249                debug!(?candidates);
1250
1251                // If the trait goal has been proven by using the environment, we want to treat
1252                // aliases as rigid if there are no applicable projection bounds in the environment.
1253                if candidates.is_empty() {
1254                    return inject_normalize_to_rigid_candidate(self);
1255                }
1256
1257                // If we're normalizing an GAT, we bail if using a where-bound would constrain
1258                // its generic arguments.
1259                if let Some(result) = inject_forced_ambiguity_candidate(self) {
1260                    return result;
1261                }
1262
1263                // We still need to prefer where-bounds over alias-bounds however.
1264                // See `tests/ui/winnowing/norm-where-bound-gt-alias-bound.rs`.
1265                if candidates.iter().any(|c| matches!(c.source, CandidateSource::ParamEnv(_))) {
1266                    candidates.retain(|c| matches!(c.source, CandidateSource::ParamEnv(_)));
1267                }
1268
1269                if let Some((response, _)) = self.try_merge_candidates(&candidates) {
1270                    Ok(response)
1271                } else {
1272                    self.flounder(&candidates).map_err(Into::into)
1273                }
1274            }
1275            TraitGoalProvenVia::Misc => {
1276                let (mut candidates, _) =
1277                    self.assemble_and_evaluate_candidates(goal, AssembleCandidatesFrom::All)?;
1278
1279                // Prefer "orphaned" param-env normalization predicates, which are used
1280                // (for example, and ideally only) when proving item bounds for an impl.
1281                if candidates.iter().any(|c| matches!(c.source, CandidateSource::ParamEnv(_))) {
1282                    candidates.retain(|c| matches!(c.source, CandidateSource::ParamEnv(_)));
1283                }
1284
1285                // We drop specialized impls to allow normalization via a final impl here. In case
1286                // the specializing impl has different inference constraints from the specialized
1287                // impl, proving the trait goal is already ambiguous, so we never get here. This
1288                // means we can just ignore inference constraints and don't have to special-case
1289                // constraining the normalized-to `term`.
1290                self.filter_specialized_impls(AllowInferenceConstraints::Yes, &mut candidates);
1291                if let Some((response, _)) = self.try_merge_candidates(&candidates) {
1292                    Ok(response)
1293                } else {
1294                    self.flounder(&candidates).map_err(Into::into)
1295                }
1296            }
1297        }
1298    }
1299
1300    /// Compute whether a param-env assumption is global or non-global after normalizing it.
1301    ///
1302    /// This is necessary because, for example, given:
1303    ///
1304    /// ```ignore,rust
1305    /// where
1306    ///     T: Trait<Assoc = u32>,
1307    ///     i32: From<T::Assoc>,
1308    /// ```
1309    ///
1310    /// The `i32: From<T::Assoc>` bound is non-global before normalization, but is global after.
1311    /// Since the old trait solver normalized param-envs eagerly, we want to emulate this
1312    /// behavior lazily.
1313    fn characterize_param_env_assumption(
1314        &mut self,
1315        param_env: I::ParamEnv,
1316        assumption: I::Clause,
1317    ) -> Result<(CandidateSource<I>, Certainty), NoSolution> {
1318        // FIXME: This should be fixed, but it also requires changing the behavior
1319        // in the old solver which is currently relied on.
1320        if assumption.has_bound_vars() {
1321            return Ok((CandidateSource::ParamEnv(ParamEnvSource::NonGlobal), Certainty::Yes));
1322        }
1323
1324        match assumption.visit_with(&mut FindParamInClause {
1325            ecx: self,
1326            param_env,
1327            universes: ::alloc::vec::Vec::new()vec![],
1328            recursion_depth: 0,
1329        }) {
1330            ControlFlow::Break(Err(NoSolution)) => Err(NoSolution),
1331            ControlFlow::Break(Ok(certainty)) => {
1332                Ok((CandidateSource::ParamEnv(ParamEnvSource::NonGlobal), certainty))
1333            }
1334            ControlFlow::Continue(()) => {
1335                Ok((CandidateSource::ParamEnv(ParamEnvSource::Global), Certainty::Yes))
1336            }
1337        }
1338    }
1339}
1340
1341struct FindParamInClause<'a, 'b, D: SolverDelegate<Interner = I>, I: Interner> {
1342    ecx: &'a mut EvalCtxt<'b, D>,
1343    param_env: I::ParamEnv,
1344    universes: Vec<Option<ty::UniverseIndex>>,
1345    recursion_depth: usize,
1346}
1347
1348impl<D, I> TypeVisitor<I> for FindParamInClause<'_, '_, D, I>
1349where
1350    D: SolverDelegate<Interner = I>,
1351    I: Interner,
1352{
1353    // - `Continue(())`: no generic parameter was found, the type is global
1354    // - `Break(Ok(Certainty::Yes))`: a generic parameter was found, the type is non-global
1355    // - `Break(Ok(Certainty::Maybe(_)))`: the recursion limit reached, assume that the type is non-global
1356    // - `Break(Err(NoSolution))`: normalization failed
1357    type Result = ControlFlow<Result<Certainty, NoSolution>>;
1358
1359    fn visit_binder<T: TypeVisitable<I>>(&mut self, t: &ty::Binder<I, T>) -> Self::Result {
1360        self.universes.push(None);
1361        t.super_visit_with(self)?;
1362        self.universes.pop();
1363        ControlFlow::Continue(())
1364    }
1365
1366    fn visit_ty(&mut self, ty: I::Ty) -> Self::Result {
1367        let ty = self.ecx.replace_bound_vars(ty, &mut self.universes);
1368        let Ok(ty) = self.ecx.structurally_normalize_ty(self.param_env, ty) else {
1369            return ControlFlow::Break(Err(NoSolution));
1370        };
1371
1372        match ty.kind() {
1373            ty::Placeholder(p) => {
1374                if p.universe() == ty::UniverseIndex::ROOT {
1375                    ControlFlow::Break(Ok(Certainty::Yes))
1376                } else {
1377                    ControlFlow::Continue(())
1378                }
1379            }
1380            ty::Infer(_) => ControlFlow::Break(Ok(Certainty::AMBIGUOUS)),
1381            _ if ty.has_type_flags(
1382                TypeFlags::HAS_PLACEHOLDER | TypeFlags::HAS_INFER | TypeFlags::HAS_ALIAS,
1383            ) =>
1384            {
1385                self.recursion_depth += 1;
1386                if self.recursion_depth > self.ecx.cx().recursion_limit() {
1387                    return ControlFlow::Break(Ok(Certainty::Maybe(MaybeInfo {
1388                        cause: MaybeCause::Overflow {
1389                            suggest_increasing_limit: true,
1390                            keep_constraints: false,
1391                        },
1392                        opaque_types_jank: OpaqueTypesJank::AllGood,
1393                        stalled_on_coroutines: StalledOnCoroutines::No,
1394                    })));
1395                }
1396                let result = ty.super_visit_with(self);
1397                self.recursion_depth -= 1;
1398                result
1399            }
1400            _ => ControlFlow::Continue(()),
1401        }
1402    }
1403
1404    fn visit_const(&mut self, ct: I::Const) -> Self::Result {
1405        let ct = self.ecx.replace_bound_vars(ct, &mut self.universes);
1406        let Ok(ct) = self.ecx.structurally_normalize_const(self.param_env, ct) else {
1407            return ControlFlow::Break(Err(NoSolution));
1408        };
1409
1410        match ct.kind() {
1411            ty::ConstKind::Placeholder(p) => {
1412                if p.universe() == ty::UniverseIndex::ROOT {
1413                    ControlFlow::Break(Ok(Certainty::Yes))
1414                } else {
1415                    ControlFlow::Continue(())
1416                }
1417            }
1418            ty::ConstKind::Infer(_) => ControlFlow::Break(Ok(Certainty::AMBIGUOUS)),
1419            _ if ct.has_type_flags(
1420                TypeFlags::HAS_PLACEHOLDER | TypeFlags::HAS_INFER | TypeFlags::HAS_ALIAS,
1421            ) =>
1422            {
1423                // FIXME(mgca): we should also check the recursion limit here
1424                ct.super_visit_with(self)
1425            }
1426            _ => ControlFlow::Continue(()),
1427        }
1428    }
1429
1430    fn visit_region(&mut self, r: I::Region) -> Self::Result {
1431        match self.ecx.eager_resolve_region(r).kind() {
1432            ty::ReStatic | ty::ReError(_) | ty::ReBound(..) => ControlFlow::Continue(()),
1433            ty::RePlaceholder(p) => {
1434                if p.universe() == ty::UniverseIndex::ROOT {
1435                    ControlFlow::Break(Ok(Certainty::Yes))
1436                } else {
1437                    ControlFlow::Continue(())
1438                }
1439            }
1440            ty::ReVar(_) => ControlFlow::Break(Ok(Certainty::Yes)),
1441            ty::ReErased | ty::ReEarlyParam(_) | ty::ReLateParam(_) => {
1442                {
    ::core::panicking::panic_fmt(format_args!("internal error: entered unreachable code: {0}",
            format_args!("unexpected region in param-env clause")));
}unreachable!("unexpected region in param-env clause")
1443            }
1444        }
1445    }
1446}