Skip to content

Commit d535e47

Browse files
Check that impl method args are WF given implied-bounds assumptions from trait
1 parent d137783 commit d535e47

File tree

3 files changed

+71
-2
lines changed

3 files changed

+71
-2
lines changed

compiler/rustc_hir_analysis/src/check/compare_method.rs

+21-2
Original file line numberDiff line numberDiff line change
@@ -251,14 +251,14 @@ fn compare_predicate_entailment<'tcx>(
251251

252252
let mut wf_tys = FxIndexSet::default();
253253

254-
let impl_sig = infcx.replace_bound_vars_with_fresh_vars(
254+
let unnormalized_impl_sig = infcx.replace_bound_vars_with_fresh_vars(
255255
impl_m_span,
256256
infer::HigherRankedType,
257257
tcx.fn_sig(impl_m.def_id),
258258
);
259259

260260
let norm_cause = ObligationCause::misc(impl_m_span, impl_m_hir_id);
261-
let impl_sig = ocx.normalize(&norm_cause, param_env, impl_sig);
261+
let impl_sig = ocx.normalize(&norm_cause, param_env, unnormalized_impl_sig);
262262
let impl_fty = tcx.mk_fn_ptr(ty::Binder::dummy(impl_sig));
263263
debug!("compare_impl_method: impl_fty={:?}", impl_fty);
264264

@@ -300,6 +300,25 @@ fn compare_predicate_entailment<'tcx>(
300300
return Err(emitted);
301301
}
302302

303+
// Finally, we need to check that the impl's args are well-formed given
304+
// the hybrid param-env (impl + trait method where-clauses).
305+
for (unnormalized_arg, arg) in
306+
std::iter::zip(unnormalized_impl_sig.inputs_and_output, impl_sig.inputs_and_output)
307+
{
308+
ocx.register_obligation(traits::Obligation::new(
309+
tcx,
310+
cause.clone(),
311+
param_env,
312+
ty::Binder::dummy(ty::PredicateKind::WellFormed(unnormalized_arg.into())),
313+
));
314+
ocx.register_obligation(traits::Obligation::new(
315+
tcx,
316+
cause.clone(),
317+
param_env,
318+
ty::Binder::dummy(ty::PredicateKind::WellFormed(arg.into())),
319+
));
320+
}
321+
303322
// Check that all obligations are satisfied by the implementation's
304323
// version.
305324
let errors = ocx.select_all_or_error();
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,18 @@
1+
use std::cell::RefCell;
2+
3+
pub struct MessageListeners<'a> {
4+
listeners: RefCell<Vec<Box<dyn FnMut(()) + 'a>>>,
5+
}
6+
7+
pub trait MessageListenersInterface {
8+
fn listeners<'c>(&'c self) -> &'c MessageListeners<'c>;
9+
}
10+
11+
impl<'a> MessageListenersInterface for MessageListeners<'a> {
12+
fn listeners<'b>(&'b self) -> &'a MessageListeners<'b> {
13+
//~^ ERROR cannot infer an appropriate lifetime for lifetime parameter 'b in generic type due to conflicting requirement
14+
self
15+
}
16+
}
17+
18+
fn main() {}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,32 @@
1+
error[E0495]: cannot infer an appropriate lifetime for lifetime parameter 'b in generic type due to conflicting requirements
2+
--> $DIR/impl-implied-bounds-compatibility.rs:12:5
3+
|
4+
LL | fn listeners<'b>(&'b self) -> &'a MessageListeners<'b> {
5+
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
6+
|
7+
note: first, the lifetime cannot outlive the lifetime `'c` as defined here...
8+
--> $DIR/impl-implied-bounds-compatibility.rs:12:5
9+
|
10+
LL | fn listeners<'b>(&'b self) -> &'a MessageListeners<'b> {
11+
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
12+
note: ...so that the method type is compatible with trait
13+
--> $DIR/impl-implied-bounds-compatibility.rs:12:5
14+
|
15+
LL | fn listeners<'b>(&'b self) -> &'a MessageListeners<'b> {
16+
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
17+
= note: expected `fn(&'c MessageListeners<'a>) -> &'c MessageListeners<'c>`
18+
found `fn(&MessageListeners<'a>) -> &'a MessageListeners<'_>`
19+
note: but, the lifetime must be valid for the lifetime `'a` as defined here...
20+
--> $DIR/impl-implied-bounds-compatibility.rs:11:6
21+
|
22+
LL | impl<'a> MessageListenersInterface for MessageListeners<'a> {
23+
| ^^
24+
note: ...so that the reference type `&'a MessageListeners<'_>` does not outlive the data it points at
25+
--> $DIR/impl-implied-bounds-compatibility.rs:12:5
26+
|
27+
LL | fn listeners<'b>(&'b self) -> &'a MessageListeners<'b> {
28+
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
29+
30+
error: aborting due to previous error
31+
32+
For more information about this error, try `rustc --explain E0495`.

0 commit comments

Comments
 (0)