Skip to content

Commit d10642e

Browse files
committed
auto merge of #20083 : eddyb/rust/fix-expectation, r=nikomatsakis
This fixes a few corner cases with expected type propagation, e.g.: ```rust fn take_int_slice(_: &[int]) {} take_int_slice(&if 1 < 0 { [ 0, 1 ] } else { [ 0, 1 ] }); ``` ```rust <anon>:2:28: 2:36 error: mismatched types: expected `[int]`, found `[int, ..2]` <anon>:2 take_int_slice(&if 1 < 0 { [ 0, 1 ] } else { [ 0, 1 ] }); ^~~~~~~~ <anon>:2:46: 2:54 error: mismatched types: expected `[int]`, found `[int, ..2]` <anon>:2 take_int_slice(&if 1 < 0 { [ 0, 1 ] } else { [ 0, 1 ] }); ^~~~~~~~ ``` Right now we unpack the expected `&[int]` and pass down `[int]`, forcing rvalue expressions to take unsized types, which causes mismatch errors. Instead, I replaced that expectation with a weaker hint, for the unsized cases - a hint is still required to infer the integer literals' types, above. Fixes #20169.
2 parents 6585294 + adabf4e commit d10642e

File tree

4 files changed

+113
-42
lines changed

4 files changed

+113
-42
lines changed

src/librustc/middle/ty.rs

+8-4
Original file line numberDiff line numberDiff line change
@@ -4212,10 +4212,14 @@ pub fn expr_kind(tcx: &ctxt, expr: &ast::Expr) -> ExprKind {
42124212
}
42134213

42144214
def::DefStruct(_) => {
4215-
match expr_ty(tcx, expr).sty {
4216-
ty_bare_fn(..) => RvalueDatumExpr,
4217-
_ => RvalueDpsExpr
4218-
}
4215+
match tcx.node_types.borrow().get(&expr.id) {
4216+
Some(ty) => match ty.sty {
4217+
ty_bare_fn(..) => RvalueDatumExpr,
4218+
_ => RvalueDpsExpr
4219+
},
4220+
// See ExprCast below for why types might be missing.
4221+
None => RvalueDatumExpr
4222+
}
42194223
}
42204224

42214225
// Special case: A unit like struct's constructor must be called without () at the

src/librustc_typeck/check/closure.rs

+7-21
Original file line numberDiff line numberDiff line change
@@ -10,9 +10,7 @@
1010

1111
//! Code for type-checking closure expressions.
1212
13-
use super::check_fn;
14-
use super::{Expectation, ExpectCastableToType, ExpectHasType, NoExpectation};
15-
use super::FnCtxt;
13+
use super::{check_fn, Expectation, FnCtxt};
1614

1715
use astconv;
1816
use middle::infer;
@@ -34,13 +32,17 @@ pub fn check_expr_closure<'a,'tcx>(fcx: &FnCtxt<'a,'tcx>,
3432
expr.repr(fcx.tcx()),
3533
expected.repr(fcx.tcx()));
3634

