|
8 | 8 | //! https://rustc-dev-guide.rust-lang.org/traits/resolution.html#confirmation
|
9 | 9 | use rustc_data_structures::stack::ensure_sufficient_stack;
|
10 | 10 | use rustc_hir::lang_items::LangItem;
|
| 11 | +use rustc_hir::Constness; |
11 | 12 | use rustc_index::bit_set::GrowableBitSet;
|
12 | 13 | use rustc_infer::infer::InferOk;
|
13 | 14 | use rustc_infer::infer::LateBoundRegionConversionTime::HigherRankedType;
|
@@ -51,6 +52,38 @@ impl<'cx, 'tcx> SelectionContext<'cx, 'tcx> {
|
51 | 52 | obligation: &TraitObligation<'tcx>,
|
52 | 53 | candidate: SelectionCandidate<'tcx>,
|
53 | 54 | ) -> Result<Selection<'tcx>, SelectionError<'tcx>> {
|
| 55 | + let mut obligation = obligation; |
| 56 | + let new_obligation; |
| 57 | + |
| 58 | + // HACK(const_trait_impl): the surrounding environment is remapped to a non-const context |
| 59 | + // because nested obligations might be actually `~const` then (incorrectly) requiring |
| 60 | + // const impls. for example: |
| 61 | + // ``` |
| 62 | + // pub trait Super {} |
| 63 | + // pub trait Sub: Super {} |
| 64 | + // |
| 65 | + // impl<A> const Super for &A where A: ~const Super {} |
| 66 | + // impl<A> const Sub for &A where A: ~const Sub {} |
| 67 | + // ``` |
| 68 | + // |
| 69 | + // The procedure to check the code above without the remapping code is as follows: |
| 70 | + // ``` |
| 71 | + // CheckWf(impl const Sub for &A where A: ~const Sub) // <- const env |
| 72 | + // CheckPredicate(&A: Super) |
| 73 | + // CheckPredicate(A: ~const Super) // <- still const env, failure |
| 74 | + // ``` |
| 75 | + if obligation.param_env.constness() == Constness::Const |
| 76 | + && obligation.predicate.skip_binder().constness == ty::BoundConstness::NotConst |
| 77 | + { |
| 78 | + new_obligation = TraitObligation { |
| 79 | + cause: obligation.cause.clone(), |
| 80 | + param_env: obligation.param_env.without_const(), |
| 81 | + ..*obligation |
| 82 | + }; |
| 83 | + |
| 84 | + obligation = &new_obligation; |
| 85 | + } |
| 86 | + |
54 | 87 | match candidate {
|
55 | 88 | BuiltinCandidate { has_nested } => {
|
56 | 89 | let data = self.confirm_builtin_candidate(obligation, has_nested);
|
|
0 commit comments