|
1 | 1 | //! Bounds are restrictions applied to some types after they've been converted into the
|
2 | 2 | //! `ty` form from the HIR.
|
3 | 3 |
|
4 |
| -use rustc_hir::LangItem; |
| 4 | +use rustc_hir::{def::DefKind, LangItem}; |
5 | 5 | use rustc_middle::ty::{self, ToPredicate, Ty, TyCtxt};
|
6 | 6 | use rustc_span::{def_id::DefId, Span};
|
7 | 7 |
|
@@ -54,15 +54,54 @@ impl<'tcx> Bounds<'tcx> {
|
54 | 54 | .to_predicate(tcx),
|
55 | 55 | span,
|
56 | 56 | ));
|
57 |
| - // For `T: ~const Tr` or `T: const Tr`, we need to add an additional bound on the |
58 |
| - // associated type of `<T as Tr>` and make sure that the effect is compatible. |
59 |
| - if let Some(compat_val) = match constness { |
| 57 | + if let Some(compat_val) = match (tcx.def_kind(defining_def_id), constness) { |
60 | 58 | // TODO: do we need `T: const Trait` anymore?
|
61 |
| - ty::BoundConstness::Const => Some(tcx.consts.false_), |
62 |
| - ty::BoundConstness::ConstIfConst => { |
| 59 | + (_, ty::BoundConstness::Const) => Some(tcx.consts.false_), |
| 60 | + // body owners that can have trait bounds |
| 61 | + (DefKind::Const | DefKind::Fn | DefKind::AssocFn, ty::BoundConstness::ConstIfConst) => { |
63 | 62 | Some(tcx.expected_host_effect_param_for_body(defining_def_id))
|
64 | 63 | }
|
65 |
| - ty::BoundConstness::NotConst => None, |
| 64 | + |
| 65 | + (_, ty::BoundConstness::NotConst) => None, |
| 66 | + |
| 67 | + // if the defining_def_id is a trait, we wire it differently than others by equating the effects. |
| 68 | + ( |
| 69 | + kind @ (DefKind::Trait | DefKind::Impl { of_trait: true }), |
| 70 | + ty::BoundConstness::ConstIfConst, |
| 71 | + ) => { |
| 72 | + let trait_we_are_in = if let DefKind::Trait = kind { |
| 73 | + ty::TraitRef::identity(tcx, defining_def_id) |
| 74 | + } else { |
| 75 | + tcx.impl_trait_ref(defining_def_id).unwrap().instantiate_identity() |
| 76 | + }; |
| 77 | + // create a new projection type `<T as TraitForBound>::Effects` |
| 78 | + let assoc = tcx.associated_type_for_effects(trait_ref.def_id()).unwrap(); |
| 79 | + let self_ty = Ty::new_projection(tcx, assoc, trait_ref.skip_binder().args); |
| 80 | + // we might have `~const Tr` where `Tr` isn't a `#[const_trait]`. |
| 81 | + let Some(assoc_def) = tcx.associated_type_for_effects(trait_we_are_in.def_id) |
| 82 | + else { |
| 83 | + tcx.dcx().span_delayed_bug( |
| 84 | + span, |
| 85 | + "`~const` bound trait has no effect param yet no errors encountered?", |
| 86 | + ); |
| 87 | + return; |
| 88 | + }; |
| 89 | + let fx_ty_trait_we_are_in = |
| 90 | + Ty::new_projection(tcx, assoc_def, trait_we_are_in.args); |
| 91 | + // make `<T as TraitForBound>::Effects: EffectsEq<<Self as TraitWeAreIn>::Effects>` |
| 92 | + let new_trait_ref = ty::TraitRef::new( |
| 93 | + tcx, |
| 94 | + tcx.require_lang_item(LangItem::EffectsEq, Some(span)), |
| 95 | + [self_ty, fx_ty_trait_we_are_in], |
| 96 | + ); |
| 97 | + self.clauses.push((trait_ref.rebind(new_trait_ref).to_predicate(tcx), span)); |
| 98 | + return; |
| 99 | + } |
| 100 | + // probably illegal in this position. |
| 101 | + (_, ty::BoundConstness::ConstIfConst) => { |
| 102 | + tcx.dcx().span_delayed_bug(span, "invalid `~const` encountered"); |
| 103 | + return; |
| 104 | + } |
66 | 105 | } {
|
67 | 106 | // create a new projection type `<T as Tr>::Effects`
|
68 | 107 | let assoc = tcx.associated_type_for_effects(trait_ref.def_id()).unwrap();
|
|
0 commit comments