35+
let expected_sig_and_kind = expected.map_to_option(fcx, |ty| {
36+
deduce_unboxed_closure_expectations_from_expected_type(fcx, ty)
37+
});
38+
3739
match opt_kind {
3840
None => {
3941
// If users didn't specify what sort of closure they want,
4042
// examine the expected type. For now, if we see explicit
4143
// evidence than an unboxed closure is desired, we'll use
4244
// that, otherwise we'll fall back to boxed closures.
43-
match deduce_unboxed_closure_expectations_from_expectation(fcx, expected) {
45+
match expected_sig_and_kind {
4446
None => { // doesn't look like an unboxed closure
4547
let region = astconv::opt_ast_region_to_region(fcx,
4648
fcx.infcx(),
@@ -66,10 +68,7 @@ pub fn check_expr_closure<'a,'tcx>(fcx: &FnCtxt<'a,'tcx>,
6668
ast::FnOnceUnboxedClosureKind => ty::FnOnceUnboxedClosureKind,
6769
};
6870

69-
let expected_sig =
70-
deduce_unboxed_closure_expectations_from_expectation(fcx, expected)
71-
.map(|t| t.0);
72-
71+
let expected_sig = expected_sig_and_kind.map(|t| t.0);
7372
check_unboxed_closure(fcx, expr, kind, decl, body, expected_sig);
7473
}
7574
}
@@ -147,19 +146,6 @@ fn check_unboxed_closure<'a,'tcx>(fcx: &FnCtxt<'a,'tcx>,
147146
.insert(expr_def_id, unboxed_closure);
148147
}
149148

150-
fn deduce_unboxed_closure_expectations_from_expectation<'a,'tcx>(
151-
fcx: &FnCtxt<'a,'tcx>,
152-
expected: Expectation<'tcx>)
153-
-> Option<(ty::FnSig<'tcx>,ty::UnboxedClosureKind)>
154-
{
155-
match expected.resolve(fcx) {
156-
NoExpectation => None,
157-
ExpectCastableToType(t) | ExpectHasType(t) => {
158-
deduce_unboxed_closure_expectations_from_expected_type(fcx, t)
159-
}
160-
}
161-
}
162-
163149
fn deduce_unboxed_closure_expectations_from_expected_type<'a,'tcx>(
164150
fcx: &FnCtxt<'a,'tcx>,
165151
expected_ty: Ty<'tcx>)

src/librustc_typeck/check/mod.rs

+68-17
Original file line numberDiff line numberDiff line change
@@ -176,6 +176,10 @@ enum Expectation<'tcx> {
176176

177177
/// This expression will be cast to the `Ty`
178178
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>),
179183
}
180184

181185
impl<'tcx> Expectation<'tcx> {
@@ -196,7 +200,7 @@ impl<'tcx> Expectation<'tcx> {
196200
// when checking the 'then' block which are incompatible with the
197201
// 'else' branch.
198202
fn adjust_for_branches<'a>(&self, fcx: &FnCtxt<'a, 'tcx>) -> Expectation<'tcx> {
199-
match self.only_has_type() {
203+
match *self {
200204
ExpectHasType(ety) => {
201205
let ety = fcx.infcx().shallow_resolve(ety);
202206
if !ty::type_is_ty_var(ety) {
@@ -205,6 +209,9 @@ impl<'tcx> Expectation<'tcx> {
205209
NoExpectation
206210
}
207211
}
212+
ExpectRvalueLikeUnsized(ety) => {
213+
ExpectRvalueLikeUnsized(ety)
214+
}
208215
_ => NoExpectation
209216
}
210217
}
@@ -3678,7 +3685,7 @@ fn check_expr_with_unifier<'a, 'tcx, F>(fcx: &FnCtxt<'a, 'tcx>,
36783685
match unop {
36793686
ast::UnUniq => match ty.sty {
36803687
ty::ty_uniq(ty) => {
3681-
ExpectHasType(ty)
3688+
Expectation::rvalue_hint(ty)
36823689
}
36833690
_ => {
36843691
NoExpectation
@@ -3767,7 +3774,16 @@ fn check_expr_with_unifier<'a, 'tcx, F>(fcx: &FnCtxt<'a, 'tcx>,
37673774
let expected = expected.only_has_type();
37683775
let hint = expected.map(fcx, |ty| {
37693776
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+
}
37713787
_ => NoExpectation
37723788
}
37733789
});
@@ -3985,15 +4001,12 @@ fn check_expr_with_unifier<'a, 'tcx, F>(fcx: &FnCtxt<'a, 'tcx>,
39854001
check_cast(fcx, expr, &**e, &**t);
39864002
}
39874003
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
39944008
}
3995-
_ => None
3996-
};
4009+
});
39974010

39984011
let typ = match uty {
39994012
Some(uty) => {
@@ -4020,8 +4033,8 @@ fn check_expr_with_unifier<'a, 'tcx, F>(fcx: &FnCtxt<'a, 'tcx>,
40204033
let uty = match expected {
40214034
ExpectHasType(uty) => {
40224035
match uty.sty {
4023-
ty::ty_vec(ty, _) => Some(ty),
4024-
_ => None
4036+
ty::ty_vec(ty, _) => Some(ty),
4037+
_ => None
40254038
}
40264039
}
40274040
_ => None
@@ -4298,10 +4311,38 @@ fn constrain_path_type_parameters(fcx: &FnCtxt,
42984311
}
42994312

43004313
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+
43014342
fn only_has_type(self) -> Expectation<'tcx> {
43024343
match self {
4303-
NoExpectation | ExpectCastableToType(..) => NoExpectation,
4304-
ExpectHasType(t) => ExpectHasType(t)
4344+
ExpectHasType(t) => ExpectHasType(t),
4345+
_ => NoExpectation
43054346
}
43064347
}
43074348

@@ -4321,6 +4362,10 @@ impl<'tcx> Expectation<'tcx> {
43214362
ExpectHasType(
43224363
fcx.infcx().resolve_type_vars_if_possible(&t))
43234364
}
4365+
ExpectRvalueLikeUnsized(t) => {
4366+
ExpectRvalueLikeUnsized(
4367+
fcx.infcx().resolve_type_vars_if_possible(&t))
4368+
}
43244369
}
43254370
}
43264371

@@ -4329,7 +4374,9 @@ impl<'tcx> Expectation<'tcx> {
43294374
{
43304375
match self.resolve(fcx) {
43314376
NoExpectation => NoExpectation,
4332-
ExpectCastableToType(ty) | ExpectHasType(ty) => unpack(ty),
4377+
ExpectCastableToType(ty) |
4378+
ExpectHasType(ty) |
4379+
ExpectRvalueLikeUnsized(ty) => unpack(ty),
43334380
}
43344381
}
43354382

@@ -4338,7 +4385,9 @@ impl<'tcx> Expectation<'tcx> {
43384385
{
43394386
match self.resolve(fcx) {
43404387
NoExpectation => None,
4341-
ExpectCastableToType(ty) | ExpectHasType(ty) => unpack(ty),
4388+
ExpectCastableToType(ty) |
4389+
ExpectHasType(ty) |
4390+
ExpectRvalueLikeUnsized(ty) => unpack(ty),
43424391
}
43434392
}
43444393
}
@@ -4351,6 +4400,8 @@ impl<'tcx> Repr<'tcx> for Expectation<'tcx> {
43514400
t.repr(tcx)),
43524401
ExpectCastableToType(t) => format!("ExpectCastableToType({})",
43534402
t.repr(tcx)),
4403+
ExpectRvalueLikeUnsized(t) => format!("ExpectRvalueLikeUnsized({})",
4404+
t.repr(tcx)),
43544405
}
43554406
}
43564407
}
+30
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,30 @@
1+
// Copyright 2014 The Rust Project Developers. See the COPYRIGHT
2+
// file at the top-level directory of this distribution and at
3+
// http://rust-lang.org/COPYRIGHT.
4+
//
5+
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
6+
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
7+
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
8+
// option. This file may not be copied, modified, or distributed
9+
// except according to those terms.
10+
11+
use std::fmt::Show;
12+
13+
// Check that coercions apply at the pointer level and don't cause
14+
// rvalue expressions to be unsized. See #20169 for more information.
15+
16+
pub fn main() {
17+
let _: Box<[int]> = box { [1, 2, 3] };
18+
let _: Box<[int]> = box if true { [1, 2, 3] } else { [1, 3, 4] };
19+
let _: Box<[int]> = box match true { true => [1, 2, 3], false => [1, 3, 4] };
20+
let _: Box<Fn(int) -> _> = box { |x| (x as u8) };
21+
let _: Box<Show> = box if true { false } else { true };
22+
let _: Box<Show> = box match true { true => 'a', false => 'b' };
23+
24+
let _: &[int] = &{ [1, 2, 3] };
25+
let _: &[int] = &if true { [1, 2, 3] } else { [1, 3, 4] };
26+
let _: &[int] = &match true { true => [1, 2, 3], false => [1, 3, 4] };
27+
let _: &Fn(int) -> _ = &{ |x| (x as u8) };
28+
let _: &Show = &if true { false } else { true };
29+
let _: &Show = &match true { true => 'a', false => 'b' };
30+
}

0 commit comments

Comments
 (0)