@@ -176,6 +176,10 @@ enum Expectation<'tcx> {
176
176
177
177
/// This expression will be cast to the `Ty`
178
178
ExpectCastableToType ( Ty < ' tcx > ) ,
179
+
180
+ /// This rvalue expression will be wrapped in `&` or `Box` and coerced
181
+ /// to `&Ty` or `Box<Ty>`, respectively. `Ty` is `[A]` or `Trait`.
182
+ ExpectRvalueLikeUnsized ( Ty < ' tcx > ) ,
179
183
}
180
184
181
185
impl < ' tcx > Expectation < ' tcx > {
@@ -196,7 +200,7 @@ impl<'tcx> Expectation<'tcx> {
196
200
// when checking the 'then' block which are incompatible with the
197
201
// 'else' branch.
198
202
fn adjust_for_branches < ' a > ( & self , fcx : & FnCtxt < ' a , ' tcx > ) -> Expectation < ' tcx > {
199
- match self . only_has_type ( ) {
203
+ match * self {
200
204
ExpectHasType ( ety) => {
201
205
let ety = fcx. infcx ( ) . shallow_resolve ( ety) ;
202
206
if !ty:: type_is_ty_var ( ety) {
@@ -205,6 +209,9 @@ impl<'tcx> Expectation<'tcx> {
205
209
NoExpectation
206
210
}
207
211
}
212
+ ExpectRvalueLikeUnsized ( ety) => {
213
+ ExpectRvalueLikeUnsized ( ety)
214
+ }
208
215
_ => NoExpectation
209
216
}
210
217
}
@@ -3678,7 +3685,7 @@ fn check_expr_with_unifier<'a, 'tcx, F>(fcx: &FnCtxt<'a, 'tcx>,
3678
3685
match unop {
3679
3686
ast:: UnUniq => match ty. sty {
3680
3687
ty:: ty_uniq( ty) => {
3681
- ExpectHasType ( ty)
3688
+ Expectation :: rvalue_hint ( ty)
3682
3689
}
3683
3690
_ => {
3684
3691
NoExpectation
@@ -3767,7 +3774,16 @@ fn check_expr_with_unifier<'a, 'tcx, F>(fcx: &FnCtxt<'a, 'tcx>,
3767
3774
let expected = expected. only_has_type ( ) ;
3768
3775
let hint = expected. map ( fcx, |ty| {
3769
3776
match ty. sty {
3770
- ty:: ty_rptr( _, ref mt) | ty:: ty_ptr( ref mt) => ExpectHasType ( mt. ty ) ,
3777
+ ty:: ty_rptr( _, ref mt) | ty:: ty_ptr( ref mt) => {
3778
+ if ty:: expr_is_lval ( fcx. tcx ( ) , & * * oprnd) {
3779
+ // Lvalues may legitimately have unsized types.
3780
+ // For example, dereferences of a fat pointer and
3781
+ // the last field of a struct can be unsized.
3782
+ ExpectHasType ( mt. ty )
3783
+ } else {
3784
+ Expectation :: rvalue_hint ( mt. ty )
3785
+ }
3786
+ }
3771
3787
_ => NoExpectation
3772
3788
}
3773
3789
} ) ;
@@ -3985,15 +4001,12 @@ fn check_expr_with_unifier<'a, 'tcx, F>(fcx: &FnCtxt<'a, 'tcx>,
3985
4001
check_cast ( fcx, expr, & * * e, & * * t) ;
3986
4002
}
3987
4003
ast:: ExprVec ( ref args) => {
3988
- let uty = match expected {
3989
- ExpectHasType ( uty) => {
3990
- match uty. sty {
3991
- ty:: ty_vec( ty, _) => Some ( ty) ,
3992
- _ => None
3993
- }
4004
+ let uty = expected. map_to_option ( fcx, |uty| {
4005
+ match uty. sty {
4006
+ ty:: ty_vec( ty, _) => Some ( ty) ,
4007
+ _ => None
3994
4008
}
3995
- _ => None
3996
- } ;
4009
+ } ) ;
3997
4010
3998
4011
let typ = match uty {
3999
4012
Some ( uty) => {
@@ -4020,8 +4033,8 @@ fn check_expr_with_unifier<'a, 'tcx, F>(fcx: &FnCtxt<'a, 'tcx>,
4020
4033
let uty = match expected {
4021
4034
ExpectHasType ( uty) => {
4022
4035
match uty. sty {
4023
- ty:: ty_vec( ty, _) => Some ( ty) ,
4024
- _ => None
4036
+ ty:: ty_vec( ty, _) => Some ( ty) ,
4037
+ _ => None
4025
4038
}
4026
4039
}
4027
4040
_ => None
@@ -4298,10 +4311,38 @@ fn constrain_path_type_parameters(fcx: &FnCtxt,
4298
4311
}
4299
4312
4300
4313
impl < ' tcx > Expectation < ' tcx > {
4314
+ /// Provide an expectation for an rvalue expression given an *optional*
4315
+ /// hint, which is not required for type safety (the resulting type might
4316
+ /// be checked higher up, as is the case with `&expr` and `box expr`), but
4317
+ /// is useful in determining the concrete type.
4318
+ ///
4319
+ /// The primary use case is where the expected type is a fat pointer,
4320
+ /// like `&[int]`. For example, consider the following statement:
4321
+ ///
4322
+ /// let x: &[int] = &[1, 2, 3];
4323
+ ///
4324
+ /// In this case, the expected type for the `&[1, 2, 3]` expression is
4325
+ /// `&[int]`. If however we were to say that `[1, 2, 3]` has the
4326
+ /// expectation `ExpectHasType([int])`, that would be too strong --
4327
+ /// `[1, 2, 3]` does not have the type `[int]` but rather `[int, ..3]`.
4328
+ /// It is only the `&[1, 2, 3]` expression as a whole that can be coerced
4329
+ /// to the type `&[int]`. Therefore, we propagate this more limited hint,
4330
+ /// which still is useful, because it informs integer literals and the like.
4331
+ /// See the test case `test/run-pass/coerce-expect-unsized.rs` and #20169
4332
+ /// for examples of where this comes up,.
4333
+ fn rvalue_hint ( ty : Ty < ' tcx > ) -> Expectation < ' tcx > {
4334
+ match ty. sty {
4335
+ ty:: ty_vec( _, None ) | ty:: ty_trait( ..) => {
4336
+ ExpectRvalueLikeUnsized ( ty)
4337
+ }
4338
+ _ => ExpectHasType ( ty)
4339
+ }
4340
+ }
4341
+
4301
4342
fn only_has_type ( self ) -> Expectation < ' tcx > {
4302
4343
match self {
4303
- NoExpectation | ExpectCastableToType ( .. ) => NoExpectation ,
4304
- ExpectHasType ( t ) => ExpectHasType ( t )
4344
+ ExpectHasType ( t ) => ExpectHasType ( t ) ,
4345
+ _ => NoExpectation
4305
4346
}
4306
4347
}
4307
4348
@@ -4321,6 +4362,10 @@ impl<'tcx> Expectation<'tcx> {
4321
4362
ExpectHasType (
4322
4363
fcx. infcx ( ) . resolve_type_vars_if_possible ( & t) )
4323
4364
}
4365
+ ExpectRvalueLikeUnsized ( t) => {
4366
+ ExpectRvalueLikeUnsized (
4367
+ fcx. infcx ( ) . resolve_type_vars_if_possible ( & t) )
4368
+ }
4324
4369
}
4325
4370
}
4326
4371
@@ -4329,7 +4374,9 @@ impl<'tcx> Expectation<'tcx> {
4329
4374
{
4330
4375
match self . resolve ( fcx) {
4331
4376
NoExpectation => NoExpectation ,
4332
- ExpectCastableToType ( ty) | ExpectHasType ( ty) => unpack ( ty) ,
4377
+ ExpectCastableToType ( ty) |
4378
+ ExpectHasType ( ty) |
4379
+ ExpectRvalueLikeUnsized ( ty) => unpack ( ty) ,
4333
4380
}
4334
4381
}
4335
4382
@@ -4338,7 +4385,9 @@ impl<'tcx> Expectation<'tcx> {
4338
4385
{
4339
4386
match self . resolve ( fcx) {
4340
4387
NoExpectation => None ,
4341
- ExpectCastableToType ( ty) | ExpectHasType ( ty) => unpack ( ty) ,
4388
+ ExpectCastableToType ( ty) |
4389
+ ExpectHasType ( ty) |
4390
+ ExpectRvalueLikeUnsized ( ty) => unpack ( ty) ,
4342
4391
}
4343
4392
}
4344
4393
}
@@ -4351,6 +4400,8 @@ impl<'tcx> Repr<'tcx> for Expectation<'tcx> {
4351
4400
t. repr( tcx) ) ,
4352
4401
ExpectCastableToType ( t) => format ! ( "ExpectCastableToType({})" ,
4353
4402
t. repr( tcx) ) ,
4403
+ ExpectRvalueLikeUnsized ( t) => format ! ( "ExpectRvalueLikeUnsized({})" ,
4404
+ t. repr( tcx) ) ,
4354
4405
}
4355
4406
}
4356
4407
}
0 commit comments