Skip to content

Commit c06e611

Browse files
do not allow inference in pred_known_to_hold_modulo_regions
1 parent dbbb424 commit c06e611

File tree

4 files changed

+38
-23
lines changed

4 files changed

+38
-23
lines changed

compiler/rustc_trait_selection/src/traits/mod.rs

+21-20
Original file line numberDiff line numberDiff line change
@@ -142,35 +142,36 @@ pub fn type_known_to_meet_bound_modulo_regions<'tcx>(
142142
fn pred_known_to_hold_modulo_regions<'tcx>(
143143
infcx: &InferCtxt<'tcx>,
144144
param_env: ty::ParamEnv<'tcx>,
145-
pred: impl ToPredicate<'tcx> + TypeVisitable<TyCtxt<'tcx>>,
145+
pred: impl ToPredicate<'tcx>,
146146
) -> bool {
147-
let has_non_region_infer = pred.has_non_region_infer();
148147
let obligation = Obligation::new(infcx.tcx, ObligationCause::dummy(), param_env, pred);
149148

150149
let result = infcx.evaluate_obligation_no_overflow(&obligation);
151150
debug!(?result);
152151

153-
if result.must_apply_modulo_regions() && !has_non_region_infer {
152+
if result.must_apply_modulo_regions() {
154153
true
155154
} else if result.may_apply() {
156-
// Because of inference "guessing", selection can sometimes claim
157-
// to succeed while the success requires a guess. To ensure
158-
// this function's result remains infallible, we must confirm
159-
// that guess. While imperfect, I believe this is sound.
160-
161-
// The handling of regions in this area of the code is terrible,
162-
// see issue #29149. We should be able to improve on this with
163-
// NLL.
164-
let ocx = ObligationCtxt::new(infcx);
165-
ocx.register_obligation(obligation);
166-
let errors = ocx.select_all_or_error();
167-
match errors.as_slice() {
168-
[] => true,
169-
errors => {
170-
debug!(?errors);
171-
false
155+
// Sometimes obligations are ambiguous because the recursive evaluator
156+
// is not smart enough, so we fall back to fulfillment when we're not certain
157+
// that an obligation holds or not. Even still, we must make sure that
158+
// the we do no inference in the process of checking this obligation.
159+
let goal = infcx.resolve_vars_if_possible((obligation.predicate, obligation.param_env));
160+
infcx.probe(|_| {
161+
let ocx = ObligationCtxt::new_in_snapshot(infcx);
162+
ocx.register_obligation(obligation);
163+
164+
let errors = ocx.select_all_or_error();
165+
match errors.as_slice() {
166+
// Only known to hold if we did no inference.
167+
[] => infcx.shallow_resolve(goal) == goal,
168+
169+
errors => {
170+
debug!(?errors);
171+
false
172+
}
172173
}
173-
}
174+
})
174175
} else {
175176
false
176177
}

compiler/rustc_trait_selection/src/traits/select/mod.rs

+2-1
Original file line numberDiff line numberDiff line change
@@ -537,7 +537,8 @@ impl<'cx, 'tcx> SelectionContext<'cx, 'tcx> {
537537
obligation: &PredicateObligation<'tcx>,
538538
) -> Result<EvaluationResult, OverflowError> {
539539
self.evaluation_probe(|this| {
540-
let goal = infcx.resolve_vars_if_possible((obligation.predicate, obligation.param_env));
540+
let goal =
541+
this.infcx.resolve_vars_if_possible((obligation.predicate, obligation.param_env));
541542
let mut result = if this.tcx().trait_solver_next() {
542543
this.evaluate_predicates_recursively_in_new_solver([obligation.clone()])?
543544
} else {

tests/ui/traits/copy-guessing.rs

+1-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,3 @@
1-
// run-pass
2-
31
#![allow(dead_code)]
42
#![allow(drop_copy)]
53

@@ -20,6 +18,7 @@ fn assert_impls_fn<R,T: Fn()->R>(_: &T){}
2018

2119
fn main() {
2220
let n = None;
21+
//~^ ERROR type annotations needed for `Option<T>`
2322
let e = S(&n);
2423
let f = || {
2524
// S being copy is critical for this to work

tests/ui/traits/copy-guessing.stderr

+14
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,14 @@
1+
error[E0282]: type annotations needed for `Option<T>`
2+
--> $DIR/copy-guessing.rs:20:9
3+
|
4+
LL | let n = None;
5+
| ^
6+
|
7+
help: consider giving `n` an explicit type, where the type for type parameter `T` is specified
8+
|
9+
LL | let n: Option<T> = None;
10+
| +++++++++++
11+
12+
error: aborting due to previous error
13+
14+
For more information about this error, try `rustc --explain E0282`.

0 commit comments

Comments
 (0)