12 files changed +69
-67
lines changed Original file line number Diff line number Diff line change @@ -40,23 +40,25 @@ pub struct OpaqueTypeDecl<'tcx> {
40
40
}
41
41
42
42
impl < ' a , ' tcx > InferCtxt < ' a , ' tcx > {
43
- pub fn replace_opaque_types_with_inference_vars (
43
+ /// This is a backwards compatibility hack to prevent breaking changes from
44
+ /// lazy TAIT around RPIT handling.
45
+ pub fn replace_opaque_types_with_inference_vars < T : TypeFoldable < ' tcx > > (
44
46
& self ,
45
- ty : Ty < ' tcx > ,
47
+ value : T ,
46
48
body_id : HirId ,
47
49
span : Span ,
48
50
param_env : ty:: ParamEnv < ' tcx > ,
49
- ) -> InferOk < ' tcx , Ty < ' tcx > > {
50
- if !ty . has_opaque_types ( ) {
51
- return InferOk { value : ty , obligations : vec ! [ ] } ;
51
+ ) -> InferOk < ' tcx , T > {
52
+ if !value . has_opaque_types ( ) {
53
+ return InferOk { value, obligations : vec ! [ ] } ;
52
54
}
53
55
let mut obligations = vec ! [ ] ;
54
56
let replace_opaque_type = |def_id : DefId | {
55
57
def_id
56
58
. as_local ( )
57
59
. map_or ( false , |def_id| self . opaque_type_origin ( def_id, span) . is_some ( ) )
58
60
} ;
59
- let value = ty . fold_with ( & mut ty:: fold:: BottomUpFolder {
61
+ let value = value . fold_with ( & mut ty:: fold:: BottomUpFolder {
60
62
tcx : self . tcx ,
61
63
lt_op : |lt| lt,
62
64
ct_op : |ct| ct,
Original file line number Diff line number Diff line change @@ -252,10 +252,20 @@ fn project_and_unify_type<'cx, 'tcx>(
252
252
Err ( InProgress ) => return ProjectAndUnifyResult :: Recursive ,
253
253
} ;
254
254
debug ! ( ?normalized, ?obligations, "project_and_unify_type result" ) ;
255
- match infcx
256
- . at ( & obligation. cause , obligation. param_env )
257
- . eq ( normalized, obligation. predicate . term )
258
- {
255
+ let actual = obligation. predicate . term ;
256
+ // For an example where this is neccessary see src/test/ui/impl-trait/nested-return-type2.rs
257
+ // This allows users to omit re-mentioning all bounds on an associated type and just use an
258
+ // `impl Trait` for the assoc type to add more bounds.
259
+ let InferOk { value : actual, obligations : new } =
260
+ selcx. infcx ( ) . replace_opaque_types_with_inference_vars (
261
+ actual,
262
+ obligation. cause . body_id ,
263
+ obligation. cause . span ,
264
+ obligation. param_env ,
265
+ ) ;
266
+ obligations. extend ( new) ;
267
+
268
+ match infcx. at ( & obligation. cause , obligation. param_env ) . eq ( normalized, actual) {
259
269
Ok ( InferOk { obligations : inferred_obligations, value : ( ) } ) => {
260
270
obligations. extend ( inferred_obligations) ;
261
271
ProjectAndUnifyResult :: Holds ( obligations)
Original file line number Diff line number Diff line change 1
1
#![ feature( type_alias_impl_trait) ]
2
2
3
3
// edition:2021
4
+ // unset-rustc-env:RUST_BACKTRACE
5
+ // compile-flags:-Z treat-err-as-bug=1
6
+ // error-pattern:stack backtrace:
7
+ // failure-status:101
8
+ // normalize-stderr-test "note: .*" -> ""
9
+ // normalize-stderr-test "thread 'rustc' .*" -> ""
10
+ // normalize-stderr-test " +[0-9]+:.*\n" -> ""
11
+ // normalize-stderr-test " +at .*\n" -> ""
4
12
5
13
use std:: future:: Future ;
6
14
@@ -23,7 +31,6 @@ struct Context {
23
31
type TransactionResult < O > = Result < O , ( ) > ;
24
32
25
33
type TransactionFuture < ' __ , O > = impl ' __ + Future < Output = TransactionResult < O > > ;
26
- //~^ ERROR unconstrained opaque type
27
34
28
35
fn execute_transaction_fut < ' f , F , O > (
29
36
f : F ,
Original file line number Diff line number Diff line change 1
- error: unconstrained opaque type
2
- --> $DIR/issue-86800.rs:25:34
3
- |
4
- LL | type TransactionFuture<'__, O> = impl '__ + Future<Output = TransactionResult<O>>;
5
- | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
6
- |
7
- = note: `TransactionFuture` must be used in combination with a concrete type within the same module
8
1
9
- error: aborting due to previous error
2
+ stack backtrace:
10
3
4
+ error: internal compiler error: unexpected panic
5
+
6
+
7
+
8
+
9
+
10
+
11
+
12
+
13
+
14
+ query stack during panic:
15
+ #0 [mir_borrowck] borrow-checking `execute_transaction_fut`
16
+ #1 [type_of] computing type of `TransactionFuture::{opaque#0}`
17
+ #2 [check_mod_item_types] checking item types in top-level module
18
+ #3 [analysis] running analysis passes on this crate
19
+ end of query stack
Original file line number Diff line number Diff line change 1
1
#![ feature( type_alias_impl_trait) ]
2
2
3
+ // check-pass
4
+
3
5
trait Duh { }
4
6
5
7
impl Duh for i32 { }
@@ -17,13 +19,13 @@ impl<R: Duh, F: FnMut() -> R> Trait for F {
17
19
18
20
type Sendable = impl Send ;
19
21
20
- // The `Sendable` here is then later compared against the inference var
21
- // created, causing the inference var to be set to `Sendable` instead of
22
+ // The `Sendable` here is converted to an inference var and then later compared
23
+ // against the inference var created, causing the inference var to be set to
24
+ // the hidden type of `Sendable` instead of
22
25
// the hidden type. We already have obligations registered on the inference
23
26
// var to make it uphold the `: Duh` bound on `Trait::Assoc`. The opaque
24
- // type does not implement `Duh`, even if its hidden type does. So we error out .
27
+ // type does not implement `Duh`, but if its hidden type does.
25
28
fn foo ( ) -> impl Trait < Assoc = Sendable > {
26
- //~^ ERROR `Sendable: Duh` is not satisfied
27
29
|| 42
28
30
}
29
31
Load Diff This file was deleted.
Original file line number Diff line number Diff line change @@ -24,8 +24,8 @@ type Traitable = impl Trait<Assoc = Sendable>;
24
24
// var to make it uphold the `: Duh` bound on `Trait::Assoc`. The opaque
25
25
// type does not implement `Duh`, even if its hidden type does. So we error out.
26
26
fn foo ( ) -> Traitable {
27
- //~^ ERROR `Sendable: Duh` is not satisfied
28
27
|| 42
28
+ //~^ ERROR `Sendable: Duh` is not satisfied
29
29
}
30
30
31
31
fn main ( ) {
Original file line number Diff line number Diff line change 1
1
error[E0277]: the trait bound `Sendable: Duh` is not satisfied
2
- --> $DIR/nested-return-type2-tait2.rs:26:13
2
+ --> $DIR/nested-return-type2-tait2.rs:27:5
3
3
|
4
- LL | fn foo() -> Traitable {
5
- | ^^^^ ^^^^^ the trait `Duh` is not implemented for `Sendable`
4
+ LL | || 42
5
+ | ^^^^^ the trait `Duh` is not implemented for `Sendable`
6
6
|
7
7
= help: the trait `Duh` is implemented for `i32`
8
- note: required because of the requirements on the impl of `Trait` for `[closure@$DIR/nested-return-type2-tait2.rs:28 :5: 28 :7]`
8
+ note: required because of the requirements on the impl of `Trait` for `[closure@$DIR/nested-return-type2-tait2.rs:27 :5: 27 :7]`
9
9
--> $DIR/nested-return-type2-tait2.rs:14:31
10
10
|
11
11
LL | impl<R: Duh, F: FnMut() -> R> Trait for F {
Original file line number Diff line number Diff line change @@ -23,8 +23,8 @@ type Traitable = impl Trait<Assoc = impl Send>;
23
23
// var to make it uphold the `: Duh` bound on `Trait::Assoc`. The opaque
24
24
// type does not implement `Duh`, even if its hidden type does. So we error out.
25
25
fn foo ( ) -> Traitable {
26
- //~^ ERROR `impl Send: Duh` is not satisfied
27
26
|| 42
27
+ //~^ ERROR `impl Send: Duh` is not satisfied
28
28
}
29
29
30
30
fn main ( ) {
Original file line number Diff line number Diff line change 1
1
error[E0277]: the trait bound `impl Send: Duh` is not satisfied
2
- --> $DIR/nested-return-type2-tait3.rs:25:13
2
+ --> $DIR/nested-return-type2-tait3.rs:26:5
3
3
|
4
- LL | fn foo() -> Traitable {
5
- | ^^^^ ^^^^^ the trait `Duh` is not implemented for `impl Send`
4
+ LL | || 42
5
+ | ^^^^^ the trait `Duh` is not implemented for `impl Send`
6
6
|
7
7
= help: the trait `Duh` is implemented for `i32`
8
- note: required because of the requirements on the impl of `Trait` for `[closure@$DIR/nested-return-type2-tait3.rs:27 :5: 27 :7]`
8
+ note: required because of the requirements on the impl of `Trait` for `[closure@$DIR/nested-return-type2-tait3.rs:26 :5: 26 :7]`
9
9
--> $DIR/nested-return-type2-tait3.rs:14:31
10
10
|
11
11
LL | impl<R: Duh, F: FnMut() -> R> Trait for F {
Original file line number Diff line number Diff line change
1
+ // check-pass
2
+
1
3
trait Duh { }
2
4
3
5
impl Duh for i32 { }
@@ -18,9 +20,11 @@ impl<R: Duh, F: FnMut() -> R> Trait for F {
18
20
// the hidden type. We already have obligations registered on the inference
19
21
// var to make it uphold the `: Duh` bound on `Trait::Assoc`. The opaque
20
22
// type does not implement `Duh`, even if its hidden type does.
23
+ // Lazy TAIT would error out, but we inserted a hack to make it work again,
24
+ // keeping backwards compatibility.
21
25
fn foo ( ) -> impl Trait < Assoc = impl Send > {
22
- //~^ ERROR `impl Send: Duh` is not satisfied
23
26
|| 42
24
27
}
25
28
26
- fn main ( ) { }
29
+ fn main ( ) {
30
+ }
Load Diff This file was deleted.
0 commit comments