1use rustc_type_ir::data_structures::IndexSet;
4use rustc_type_ir::fast_reject::DeepRejectCtxt;
5use rustc_type_ir::inherent::*;
6use rustc_type_ir::lang_items::SolverTraitLangItem;
7use rustc_type_ir::solve::{
8 AliasBoundKind, CandidatePreferenceMode, CanonicalResponse, MaybeInfo,
9 NoSolutionOrRerunNonErased, OpaqueTypesJank, QueryResultOrRerunNonErased, RerunNonErased,
10 RerunReason, RerunResultExt, SizedTraitKind,
11};
12use rustc_type_ir::{
13 self as ty, FieldInfo, Interner, MayBeErased, Movability, PredicatePolarity, TraitPredicate,
14 TraitRef, TypeVisitableExt as _, TypingMode, Unnormalized, Upcast as _, elaborate,
15};
16use tracing::{debug, instrument, trace, warn};
17
18use crate::delegate::SolverDelegate;
19use crate::solve::assembly::structural_traits::{self, AsyncCallableRelevantTypes};
20use crate::solve::assembly::{
21 self, AllowInferenceConstraints, AssembleCandidatesFrom, Candidate, FailedCandidateInfo,
22};
23use crate::solve::inspect::ProbeKind;
24use crate::solve::{
25 BuiltinImplSource, CandidateSource, Certainty, EvalCtxt, Goal, GoalSource, MaybeCause,
26 MergeCandidateInfo, NoSolution, ParamEnvSource, StalledOnCoroutines,
27 has_only_region_constraints,
28};
29
30impl<D, I> assembly::GoalKind<D> for TraitPredicate<I>
31where
32 D: SolverDelegate<Interner = I>,
33 I: Interner,
34{
35 fn self_ty(self) -> I::Ty {
36 self.self_ty()
37 }
38
39 fn trait_ref(self, _: I) -> ty::TraitRef<I> {
40 self.trait_ref
41 }
42
43 fn with_replaced_self_ty(self, cx: I, self_ty: I::Ty) -> Self {
44 self.with_replaced_self_ty(cx, self_ty)
45 }
46
47 fn trait_def_id(self, _: I) -> I::TraitId {
48 self.def_id()
49 }
50
51 fn consider_additional_alias_assumptions(
52 _ecx: &mut EvalCtxt<'_, D>,
53 _goal: Goal<I, Self>,
54 _alias_ty: ty::AliasTy<I>,
55 ) -> Vec<Candidate<I>> {
56 ::alloc::vec::Vec::new()vec![]
57 }
58
59 fn consider_impl_candidate(
60 ecx: &mut EvalCtxt<'_, D>,
61 goal: Goal<I, TraitPredicate<I>>,
62 impl_def_id: I::ImplId,
63 then: impl FnOnce(&mut EvalCtxt<'_, D>, Certainty) -> QueryResultOrRerunNonErased<I>,
64 ) -> Result<Candidate<I>, NoSolutionOrRerunNonErased> {
65 let cx = ecx.cx();
66
67 let impl_trait_ref = cx.impl_trait_ref(impl_def_id);
68 if !DeepRejectCtxt::relate_rigid_infer(ecx.cx())
69 .args_may_unify(goal.predicate.trait_ref.args, impl_trait_ref.skip_binder().args)
70 {
71 return Err(NoSolution.into());
72 }
73
74 let impl_polarity = cx.impl_polarity(impl_def_id);
77 let maximal_certainty = match (impl_polarity, goal.predicate.polarity) {
78 (ty::ImplPolarity::Reservation, _) => {
80 if ecx.typing_mode().is_coherence() {
81 Certainty::AMBIGUOUS
82 } else {
83 return Err(NoSolution.into());
84 }
85 }
86
87 (ty::ImplPolarity::Positive, ty::PredicatePolarity::Positive)
89 | (ty::ImplPolarity::Negative, ty::PredicatePolarity::Negative) => Certainty::Yes,
90
91 (ty::ImplPolarity::Positive, ty::PredicatePolarity::Negative)
93 | (ty::ImplPolarity::Negative, ty::PredicatePolarity::Positive) => {
94 return Err(NoSolution.into());
95 }
96 };
97
98 ecx.probe_trait_candidate(CandidateSource::Impl(impl_def_id)).enter(|ecx| {
99 let impl_args = ecx.fresh_args_for_item(impl_def_id.into());
100 ecx.record_impl_args(impl_args);
101 let impl_trait_ref = impl_trait_ref.instantiate(cx, impl_args).skip_norm_wip();
102
103 ecx.eq(goal.param_env, goal.predicate.trait_ref, impl_trait_ref)?;
104 let where_clause_bounds = cx
105 .predicates_of(impl_def_id.into())
106 .iter_instantiated(cx, impl_args)
107 .map(Unnormalized::skip_norm_wip)
108 .map(|pred| goal.with(cx, pred));
109 ecx.add_goals(GoalSource::ImplWhereBound, where_clause_bounds);
110
111 ecx.add_goals(
115 GoalSource::Misc,
116 cx.impl_super_outlives(impl_def_id)
117 .iter_instantiated(cx, impl_args)
118 .map(Unnormalized::skip_norm_wip)
119 .map(|pred| goal.with(cx, pred)),
120 );
121
122 then(ecx, maximal_certainty).map_err(Into::into)
123 })
124 }
125
126 fn consider_error_guaranteed_candidate(
127 ecx: &mut EvalCtxt<'_, D>,
128 _guar: I::ErrorGuaranteed,
129 ) -> Result<Candidate<I>, NoSolutionOrRerunNonErased> {
130 ecx.probe_builtin_trait_candidate(BuiltinImplSource::Misc)
131 .enter(|ecx| ecx.evaluate_added_goals_and_make_canonical_response(Certainty::Yes))
132 }
133
134 fn fast_reject_assumption(
135 ecx: &mut EvalCtxt<'_, D>,
136 goal: Goal<I, Self>,
137 assumption: I::Clause,
138 ) -> Result<(), NoSolution> {
139 fn trait_def_id_matches<I: Interner>(
140 cx: I,
141 clause_def_id: I::TraitId,
142 goal_def_id: I::TraitId,
143 polarity: PredicatePolarity,
144 ) -> bool {
145 clause_def_id == goal_def_id
146 || (polarity == PredicatePolarity::Positive
151 && cx.is_trait_lang_item(clause_def_id, SolverTraitLangItem::Sized)
152 && cx.is_trait_lang_item(goal_def_id, SolverTraitLangItem::MetaSized))
153 }
154
155 if let Some(trait_clause) = assumption.as_trait_clause()
156 && trait_clause.polarity() == goal.predicate.polarity
157 && trait_def_id_matches(
158 ecx.cx(),
159 trait_clause.def_id(),
160 goal.predicate.def_id(),
161 goal.predicate.polarity,
162 )
163 && DeepRejectCtxt::relate_rigid_rigid(ecx.cx()).args_may_unify(
164 goal.predicate.trait_ref.args,
165 trait_clause.skip_binder().trait_ref.args,
166 )
167 {
168 return Ok(());
169 } else {
170 Err(NoSolution)
171 }
172 }
173
174 fn match_assumption(
175 ecx: &mut EvalCtxt<'_, D>,
176 goal: Goal<I, Self>,
177 assumption: I::Clause,
178 then: impl FnOnce(&mut EvalCtxt<'_, D>) -> QueryResultOrRerunNonErased<I>,
179 ) -> QueryResultOrRerunNonErased<I> {
180 let trait_clause = assumption.as_trait_clause().unwrap();
181
182 if ecx.cx().is_trait_lang_item(goal.predicate.def_id(), SolverTraitLangItem::MetaSized)
188 && ecx.cx().is_trait_lang_item(trait_clause.def_id(), SolverTraitLangItem::Sized)
189 {
190 let meta_sized_clause =
191 trait_predicate_with_def_id(ecx.cx(), trait_clause, goal.predicate.def_id());
192 return Self::match_assumption(ecx, goal, meta_sized_clause, then);
193 }
194
195 let assumption_trait_pred = ecx.instantiate_binder_with_infer(trait_clause);
196 ecx.eq(goal.param_env, goal.predicate.trait_ref, assumption_trait_pred.trait_ref)?;
197
198 then(ecx)
199 }
200
201 fn consider_auto_trait_candidate(
202 ecx: &mut EvalCtxt<'_, D>,
203 goal: Goal<I, Self>,
204 ) -> Result<Candidate<I>, NoSolutionOrRerunNonErased> {
205 let cx = ecx.cx();
206 if goal.predicate.polarity != ty::PredicatePolarity::Positive {
207 return Err(NoSolution.into());
208 }
209
210 if let Some(result) = ecx.disqualify_auto_trait_candidate_due_to_possible_impl(goal) {
211 return result;
212 }
213
214 if cx.trait_is_unsafe(goal.predicate.def_id())
217 && goal.predicate.self_ty().has_unsafe_fields()
218 {
219 return Err(NoSolution.into());
220 }
221
222 if let ty::Alias(ty::AliasTy { kind: ty::Opaque { def_id }, .. }) =
238 goal.predicate.self_ty().kind()
239 {
240 if ecx.opaque_accesses.might_rerun() {
241 ecx.opaque_accesses.rerun_always(RerunReason::AutoTraitLeakage)?;
242 return Err(NoSolution.into());
243 }
244
245 if true {
if !ecx.opaque_type_is_rigid(def_id) {
::core::panicking::panic("assertion failed: ecx.opaque_type_is_rigid(def_id)")
};
};debug_assert!(ecx.opaque_type_is_rigid(def_id));
246 for item_bound in cx.item_self_bounds(def_id.into()).skip_binder() {
247 if item_bound
248 .as_trait_clause()
249 .is_some_and(|b| b.def_id() == goal.predicate.def_id())
250 {
251 return Err(NoSolution.into());
252 }
253 }
254 }
255
256 if let Some(cand) = ecx.try_stall_coroutine(goal.predicate.self_ty()) {
258 return cand;
259 }
260
261 ecx.probe_and_evaluate_goal_for_constituent_tys(
262 CandidateSource::BuiltinImpl(BuiltinImplSource::Misc),
263 goal,
264 structural_traits::instantiate_constituent_tys_for_auto_trait,
265 )
266 }
267
268 fn consider_trait_alias_candidate(
269 ecx: &mut EvalCtxt<'_, D>,
270 goal: Goal<I, Self>,
271 ) -> Result<Candidate<I>, NoSolutionOrRerunNonErased> {
272 if goal.predicate.polarity != ty::PredicatePolarity::Positive {
273 return Err(NoSolution.into());
274 }
275
276 let cx = ecx.cx();
277
278 ecx.probe_builtin_trait_candidate(BuiltinImplSource::Misc).enter(|ecx| {
279 let nested_obligations = cx
280 .predicates_of(goal.predicate.def_id().into())
281 .iter_instantiated(cx, goal.predicate.trait_ref.args)
282 .map(Unnormalized::skip_norm_wip)
283 .map(|p| goal.with(cx, p));
284 ecx.add_goals(GoalSource::Misc, nested_obligations);
290 ecx.evaluate_added_goals_and_make_canonical_response(Certainty::Yes)
291 })
292 }
293
294 fn consider_builtin_sizedness_candidates(
295 ecx: &mut EvalCtxt<'_, D>,
296 goal: Goal<I, Self>,
297 sizedness: SizedTraitKind,
298 ) -> Result<Candidate<I>, NoSolutionOrRerunNonErased> {
299 if goal.predicate.polarity != ty::PredicatePolarity::Positive {
300 return Err(NoSolution.into());
301 }
302
303 ecx.probe_and_evaluate_goal_for_constituent_tys(
304 CandidateSource::BuiltinImpl(BuiltinImplSource::Trivial),
305 goal,
306 |ecx, ty| {
307 structural_traits::instantiate_constituent_tys_for_sizedness_trait(
308 ecx, sizedness, ty,
309 )
310 },
311 )
312 }
313
314 fn consider_builtin_copy_clone_candidate(
315 ecx: &mut EvalCtxt<'_, D>,
316 goal: Goal<I, Self>,
317 ) -> Result<Candidate<I>, NoSolutionOrRerunNonErased> {
318 if goal.predicate.polarity != ty::PredicatePolarity::Positive {
319 return Err(NoSolution.into());
320 }
321
322 if let Some(cand) = ecx.try_stall_coroutine(goal.predicate.self_ty()) {
324 return cand;
325 }
326
327 ecx.probe_and_evaluate_goal_for_constituent_tys(
328 CandidateSource::BuiltinImpl(BuiltinImplSource::Misc),
329 goal,
330 structural_traits::instantiate_constituent_tys_for_copy_clone_trait,
331 )
332 }
333
334 fn consider_builtin_fn_ptr_trait_candidate(
335 ecx: &mut EvalCtxt<'_, D>,
336 goal: Goal<I, Self>,
337 ) -> Result<Candidate<I>, NoSolutionOrRerunNonErased> {
338 let self_ty = goal.predicate.self_ty();
339 match goal.predicate.polarity {
340 ty::PredicatePolarity::Positive => {
342 if self_ty.is_fn_ptr() {
343 ecx.probe_builtin_trait_candidate(BuiltinImplSource::Misc).enter(|ecx| {
344 ecx.evaluate_added_goals_and_make_canonical_response(Certainty::Yes)
345 })
346 } else {
347 Err(NoSolution.into())
348 }
349 }
350 ty::PredicatePolarity::Negative => {
352 if !self_ty.is_fn_ptr() && self_ty.is_known_rigid() {
355 ecx.probe_builtin_trait_candidate(BuiltinImplSource::Misc).enter(|ecx| {
356 ecx.evaluate_added_goals_and_make_canonical_response(Certainty::Yes)
357 })
358 } else {
359 Err(NoSolution.into())
360 }
361 }
362 }
363 }
364
365 fn consider_builtin_fn_trait_candidates(
366 ecx: &mut EvalCtxt<'_, D>,
367 goal: Goal<I, Self>,
368 goal_kind: ty::ClosureKind,
369 ) -> Result<Candidate<I>, NoSolutionOrRerunNonErased> {
370 if goal.predicate.polarity != ty::PredicatePolarity::Positive {
371 return Err(NoSolution.into());
372 }
373
374 let cx = ecx.cx();
375 let Some(tupled_inputs_and_output) =
376 structural_traits::extract_tupled_inputs_and_output_from_callable(
377 cx,
378 goal.predicate.self_ty(),
379 goal_kind,
380 )?
381 else {
382 return ecx.forced_ambiguity(MaybeInfo::AMBIGUOUS);
383 };
384 let (inputs, output) = ecx.instantiate_binder_with_infer(tupled_inputs_and_output);
385
386 let output_is_sized_pred =
389 ty::TraitRef::new(cx, cx.require_trait_lang_item(SolverTraitLangItem::Sized), [output]);
390
391 let pred =
392 ty::TraitRef::new(cx, goal.predicate.def_id(), [goal.predicate.self_ty(), inputs])
393 .upcast(cx);
394 Self::probe_and_consider_implied_clause(
395 ecx,
396 CandidateSource::BuiltinImpl(BuiltinImplSource::Misc),
397 goal,
398 pred,
399 [(GoalSource::ImplWhereBound, goal.with(cx, output_is_sized_pred))],
400 )
401 .map_err(Into::into)
402 }
403
404 fn consider_builtin_async_fn_trait_candidates(
405 ecx: &mut EvalCtxt<'_, D>,
406 goal: Goal<I, Self>,
407 goal_kind: ty::ClosureKind,
408 ) -> Result<Candidate<I>, NoSolutionOrRerunNonErased> {
409 if goal.predicate.polarity != ty::PredicatePolarity::Positive {
410 return Err(NoSolution.into());
411 }
412
413 let cx = ecx.cx();
414 let (tupled_inputs_and_output_and_coroutine, nested_preds) =
415 structural_traits::extract_tupled_inputs_and_output_from_async_callable(
416 cx,
417 goal.predicate.self_ty(),
418 goal_kind,
419 Region::new_static(cx),
421 )?;
422 let AsyncCallableRelevantTypes {
423 tupled_inputs_ty,
424 output_coroutine_ty,
425 coroutine_return_ty: _,
426 } = ecx.instantiate_binder_with_infer(tupled_inputs_and_output_and_coroutine);
427
428 let output_is_sized_pred = ty::TraitRef::new(
431 cx,
432 cx.require_trait_lang_item(SolverTraitLangItem::Sized),
433 [output_coroutine_ty],
434 );
435
436 let pred = ty::TraitRef::new(
437 cx,
438 goal.predicate.def_id(),
439 [goal.predicate.self_ty(), tupled_inputs_ty],
440 )
441 .upcast(cx);
442 Self::probe_and_consider_implied_clause(
443 ecx,
444 CandidateSource::BuiltinImpl(BuiltinImplSource::Misc),
445 goal,
446 pred,
447 [goal.with(cx, output_is_sized_pred)]
448 .into_iter()
449 .chain(nested_preds.into_iter().map(|pred| goal.with(cx, pred)))
450 .map(|goal| (GoalSource::ImplWhereBound, goal)),
451 )
452 .map_err(Into::into)
453 }
454
455 fn consider_builtin_async_fn_kind_helper_candidate(
456 ecx: &mut EvalCtxt<'_, D>,
457 goal: Goal<I, Self>,
458 ) -> Result<Candidate<I>, NoSolutionOrRerunNonErased> {
459 let [closure_fn_kind_ty, goal_kind_ty] = *goal.predicate.trait_ref.args.as_slice() else {
460 ::core::panicking::panic("explicit panic");panic!();
461 };
462
463 let Some(closure_kind) = closure_fn_kind_ty.expect_ty().to_opt_closure_kind() else {
464 return Err(NoSolution.into());
466 };
467 let goal_kind = goal_kind_ty.expect_ty().to_opt_closure_kind().unwrap();
468 if closure_kind.extends(goal_kind) {
469 ecx.probe_builtin_trait_candidate(BuiltinImplSource::Misc)
470 .enter(|ecx| ecx.evaluate_added_goals_and_make_canonical_response(Certainty::Yes))
471 } else {
472 Err(NoSolution.into())
473 }
474 }
475
476 fn consider_builtin_tuple_candidate(
483 ecx: &mut EvalCtxt<'_, D>,
484 goal: Goal<I, Self>,
485 ) -> Result<Candidate<I>, NoSolutionOrRerunNonErased> {
486 if goal.predicate.polarity != ty::PredicatePolarity::Positive {
487 return Err(NoSolution.into());
488 }
489
490 if let ty::Tuple(..) = goal.predicate.self_ty().kind() {
491 ecx.probe_builtin_trait_candidate(BuiltinImplSource::Misc)
492 .enter(|ecx| ecx.evaluate_added_goals_and_make_canonical_response(Certainty::Yes))
493 } else {
494 Err(NoSolution.into())
495 }
496 }
497
498 fn consider_builtin_pointee_candidate(
499 ecx: &mut EvalCtxt<'_, D>,
500 goal: Goal<I, Self>,
501 ) -> Result<Candidate<I>, NoSolutionOrRerunNonErased> {
502 if goal.predicate.polarity != ty::PredicatePolarity::Positive {
503 return Err(NoSolution.into());
504 }
505
506 ecx.probe_builtin_trait_candidate(BuiltinImplSource::Misc)
507 .enter(|ecx| ecx.evaluate_added_goals_and_make_canonical_response(Certainty::Yes))
508 }
509
510 fn consider_builtin_future_candidate(
511 ecx: &mut EvalCtxt<'_, D>,
512 goal: Goal<I, Self>,
513 ) -> Result<Candidate<I>, NoSolutionOrRerunNonErased> {
514 if goal.predicate.polarity != ty::PredicatePolarity::Positive {
515 return Err(NoSolution.into());
516 }
517
518 let ty::Coroutine(def_id, _) = goal.predicate.self_ty().kind() else {
519 return Err(NoSolution.into());
520 };
521
522 let cx = ecx.cx();
524 if !cx.coroutine_is_async(def_id) {
525 return Err(NoSolution.into());
526 }
527
528 ecx.probe_builtin_trait_candidate(BuiltinImplSource::Misc)
532 .enter(|ecx| ecx.evaluate_added_goals_and_make_canonical_response(Certainty::Yes))
533 }
534
535 fn consider_builtin_iterator_candidate(
536 ecx: &mut EvalCtxt<'_, D>,
537 goal: Goal<I, Self>,
538 ) -> Result<Candidate<I>, NoSolutionOrRerunNonErased> {
539 if goal.predicate.polarity != ty::PredicatePolarity::Positive {
540 return Err(NoSolution.into());
541 }
542
543 let ty::Coroutine(def_id, _) = goal.predicate.self_ty().kind() else {
544 return Err(NoSolution.into());
545 };
546
547 let cx = ecx.cx();
549 if !cx.coroutine_is_gen(def_id) {
550 return Err(NoSolution.into());
551 }
552
553 ecx.probe_builtin_trait_candidate(BuiltinImplSource::Misc)
557 .enter(|ecx| ecx.evaluate_added_goals_and_make_canonical_response(Certainty::Yes))
558 }
559
560 fn consider_builtin_fused_iterator_candidate(
561 ecx: &mut EvalCtxt<'_, D>,
562 goal: Goal<I, Self>,
563 ) -> Result<Candidate<I>, NoSolutionOrRerunNonErased> {
564 if goal.predicate.polarity != ty::PredicatePolarity::Positive {
565 return Err(NoSolution.into());
566 }
567
568 let ty::Coroutine(def_id, _) = goal.predicate.self_ty().kind() else {
569 return Err(NoSolution.into());
570 };
571
572 let cx = ecx.cx();
574 if !cx.coroutine_is_gen(def_id) {
575 return Err(NoSolution.into());
576 }
577
578 ecx.probe_builtin_trait_candidate(BuiltinImplSource::Misc)
580 .enter(|ecx| ecx.evaluate_added_goals_and_make_canonical_response(Certainty::Yes))
581 }
582
583 fn consider_builtin_async_iterator_candidate(
584 ecx: &mut EvalCtxt<'_, D>,
585 goal: Goal<I, Self>,
586 ) -> Result<Candidate<I>, NoSolutionOrRerunNonErased> {
587 if goal.predicate.polarity != ty::PredicatePolarity::Positive {
588 return Err(NoSolution.into());
589 }
590
591 let ty::Coroutine(def_id, _) = goal.predicate.self_ty().kind() else {
592 return Err(NoSolution.into());
593 };
594
595 let cx = ecx.cx();
597 if !cx.coroutine_is_async_gen(def_id) {
598 return Err(NoSolution.into());
599 }
600
601 ecx.probe_builtin_trait_candidate(BuiltinImplSource::Misc)
605 .enter(|ecx| ecx.evaluate_added_goals_and_make_canonical_response(Certainty::Yes))
606 }
607
608 fn consider_builtin_coroutine_candidate(
609 ecx: &mut EvalCtxt<'_, D>,
610 goal: Goal<I, Self>,
611 ) -> Result<Candidate<I>, NoSolutionOrRerunNonErased> {
612 if goal.predicate.polarity != ty::PredicatePolarity::Positive {
613 return Err(NoSolution.into());
614 }
615
616 let self_ty = goal.predicate.self_ty();
617 let ty::Coroutine(def_id, args) = self_ty.kind() else {
618 return Err(NoSolution.into());
619 };
620
621 let cx = ecx.cx();
623 if !cx.is_general_coroutine(def_id) {
624 return Err(NoSolution.into());
625 }
626
627 let coroutine = args.as_coroutine();
628 Self::probe_and_consider_implied_clause(
629 ecx,
630 CandidateSource::BuiltinImpl(BuiltinImplSource::Misc),
631 goal,
632 ty::TraitRef::new(cx, goal.predicate.def_id(), [self_ty, coroutine.resume_ty()])
633 .upcast(cx),
634 [],
637 )
638 }
639
640 fn consider_builtin_discriminant_kind_candidate(
641 ecx: &mut EvalCtxt<'_, D>,
642 goal: Goal<I, Self>,
643 ) -> Result<Candidate<I>, NoSolutionOrRerunNonErased> {
644 if goal.predicate.polarity != ty::PredicatePolarity::Positive {
645 return Err(NoSolution.into());
646 }
647
648 ecx.probe_builtin_trait_candidate(BuiltinImplSource::Misc)
650 .enter(|ecx| ecx.evaluate_added_goals_and_make_canonical_response(Certainty::Yes))
651 }
652
653 fn consider_builtin_destruct_candidate(
654 ecx: &mut EvalCtxt<'_, D>,
655 goal: Goal<I, Self>,
656 ) -> Result<Candidate<I>, NoSolutionOrRerunNonErased> {
657 if goal.predicate.polarity != ty::PredicatePolarity::Positive {
658 return Err(NoSolution.into());
659 }
660
661 ecx.probe_builtin_trait_candidate(BuiltinImplSource::Misc)
664 .enter(|ecx| ecx.evaluate_added_goals_and_make_canonical_response(Certainty::Yes))
665 }
666
667 fn consider_builtin_transmute_candidate(
668 ecx: &mut EvalCtxt<'_, D>,
669 goal: Goal<I, Self>,
670 ) -> Result<Candidate<I>, NoSolutionOrRerunNonErased> {
671 if goal.predicate.polarity != ty::PredicatePolarity::Positive {
672 return Err(NoSolution.into());
673 }
674
675 if goal.predicate.has_non_region_placeholders() {
677 return Err(NoSolution.into());
678 }
679
680 if goal.has_non_region_infer() {
683 return ecx.forced_ambiguity(MaybeInfo::AMBIGUOUS);
684 }
685
686 ecx.probe_builtin_trait_candidate(BuiltinImplSource::Misc).enter(
687 |ecx| -> Result<_, NoSolutionOrRerunNonErased> {
688 let assume = ecx.structurally_normalize_const(
689 goal.param_env,
690 goal.predicate.trait_ref.args.const_at(2),
691 )?;
692
693 let certainty = ecx.is_transmutable(
694 goal.predicate.trait_ref.args.type_at(0),
695 goal.predicate.trait_ref.args.type_at(1),
696 assume,
697 )?;
698 ecx.evaluate_added_goals_and_make_canonical_response(certainty).map_err(Into::into)
699 },
700 )
701 }
702
703 fn consider_builtin_bikeshed_guaranteed_no_drop_candidate(
716 ecx: &mut EvalCtxt<'_, D>,
717 goal: Goal<I, Self>,
718 ) -> Result<Candidate<I>, NoSolutionOrRerunNonErased> {
719 if goal.predicate.polarity != ty::PredicatePolarity::Positive {
720 return Err(NoSolution.into());
721 }
722
723 let cx = ecx.cx();
724 ecx.probe_builtin_trait_candidate(BuiltinImplSource::Misc).enter(|ecx| {
725 let ty = goal.predicate.self_ty();
726 match ty.kind() {
727 ty::Ref(..) => {}
729 ty::Adt(def, _) if def.is_manually_drop() => {}
731 ty::Tuple(tys) => {
734 ecx.add_goals(
735 GoalSource::ImplWhereBound,
736 tys.iter().map(|elem_ty| {
737 goal.with(cx, ty::TraitRef::new(cx, goal.predicate.def_id(), [elem_ty]))
738 }),
739 );
740 }
741 ty::Array(elem_ty, _) => {
742 ecx.add_goal(
743 GoalSource::ImplWhereBound,
744 goal.with(cx, ty::TraitRef::new(cx, goal.predicate.def_id(), [elem_ty])),
745 );
746 }
747
748 ty::FnDef(..)
752 | ty::FnPtr(..)
753 | ty::Error(_)
754 | ty::Uint(_)
755 | ty::Int(_)
756 | ty::Infer(ty::IntVar(_) | ty::FloatVar(_))
757 | ty::Bool
758 | ty::Float(_)
759 | ty::Char
760 | ty::RawPtr(..)
761 | ty::Never
762 | ty::Pat(..)
763 | ty::Dynamic(..)
764 | ty::Str
765 | ty::Slice(_)
766 | ty::Foreign(..)
767 | ty::Adt(..)
768 | ty::Alias(..)
769 | ty::Param(_)
770 | ty::Placeholder(..)
771 | ty::Closure(..)
772 | ty::CoroutineClosure(..)
773 | ty::Coroutine(..)
774 | ty::UnsafeBinder(_)
775 | ty::CoroutineWitness(..) => {
776 ecx.add_goal(
777 GoalSource::ImplWhereBound,
778 goal.with(
779 cx,
780 ty::TraitRef::new(
781 cx,
782 cx.require_trait_lang_item(SolverTraitLangItem::Copy),
783 [ty],
784 ),
785 ),
786 );
787 }
788
789 ty::Bound(..)
790 | ty::Infer(
791 ty::TyVar(_) | ty::FreshTy(_) | ty::FreshIntTy(_) | ty::FreshFloatTy(_),
792 ) => {
793 { ::core::panicking::panic_fmt(format_args!("unexpected type `{0:?}`", ty)); }panic!("unexpected type `{ty:?}`")
794 }
795 }
796
797 ecx.evaluate_added_goals_and_make_canonical_response(Certainty::Yes)
798 })
799 }
800
801 fn consider_structural_builtin_unsize_candidates(
809 ecx: &mut EvalCtxt<'_, D>,
810 goal: Goal<I, Self>,
811 ) -> Result<Vec<Candidate<I>>, RerunNonErased> {
812 if goal.predicate.polarity != ty::PredicatePolarity::Positive {
813 return Ok(::alloc::vec::Vec::new()vec![]);
814 }
815
816 let result = ecx.probe(|_| ProbeKind::UnsizeAssembly).enter(
817 |ecx| -> Result<Vec<Candidate<I>>, NoSolutionOrRerunNonErased> {
818 let a_ty = goal.predicate.self_ty();
819 let b_ty = ecx.structurally_normalize_ty(
822 goal.param_env,
823 goal.predicate.trait_ref.args.type_at(1),
824 )?;
825
826 let goal = goal.with(ecx.cx(), (a_ty, b_ty));
827 match (a_ty.kind(), b_ty.kind()) {
828 (ty::Infer(ty::TyVar(..)), ..) => {
::core::panicking::panic_fmt(format_args!("unexpected infer {0:?} {1:?}",
a_ty, b_ty));
}panic!("unexpected infer {a_ty:?} {b_ty:?}"),
829
830 (_, ty::Infer(ty::TyVar(..))) => {
831 Ok(::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[ecx.forced_ambiguity(MaybeInfo::AMBIGUOUS)?]))vec![ecx.forced_ambiguity(MaybeInfo::AMBIGUOUS)?])
832 }
833
834 (ty::Dynamic(a_data, a_region), ty::Dynamic(b_data, b_region)) => Ok(ecx
836 .consider_builtin_dyn_upcast_candidates(
837 goal, a_data, a_region, b_data, b_region,
838 )),
839
840 (_, ty::Dynamic(b_region, b_data)) => Ok(::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[ecx.consider_builtin_unsize_to_dyn_candidate(goal, b_region,
b_data)?]))vec![
842 ecx.consider_builtin_unsize_to_dyn_candidate(goal, b_region, b_data)?,
843 ]),
844
845 (ty::Array(a_elem_ty, ..), ty::Slice(b_elem_ty)) => {
847 Ok(::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[ecx.consider_builtin_array_unsize(goal, a_elem_ty, b_elem_ty)?]))vec![ecx.consider_builtin_array_unsize(goal, a_elem_ty, b_elem_ty)?])
848 }
849
850 (ty::Adt(a_def, a_args), ty::Adt(b_def, b_args))
852 if a_def.is_struct() && a_def == b_def =>
853 {
854 Ok(::alloc::boxed::box_assume_init_into_vec_unsafe(::alloc::intrinsics::write_box_via_move(::alloc::boxed::Box::new_uninit(),
[ecx.consider_builtin_struct_unsize(goal, a_def, a_args, b_args)?]))vec![ecx.consider_builtin_struct_unsize(goal, a_def, a_args, b_args)?])
855 }
856
857 _ => Err(NoSolution.into()),
858 }
859 },
860 );
861
862 match result.map_err_to_rerun()? {
863 Ok(resp) => Ok(resp),
864 Err(NoSolution) => Ok(::alloc::vec::Vec::new()vec![]),
865 }
866 }
867
868 fn consider_builtin_field_candidate(
869 ecx: &mut EvalCtxt<'_, D>,
870 goal: Goal<I, Self>,
871 ) -> Result<Candidate<I>, NoSolutionOrRerunNonErased> {
872 if goal.predicate.polarity != ty::PredicatePolarity::Positive {
873 return Err(NoSolution.into());
874 }
875 if let ty::Adt(def, args) = goal.predicate.self_ty().kind()
876 && let Some(FieldInfo { base, ty, .. }) =
877 def.field_representing_type_info(ecx.cx(), args)
878 && {
879 let sized_trait = ecx.cx().require_trait_lang_item(SolverTraitLangItem::Sized);
880 ecx.add_goal(
888 GoalSource::ImplWhereBound,
889 Goal {
890 param_env: goal.param_env,
891 predicate: TraitRef::new(ecx.cx(), sized_trait, [base]).upcast(ecx.cx()),
892 },
893 );
894 ecx.add_goal(
895 GoalSource::ImplWhereBound,
896 Goal {
897 param_env: goal.param_env,
898 predicate: TraitRef::new(ecx.cx(), sized_trait, [ty]).upcast(ecx.cx()),
899 },
900 );
901 ecx.try_evaluate_added_goals()? == Certainty::Yes
902 }
903 && match base.kind() {
904 ty::Adt(def, _) => def.is_struct() && !def.is_packed(),
905 ty::Tuple(..) => true,
906 _ => false,
907 }
908 {
909 ecx.probe_builtin_trait_candidate(BuiltinImplSource::Misc)
910 .enter(|ecx| ecx.evaluate_added_goals_and_make_canonical_response(Certainty::Yes))
911 } else {
912 Err(NoSolution.into())
913 }
914 }
915}
916
917#[inline(always)]
923fn trait_predicate_with_def_id<I: Interner>(
924 cx: I,
925 clause: ty::Binder<I, ty::TraitPredicate<I>>,
926 did: I::TraitId,
927) -> I::Clause {
928 clause
929 .map_bound(|c| TraitPredicate {
930 trait_ref: TraitRef::new_from_args(cx, did, c.trait_ref.args),
931 polarity: c.polarity,
932 })
933 .upcast(cx)
934}
935
936impl<D, I> EvalCtxt<'_, D>
937where
938 D: SolverDelegate<Interner = I>,
939 I: Interner,
940{
941 fn consider_builtin_dyn_upcast_candidates(
951 &mut self,
952 goal: Goal<I, (I::Ty, I::Ty)>,
953 a_data: I::BoundExistentialPredicates,
954 a_region: I::Region,
955 b_data: I::BoundExistentialPredicates,
956 b_region: I::Region,
957 ) -> Vec<Candidate<I>> {
958 let cx = self.cx();
959 let Goal { predicate: (a_ty, _b_ty), .. } = goal;
960
961 let mut responses = ::alloc::vec::Vec::new()vec![];
962 let b_principal_def_id = b_data.principal_def_id();
965 if a_data.principal_def_id() == b_principal_def_id || b_principal_def_id.is_none() {
966 responses.extend(self.consider_builtin_upcast_to_principal(
967 goal,
968 CandidateSource::BuiltinImpl(BuiltinImplSource::Misc),
969 a_data,
970 a_region,
971 b_data,
972 b_region,
973 a_data.principal(),
974 ));
975 } else if let Some(a_principal) = a_data.principal() {
976 for (idx, new_a_principal) in
977 elaborate::supertraits(self.cx(), a_principal.with_self_ty(cx, a_ty))
978 .enumerate()
979 .skip(1)
980 {
981 responses.extend(self.consider_builtin_upcast_to_principal(
982 goal,
983 CandidateSource::BuiltinImpl(BuiltinImplSource::TraitUpcasting(idx)),
984 a_data,
985 a_region,
986 b_data,
987 b_region,
988 Some(new_a_principal.map_bound(|trait_ref| {
989 ty::ExistentialTraitRef::erase_self_ty(cx, trait_ref)
990 })),
991 ));
992 }
993 }
994
995 responses
996 }
997
998 fn consider_builtin_unsize_to_dyn_candidate(
999 &mut self,
1000 goal: Goal<I, (I::Ty, I::Ty)>,
1001 b_data: I::BoundExistentialPredicates,
1002 b_region: I::Region,
1003 ) -> Result<Candidate<I>, NoSolutionOrRerunNonErased> {
1004 let cx = self.cx();
1005 let Goal { predicate: (a_ty, _), .. } = goal;
1006
1007 if b_data.principal_def_id().is_some_and(|def_id| !cx.trait_is_dyn_compatible(def_id)) {
1009 return Err(NoSolution.into());
1010 }
1011
1012 self.probe_builtin_trait_candidate(BuiltinImplSource::Misc).enter(|ecx| {
1013 ecx.add_goals(
1016 GoalSource::ImplWhereBound,
1017 b_data.iter().map(|pred| goal.with(cx, pred.with_self_ty(cx, a_ty))),
1018 );
1019
1020 ecx.add_goal(
1022 GoalSource::ImplWhereBound,
1023 goal.with(
1024 cx,
1025 ty::TraitRef::new(
1026 cx,
1027 cx.require_trait_lang_item(SolverTraitLangItem::Sized),
1028 [a_ty],
1029 ),
1030 ),
1031 );
1032
1033 ecx.add_goal(GoalSource::Misc, goal.with(cx, ty::OutlivesPredicate(a_ty, b_region)));
1035 ecx.evaluate_added_goals_and_make_canonical_response(Certainty::Yes)
1036 })
1037 }
1038
1039 fn consider_builtin_upcast_to_principal(
1040 &mut self,
1041 goal: Goal<I, (I::Ty, I::Ty)>,
1042 source: CandidateSource<I>,
1043 a_data: I::BoundExistentialPredicates,
1044 a_region: I::Region,
1045 b_data: I::BoundExistentialPredicates,
1046 b_region: I::Region,
1047 upcast_principal: Option<ty::Binder<I, ty::ExistentialTraitRef<I>>>,
1048 ) -> Result<Candidate<I>, NoSolutionOrRerunNonErased> {
1049 let param_env = goal.param_env;
1050
1051 let a_auto_traits: IndexSet<I::TraitId> = a_data
1055 .auto_traits()
1056 .into_iter()
1057 .chain(a_data.principal_def_id().into_iter().flat_map(|principal_def_id| {
1058 elaborate::supertrait_def_ids(self.cx(), principal_def_id)
1059 .filter(|def_id| self.cx().trait_is_auto(*def_id))
1060 }))
1061 .collect();
1062
1063 let projection_may_match =
1068 |ecx: &mut EvalCtxt<'_, D>,
1069 source_projection: ty::Binder<I, ty::ExistentialProjection<I>>,
1070 target_projection: ty::Binder<I, ty::ExistentialProjection<I>>| {
1071 source_projection.item_def_id() == target_projection.item_def_id()
1072 && ecx
1073 .probe(|_| ProbeKind::ProjectionCompatibility)
1074 .enter(|ecx| {
1075 ecx.enter_forall_with_assumptions(
1076 target_projection,
1077 param_env,
1078 |ecx, target_projection| {
1079 let source_projection =
1080 ecx.instantiate_binder_with_infer(source_projection);
1081 ecx.eq(param_env, source_projection, target_projection)?;
1082 ecx.try_evaluate_added_goals()
1083 },
1084 )
1085 .map_err(Into::into)
1086 })
1087 .is_ok()
1088 };
1089
1090 self.probe_trait_candidate(source).enter(|ecx| {
1091 for bound in b_data.iter() {
1092 match bound.skip_binder() {
1093 ty::ExistentialPredicate::Trait(target_principal) => {
1096 let source_principal = upcast_principal.unwrap();
1097 let target_principal = bound.rebind(target_principal);
1098 ecx.enter_forall_with_assumptions(
1099 target_principal,
1100 param_env,
1101 |ecx, target_principal| {
1102 let source_principal =
1103 ecx.instantiate_binder_with_infer(source_principal);
1104 ecx.eq(param_env, source_principal, target_principal)?;
1105 ecx.try_evaluate_added_goals()
1106 },
1107 )?;
1108 }
1109 ty::ExistentialPredicate::Projection(target_projection) => {
1115 let target_projection = bound.rebind(target_projection);
1116 let mut matching_projections =
1117 a_data.projection_bounds().into_iter().filter(|source_projection| {
1118 projection_may_match(ecx, *source_projection, target_projection)
1119 });
1120 let Some(source_projection) = matching_projections.next() else {
1121 return Err(NoSolution.into());
1122 };
1123 if matching_projections.next().is_some() {
1124 return ecx
1125 .evaluate_added_goals_and_make_canonical_response(
1126 Certainty::AMBIGUOUS,
1127 )
1128 .map_err(Into::into);
1129 }
1130 ecx.enter_forall_with_assumptions(
1131 target_projection,
1132 param_env,
1133 |ecx, target_projection| {
1134 let source_projection =
1135 ecx.instantiate_binder_with_infer(source_projection);
1136 ecx.eq(param_env, source_projection, target_projection)?;
1137 ecx.try_evaluate_added_goals()
1138 },
1139 )?;
1140 }
1141 ty::ExistentialPredicate::AutoTrait(def_id) => {
1143 if !a_auto_traits.contains(&def_id) {
1144 return Err(NoSolution.into());
1145 }
1146 }
1147 }
1148 }
1149
1150 ecx.add_goal(
1152 GoalSource::ImplWhereBound,
1153 Goal::new(ecx.cx(), param_env, ty::OutlivesPredicate(a_region, b_region)),
1154 );
1155
1156 ecx.evaluate_added_goals_and_make_canonical_response(Certainty::Yes).map_err(Into::into)
1157 })
1158 }
1159
1160 fn consider_builtin_array_unsize(
1169 &mut self,
1170 goal: Goal<I, (I::Ty, I::Ty)>,
1171 a_elem_ty: I::Ty,
1172 b_elem_ty: I::Ty,
1173 ) -> Result<Candidate<I>, NoSolutionOrRerunNonErased> {
1174 self.eq(goal.param_env, a_elem_ty, b_elem_ty)?;
1175 self.probe_builtin_trait_candidate(BuiltinImplSource::Misc)
1176 .enter(|ecx| ecx.evaluate_added_goals_and_make_canonical_response(Certainty::Yes))
1177 }
1178
1179 fn consider_builtin_struct_unsize(
1193 &mut self,
1194 goal: Goal<I, (I::Ty, I::Ty)>,
1195 def: I::AdtDef,
1196 a_args: I::GenericArgs,
1197 b_args: I::GenericArgs,
1198 ) -> Result<Candidate<I>, NoSolutionOrRerunNonErased> {
1199 let cx = self.cx();
1200 let Goal { predicate: (_a_ty, b_ty), .. } = goal;
1201
1202 let unsizing_params = cx.unsizing_params_for_adt(def.def_id());
1203 if unsizing_params.is_empty() {
1206 return Err(NoSolution.into());
1207 }
1208
1209 let tail_field_ty = def.struct_tail_ty(cx).unwrap();
1210
1211 let a_tail_ty = tail_field_ty.instantiate(cx, a_args).skip_norm_wip();
1212 let b_tail_ty = tail_field_ty.instantiate(cx, b_args).skip_norm_wip();
1213
1214 let new_a_args = cx.mk_args_from_iter(a_args.iter().enumerate().map(|(i, a)| {
1218 if unsizing_params.contains(i as u32) { b_args.get(i).unwrap() } else { a }
1219 }));
1220 let unsized_a_ty = Ty::new_adt(cx, def, new_a_args);
1221
1222 self.eq(goal.param_env, unsized_a_ty, b_ty)?;
1225 self.add_goal(
1226 GoalSource::ImplWhereBound,
1227 goal.with(
1228 cx,
1229 ty::TraitRef::new(
1230 cx,
1231 cx.require_trait_lang_item(SolverTraitLangItem::Unsize),
1232 [a_tail_ty, b_tail_ty],
1233 ),
1234 ),
1235 );
1236 self.probe_builtin_trait_candidate(BuiltinImplSource::Misc)
1237 .enter(|ecx| ecx.evaluate_added_goals_and_make_canonical_response(Certainty::Yes))
1238 }
1239
1240 fn disqualify_auto_trait_candidate_due_to_possible_impl(
1245 &mut self,
1246 goal: Goal<I, TraitPredicate<I>>,
1247 ) -> Option<Result<Candidate<I>, NoSolutionOrRerunNonErased>> {
1248 let self_ty = goal.predicate.self_ty();
1249 let check_impls = || {
1250 let mut disqualifying_impl = None;
1251 self.cx().for_each_relevant_impl(
1252 goal.predicate.def_id(),
1253 goal.predicate.self_ty(),
1254 |impl_def_id| {
1255 disqualifying_impl = Some(impl_def_id);
1256 },
1257 );
1258 if let Some(def_id) = disqualifying_impl {
1259 {
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/trait_goals.rs:1259",
"rustc_next_trait_solver::solve::trait_goals",
::tracing::Level::TRACE,
::tracing_core::__macro_support::Option::Some("compiler/rustc_next_trait_solver/src/solve/trait_goals.rs"),
::tracing_core::__macro_support::Option::Some(1259u32),
::tracing_core::__macro_support::Option::Some("rustc_next_trait_solver::solve::trait_goals"),
::tracing_core::field::FieldSet::new(&["message", "def_id",
"goal"], ::tracing_core::callsite::Identifier(&__CALLSITE)),
::tracing::metadata::Kind::EVENT)
};
::tracing::callsite::DefaultCallsite::new(&META)
};
let enabled =
::tracing::Level::TRACE <= ::tracing::level_filters::STATIC_MAX_LEVEL
&&
::tracing::Level::TRACE <=
::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!("disqualified auto-trait implementation")
as &dyn Value)),
(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&debug(&def_id) as
&dyn Value)),
(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&debug(&goal) as
&dyn Value))])
});
} else { ; }
};trace!(?def_id, ?goal, "disqualified auto-trait implementation");
1260 return Some(Err(NoSolution.into()));
1263 } else {
1264 None
1265 }
1266 };
1267
1268 match self_ty.kind() {
1269 ty::Infer(ty::IntVar(_) | ty::FloatVar(_)) => {
1275 Some(self.forced_ambiguity(MaybeInfo::AMBIGUOUS))
1276 }
1277
1278 ty::Foreign(..) if self.cx().is_default_trait(goal.predicate.def_id()) => check_impls(),
1281
1282 ty::Dynamic(..)
1285 | ty::Param(..)
1286 | ty::Foreign(..)
1287 | ty::Alias(ty::AliasTy {
1288 kind: ty::Projection { .. } | ty::Free { .. } | ty::Inherent { .. },
1289 ..
1290 })
1291 | ty::Placeholder(..) => Some(Err(NoSolution.into())),
1292
1293 ty::Infer(_) | ty::Bound(_, _) => {
::core::panicking::panic_fmt(format_args!("unexpected type `{0:?}`",
self_ty));
}panic!("unexpected type `{self_ty:?}`"),
1294
1295 ty::Coroutine(def_id, _)
1299 if self
1300 .cx()
1301 .is_trait_lang_item(goal.predicate.def_id(), SolverTraitLangItem::Unpin) =>
1302 {
1303 match self.cx().coroutine_movability(def_id) {
1304 Movability::Static => Some(Err(NoSolution.into())),
1305 Movability::Movable => Some(
1306 self.probe_builtin_trait_candidate(BuiltinImplSource::Misc).enter(|ecx| {
1307 ecx.evaluate_added_goals_and_make_canonical_response(Certainty::Yes)
1308 }),
1309 ),
1310 }
1311 }
1312
1313 ty::Alias(..) => None,
1318
1319 ty::Bool
1326 | ty::Char
1327 | ty::Int(_)
1328 | ty::Uint(_)
1329 | ty::Float(_)
1330 | ty::Str
1331 | ty::Array(_, _)
1332 | ty::Pat(_, _)
1333 | ty::Slice(_)
1334 | ty::RawPtr(_, _)
1335 | ty::Ref(_, _, _)
1336 | ty::FnDef(_, _)
1337 | ty::FnPtr(..)
1338 | ty::Closure(..)
1339 | ty::CoroutineClosure(..)
1340 | ty::Coroutine(_, _)
1341 | ty::CoroutineWitness(..)
1342 | ty::Never
1343 | ty::Tuple(_)
1344 | ty::Adt(_, _)
1345 | ty::UnsafeBinder(_) => check_impls(),
1346 ty::Error(_) => None,
1347 }
1348 }
1349
1350 fn probe_and_evaluate_goal_for_constituent_tys(
1355 &mut self,
1356 source: CandidateSource<I>,
1357 goal: Goal<I, TraitPredicate<I>>,
1358 constituent_tys: impl Fn(
1359 &EvalCtxt<'_, D>,
1360 I::Ty,
1361 ) -> Result<ty::Binder<I, Vec<I::Ty>>, NoSolution>,
1362 ) -> Result<Candidate<I>, NoSolutionOrRerunNonErased> {
1363 self.probe_trait_candidate(source).enter(|ecx| {
1364 let goals = ecx.enter_forall_with_assumptions(
1365 constituent_tys(ecx, goal.predicate.self_ty())?,
1366 goal.param_env,
1367 |ecx, tys| {
1368 tys.into_iter()
1369 .map(|ty| {
1370 goal.with(ecx.cx(), goal.predicate.with_replaced_self_ty(ecx.cx(), ty))
1371 })
1372 .collect::<Vec<_>>()
1373 },
1374 );
1375 ecx.add_goals(GoalSource::ImplWhereBound, goals);
1376 ecx.evaluate_added_goals_and_make_canonical_response(Certainty::Yes)
1377 })
1378 }
1379}
1380
1381#[derive(#[automatically_derived]
impl ::core::fmt::Debug for TraitGoalProvenVia {
#[inline]
fn fmt(&self, f: &mut ::core::fmt::Formatter) -> ::core::fmt::Result {
::core::fmt::Formatter::write_str(f,
match self {
TraitGoalProvenVia::Misc => "Misc",
TraitGoalProvenVia::ParamEnv => "ParamEnv",
TraitGoalProvenVia::AliasBound => "AliasBound",
})
}
}Debug, #[automatically_derived]
impl ::core::clone::Clone for TraitGoalProvenVia {
#[inline]
fn clone(&self) -> TraitGoalProvenVia { *self }
}Clone, #[automatically_derived]
impl ::core::marker::Copy for TraitGoalProvenVia { }Copy)]
1392pub(super) enum TraitGoalProvenVia {
1393 Misc,
1399 ParamEnv,
1400 AliasBound,
1401}
1402
1403impl<D, I> EvalCtxt<'_, D>
1404where
1405 D: SolverDelegate<Interner = I>,
1406 I: Interner,
1407{
1408 pub(super) fn unsound_prefer_builtin_dyn_impl(&mut self, candidates: &mut Vec<Candidate<I>>) {
1421 if self.typing_mode().is_coherence() {
1422 return;
1423 }
1424
1425 if candidates
1426 .iter()
1427 .find(|c| {
1428 #[allow(non_exhaustive_omitted_patterns)] match c.source {
CandidateSource::BuiltinImpl(BuiltinImplSource::Object(_)) => true,
_ => false,
}matches!(c.source, CandidateSource::BuiltinImpl(BuiltinImplSource::Object(_)))
1429 })
1430 .is_some_and(|c| has_only_region_constraints(c.result))
1431 {
1432 candidates.retain(|c| {
1433 if #[allow(non_exhaustive_omitted_patterns)] match c.source {
CandidateSource::Impl(_) => true,
_ => false,
}matches!(c.source, CandidateSource::Impl(_)) {
1434 {
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/trait_goals.rs:1434",
"rustc_next_trait_solver::solve::trait_goals",
::tracing::Level::DEBUG,
::tracing_core::__macro_support::Option::Some("compiler/rustc_next_trait_solver/src/solve/trait_goals.rs"),
::tracing_core::__macro_support::Option::Some(1434u32),
::tracing_core::__macro_support::Option::Some("rustc_next_trait_solver::solve::trait_goals"),
::tracing_core::field::FieldSet::new(&["message", "c"],
::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!("unsoundly dropping impl in favor of builtin dyn-candidate")
as &dyn Value)),
(&::tracing::__macro_support::Iterator::next(&mut iter).expect("FieldSet corrupted (this is a bug)"),
::tracing::__macro_support::Option::Some(&debug(&c) as
&dyn Value))])
});
} else { ; }
};debug!(?c, "unsoundly dropping impl in favor of builtin dyn-candidate");
1435 false
1436 } else {
1437 true
1438 }
1439 });
1440 }
1441 }
1442
1443 x;#[instrument(level = "debug", skip(self), ret)]
1444 pub(super) fn merge_trait_candidates(
1445 &mut self,
1446 candidate_preference_mode: CandidatePreferenceMode,
1447 mut candidates: Vec<Candidate<I>>,
1448 failed_candidate_info: FailedCandidateInfo,
1449 ) -> Result<(CanonicalResponse<I>, Option<TraitGoalProvenVia>), NoSolution> {
1450 if self.typing_mode().is_coherence() {
1451 return if let Some((response, _)) = self.try_merge_candidates(&candidates) {
1452 Ok((response, Some(TraitGoalProvenVia::Misc)))
1453 } else {
1454 self.flounder(&candidates).map(|r| (r, None))
1455 };
1456 }
1457
1458 let mut trivial_builtin_impls = candidates.iter().filter(|c| {
1463 matches!(c.source, CandidateSource::BuiltinImpl(BuiltinImplSource::Trivial))
1464 });
1465 if let Some(candidate) = trivial_builtin_impls.next() {
1466 assert!(trivial_builtin_impls.next().is_none());
1469 return Ok((candidate.result, Some(TraitGoalProvenVia::Misc)));
1470 }
1471
1472 if matches!(candidate_preference_mode, CandidatePreferenceMode::Marker)
1475 && candidates.iter().any(|c| {
1476 matches!(c.source, CandidateSource::AliasBound(AliasBoundKind::SelfBounds))
1477 })
1478 {
1479 let alias_bounds: Vec<_> = candidates
1480 .extract_if(.., |c| matches!(c.source, CandidateSource::AliasBound(..)))
1481 .collect();
1482 return if let Some((response, _)) = self.try_merge_candidates(&alias_bounds) {
1483 Ok((response, Some(TraitGoalProvenVia::AliasBound)))
1484 } else {
1485 Ok((self.bail_with_ambiguity(&alias_bounds), None))
1486 };
1487 }
1488
1489 let has_non_global_where_bounds = candidates
1492 .iter()
1493 .any(|c| matches!(c.source, CandidateSource::ParamEnv(ParamEnvSource::NonGlobal)));
1494 if has_non_global_where_bounds {
1495 let where_bounds: Vec<_> = candidates
1496 .extract_if(.., |c| matches!(c.source, CandidateSource::ParamEnv(_)))
1497 .collect();
1498 let Some((response, info)) = self.try_merge_candidates(&where_bounds) else {
1499 return Ok((self.bail_with_ambiguity(&where_bounds), None));
1500 };
1501 match info {
1502 MergeCandidateInfo::AlwaysApplicable(i) => {
1518 for (j, c) in where_bounds.into_iter().enumerate() {
1519 if i != j {
1520 self.ignore_candidate_head_usages(c.head_usages)
1521 }
1522 }
1523 self.ignore_candidate_head_usages(failed_candidate_info.param_env_head_usages);
1527 }
1528 MergeCandidateInfo::EqualResponse => {}
1529 }
1530 return Ok((response, Some(TraitGoalProvenVia::ParamEnv)));
1531 }
1532
1533 if candidates.iter().any(|c| matches!(c.source, CandidateSource::AliasBound(_))) {
1535 let alias_bounds: Vec<_> = candidates
1536 .extract_if(.., |c| matches!(c.source, CandidateSource::AliasBound(_)))
1537 .collect();
1538 return if let Some((response, _)) = self.try_merge_candidates(&alias_bounds) {
1539 Ok((response, Some(TraitGoalProvenVia::AliasBound)))
1540 } else {
1541 Ok((self.bail_with_ambiguity(&alias_bounds), None))
1542 };
1543 }
1544
1545 self.filter_specialized_impls(AllowInferenceConstraints::No, &mut candidates);
1546 self.unsound_prefer_builtin_dyn_impl(&mut candidates);
1547
1548 let proven_via = if candidates
1553 .iter()
1554 .all(|c| matches!(c.source, CandidateSource::ParamEnv(ParamEnvSource::Global)))
1555 {
1556 TraitGoalProvenVia::ParamEnv
1557 } else {
1558 candidates
1559 .retain(|c| !matches!(c.source, CandidateSource::ParamEnv(ParamEnvSource::Global)));
1560 TraitGoalProvenVia::Misc
1561 };
1562
1563 if let Some((response, _)) = self.try_merge_candidates(&candidates) {
1564 Ok((response, Some(proven_via)))
1565 } else {
1566 self.flounder(&candidates).map(|r| (r, None))
1567 }
1568 }
1569
1570 #[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("compute_trait_goal",
"rustc_next_trait_solver::solve::trait_goals",
::tracing::Level::TRACE,
::tracing_core::__macro_support::Option::Some("compiler/rustc_next_trait_solver/src/solve/trait_goals.rs"),
::tracing_core::__macro_support::Option::Some(1570u32),
::tracing_core::__macro_support::Option::Some("rustc_next_trait_solver::solve::trait_goals"),
::tracing_core::field::FieldSet::new(&["goal"],
::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,
&{
#[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))])
})
} 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<(CanonicalResponse<I>, Option<TraitGoalProvenVia>),
NoSolutionOrRerunNonErased> = loop {};
return __tracing_attr_fake_return;
}
{
let (candidates, failed_candidate_info) =
self.assemble_and_evaluate_candidates(goal,
AssembleCandidatesFrom::All)?;
let candidate_preference_mode =
CandidatePreferenceMode::compute(self.cx(),
goal.predicate.def_id());
self.merge_trait_candidates(candidate_preference_mode, candidates,
failed_candidate_info).map_err(Into::into)
}
}
}#[instrument(level = "trace", skip(self))]
1571 pub(super) fn compute_trait_goal(
1572 &mut self,
1573 goal: Goal<I, TraitPredicate<I>>,
1574 ) -> Result<(CanonicalResponse<I>, Option<TraitGoalProvenVia>), NoSolutionOrRerunNonErased>
1575 {
1576 let (candidates, failed_candidate_info) =
1577 self.assemble_and_evaluate_candidates(goal, AssembleCandidatesFrom::All)?;
1578 let candidate_preference_mode =
1579 CandidatePreferenceMode::compute(self.cx(), goal.predicate.def_id());
1580 self.merge_trait_candidates(candidate_preference_mode, candidates, failed_candidate_info)
1581 .map_err(Into::into)
1582 }
1583
1584 fn try_stall_coroutine(
1585 &mut self,
1586 self_ty: I::Ty,
1587 ) -> Option<Result<Candidate<I>, NoSolutionOrRerunNonErased>> {
1588 if let ty::Coroutine(def_id, _) = self_ty.kind() {
1589 match self.typing_mode() {
1590 TypingMode::Analysis {
1591 defining_opaque_types_and_generators: stalled_generators,
1592 } => {
1593 if def_id.as_local().is_some_and(|def_id| stalled_generators.contains(&def_id))
1594 {
1595 return Some(self.forced_ambiguity(MaybeInfo {
1596 cause: MaybeCause::Ambiguity,
1597 opaque_types_jank: OpaqueTypesJank::AllGood,
1598 stalled_on_coroutines: StalledOnCoroutines::Yes,
1599 }));
1600 }
1601 }
1602 TypingMode::ErasedNotCoherence(MayBeErased) => {
1603 return Some(
1605 match self.opaque_accesses.rerun_always(RerunReason::TryStallCoroutine) {
1606 Err(e) => Err(e.into()),
1607 },
1608 );
1609 }
1610 TypingMode::Coherence
1611 | TypingMode::PostAnalysis
1612 | TypingMode::Borrowck { defining_opaque_types: _ }
1613 | TypingMode::PostBorrowckAnalysis { defined_opaque_types: _ } => {}
1614 }
1615 }
1616
1617 None
1618 }
1619}