Skip to content

Commit 059bbf7

Browse files
authored
Rollup merge of #103008 - aliemjay:opaque-parent-substs, r=oli-obk
replace ReErased with fresh region vars in opaque types See inline comments. Prior art #102943. cc ``@compiler-errors`` ``@oli-obk`` Fixes #100267 Fixes #101940 Fixes #102649 Fixes #102510
2 parents 9692d98 + d2d3d94 commit 059bbf7

File tree

4 files changed

+209
-5
lines changed

4 files changed

+209
-5
lines changed

compiler/rustc_hir_analysis/src/check/check.rs

+15-5
Original file line numberDiff line numberDiff line change
@@ -732,8 +732,6 @@ fn check_opaque_meets_bounds<'tcx>(
732732
span: Span,
733733
origin: &hir::OpaqueTyOrigin,
734734
) {
735-
let hidden_type = tcx.bound_type_of(def_id.to_def_id()).subst(tcx, substs);
736-
737735
let hir_id = tcx.hir().local_def_id_to_hir_id(def_id);
738736
let defining_use_anchor = match *origin {
739737
hir::OpaqueTyOrigin::FnReturn(did) | hir::OpaqueTyOrigin::AsyncFn(did) => did,
@@ -748,14 +746,26 @@ fn check_opaque_meets_bounds<'tcx>(
748746
let ocx = ObligationCtxt::new(&infcx);
749747
let opaque_ty = tcx.mk_opaque(def_id.to_def_id(), substs);
750748

749+
// `ReErased` regions appear in the "parent_substs" of closures/generators.
750+
// We're ignoring them here and replacing them with fresh region variables.
751+
// See tests in ui/type-alias-impl-trait/closure_{parent_substs,wf_outlives}.rs.
752+
//
753+
// FIXME: Consider wrapping the hidden type in an existential `Binder` and instantiating it
754+
// here rather than using ReErased.
755+
let hidden_ty = tcx.bound_type_of(def_id.to_def_id()).subst(tcx, substs);
756+
let hidden_ty = tcx.fold_regions(hidden_ty, |re, _dbi| match re.kind() {
757+
ty::ReErased => infcx.next_region_var(RegionVariableOrigin::MiscVariable(span)),
758+
_ => re,
759+
});
760+
751761
let misc_cause = traits::ObligationCause::misc(span, hir_id);
752762

753-
match infcx.at(&misc_cause, param_env).eq(opaque_ty, hidden_type) {
763+
match infcx.at(&misc_cause, param_env).eq(opaque_ty, hidden_ty) {
754764
Ok(infer_ok) => ocx.register_infer_ok_obligations(infer_ok),
755765
Err(ty_err) => {
756766
tcx.sess.delay_span_bug(
757767
span,
758-
&format!("could not unify `{hidden_type}` with revealed type:\n{ty_err}"),
768+
&format!("could not unify `{hidden_ty}` with revealed type:\n{ty_err}"),
759769
);
760770
}
761771
}
@@ -764,7 +774,7 @@ fn check_opaque_meets_bounds<'tcx>(
764774
// Defining use functions may have more bounds than the opaque type, which is ok, as long as the
765775
// hidden type is well formed even without those bounds.
766776
let predicate =
767-
ty::Binder::dummy(ty::PredicateKind::WellFormed(hidden_type.into())).to_predicate(tcx);
777+
ty::Binder::dummy(ty::PredicateKind::WellFormed(hidden_ty.into())).to_predicate(tcx);
768778
ocx.register_obligation(Obligation::new(misc_cause, param_env, predicate));
769779

770780
// Check that all obligations are satisfied by the implementation's
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,65 @@
1+
// When WF checking the hidden type in the ParamEnv of the opaque type,
2+
// one complication arises when the hidden type is a closure/generator:
3+
// the "parent_substs" of the type may reference lifetime parameters
4+
// not present in the opaque type.
5+
// These region parameters are not really useful in this check.
6+
// So here we ignore them and replace them with fresh region variables.
7+
8+
// check-pass
9+
10+
#![feature(type_alias_impl_trait)]
11+
#![allow(dead_code)]
12+
13+
// Basic test
14+
mod test1 {
15+
// Hidden type = Closure['_#0r]
16+
type Opaque = impl Sized;
17+
18+
fn define<'a: 'a>() -> Opaque {
19+
|| {}
20+
}
21+
}
22+
23+
// the region vars cannot both be equal to `'static` or `'empty`
24+
mod test2 {
25+
trait Trait {}
26+
27+
// Hidden type = Closure['a, '_#0r, '_#1r]
28+
// Constraints = [('_#0r: 'a), ('a: '_#1r)]
29+
type Opaque<'a>
30+
where
31+
&'a (): Trait,
32+
= impl Sized + 'a;
33+
34+
fn define<'a, 'x, 'y>() -> Opaque<'a>
35+
where
36+
&'a (): Trait,
37+
'x: 'a,
38+
'a: 'y,
39+
{
40+
|| {}
41+
}
42+
}
43+
44+
// the region var cannot be equal to `'a` or `'b`
45+
mod test3 {
46+
trait Trait {}
47+
48+
// Hidden type = Closure['a, 'b, '_#0r]
49+
// Constraints = [('_#0r: 'a), ('_#0r: 'b)]
50+
type Opaque<'a, 'b>
51+
where
52+
(&'a (), &'b ()): Trait,
53+
= impl Sized + 'a + 'b;
54+
55+
fn define<'a, 'b, 'x>() -> Opaque<'a, 'b>
56+
where
57+
(&'a (), &'b ()): Trait,
58+
'x: 'a,
59+
'x: 'b,
60+
{
61+
|| {}
62+
}
63+
}
64+
65+
fn main() {}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,65 @@
1+
// If the hidden type is a closure, we require the "outlives" bounds that appear on the
2+
// defining site to also appear on the opaque type.
3+
//
4+
// It's not clear if this is the desired behavior but at least
5+
// it's consistent and has no back-compat risk.
6+
7+
// check-fail
8+
9+
#![feature(type_alias_impl_trait)]
10+
#![allow(dead_code)]
11+
12+
// requires `'a: 'b` bound
13+
mod test1 {
14+
type Opaque<'a, 'b> = impl Sized + 'a + 'b;
15+
//~^ ERROR lifetime bound not satisfied
16+
17+
fn define<'a, 'b>() -> Opaque<'a, 'b>
18+
where
19+
'a: 'b,
20+
{
21+
|| {}
22+
}
23+
}
24+
25+
// Same as the above but through indirection `'x`
26+
mod test2 {
27+
type Opaque<'a, 'b> = impl Sized + 'a + 'b;
28+
//~^ ERROR cannot infer an appropriate lifetime
29+
30+
fn define<'a, 'b, 'x>() -> Opaque<'a, 'b>
31+
where
32+
'a: 'x,
33+
'x: 'b,
34+
{
35+
|| {}
36+
}
37+
}
38+
39+
// fixed version of the above
40+
mod test2_fixed {
41+
type Opaque<'a: 'b, 'b> = impl Sized + 'a + 'b;
42+
43+
fn define<'a, 'b, 'x>() -> Opaque<'a, 'b>
44+
where
45+
'a: 'x,
46+
'x: 'b,
47+
{
48+
|| {}
49+
}
50+
}
51+
52+
// requires `T: 'static`
53+
mod test3 {
54+
type Opaque<T> = impl Sized;
55+
//~^ ERROR the parameter type `T` may not live long enough
56+
57+
fn define<T>() -> Opaque<T>
58+
where
59+
T: 'static,
60+
{
61+
|| {}
62+
}
63+
}
64+
65+
fn main() {}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,64 @@
1+
error[E0478]: lifetime bound not satisfied
2+
--> $DIR/closure_wf_outlives.rs:14:27
3+
|
4+
LL | type Opaque<'a, 'b> = impl Sized + 'a + 'b;
5+
| ^^^^^^^^^^^^^^^^^^^^
6+
|
7+
note: lifetime parameter instantiated with the lifetime `'a` as defined here
8+
--> $DIR/closure_wf_outlives.rs:14:17
9+
|
10+
LL | type Opaque<'a, 'b> = impl Sized + 'a + 'b;
11+
| ^^
12+
note: but lifetime parameter must outlive the lifetime `'b` as defined here
13+
--> $DIR/closure_wf_outlives.rs:14:21
14+
|
15+
LL | type Opaque<'a, 'b> = impl Sized + 'a + 'b;
16+
| ^^
17+
18+
error[E0495]: cannot infer an appropriate lifetime due to conflicting requirements
19+
--> $DIR/closure_wf_outlives.rs:27:27
20+
|
21+
LL | type Opaque<'a, 'b> = impl Sized + 'a + 'b;
22+
| ^^^^^^^^^^^^^^^^^^^^
23+
|
24+
note: first, the lifetime cannot outlive the lifetime `'a` as defined here...
25+
--> $DIR/closure_wf_outlives.rs:27:17
26+
|
27+
LL | type Opaque<'a, 'b> = impl Sized + 'a + 'b;
28+
| ^^
29+
note: ...so that the declared lifetime parameter bounds are satisfied
30+
--> $DIR/closure_wf_outlives.rs:27:27
31+
|
32+
LL | type Opaque<'a, 'b> = impl Sized + 'a + 'b;
33+
| ^^^^^^^^^^^^^^^^^^^^
34+
note: but, the lifetime must be valid for the lifetime `'b` as defined here...
35+
--> $DIR/closure_wf_outlives.rs:27:21
36+
|
37+
LL | type Opaque<'a, 'b> = impl Sized + 'a + 'b;
38+
| ^^
39+
note: ...so that the declared lifetime parameter bounds are satisfied
40+
--> $DIR/closure_wf_outlives.rs:27:27
41+
|
42+
LL | type Opaque<'a, 'b> = impl Sized + 'a + 'b;
43+
| ^^^^^^^^^^^^^^^^^^^^
44+
45+
error[E0310]: the parameter type `T` may not live long enough
46+
--> $DIR/closure_wf_outlives.rs:54:22
47+
|
48+
LL | type Opaque<T> = impl Sized;
49+
| ^^^^^^^^^^ ...so that the type `T` will meet its required lifetime bounds...
50+
|
51+
note: ...that is required by this bound
52+
--> $DIR/closure_wf_outlives.rs:59:12
53+
|
54+
LL | T: 'static,
55+
| ^^^^^^^
56+
help: consider adding an explicit lifetime bound...
57+
|
58+
LL | type Opaque<T: 'static> = impl Sized;
59+
| +++++++++
60+
61+
error: aborting due to 3 previous errors
62+
63+
Some errors have detailed explanations: E0310, E0478, E0495.
64+
For more information about an error, try `rustc --explain E0310`.

0 commit comments

Comments
 (0)