Skip to content

Commit

Permalink
Auto merge of #50643 - dlrobertson:fix_ice, r=oli-obk
Browse files Browse the repository at this point in the history
typeck: Fix ICE with struct update syntax

If check_expr_struct_fields fails, do not continue to record update.
If we continue to record update, the struct may cause us to ICE later
on indexing a field that may or may not exist.

Fixes: #50618
  • Loading branch information
bors committed May 13, 2018
2 parents f9ae5bc + f6a46cf commit 0d48c96
Show file tree
Hide file tree
Showing 3 changed files with 64 additions and 18 deletions.
42 changes: 24 additions & 18 deletions src/librustc_typeck/check/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -3278,7 +3278,7 @@ impl<'a, 'gcx, 'tcx> FnCtxt<'a, 'gcx, 'tcx> {
span: Span,
variant: &'tcx ty::VariantDef,
ast_fields: &'gcx [hir::Field],
check_completeness: bool) {
check_completeness: bool) -> bool {
let tcx = self.tcx;

let adt_ty_hint =
Expand Down Expand Up @@ -3380,6 +3380,7 @@ impl<'a, 'gcx, 'tcx> FnCtxt<'a, 'gcx, 'tcx> {
truncated_fields_error))
.emit();
}
error_happened
}

fn check_struct_fields_on_error(&self,
Expand Down Expand Up @@ -3478,24 +3479,29 @@ impl<'a, 'gcx, 'tcx> FnCtxt<'a, 'gcx, 'tcx> {
}
}

self.check_expr_struct_fields(struct_ty, expected, expr.id, path_span, variant, fields,
base_expr.is_none());
let error_happened = self.check_expr_struct_fields(struct_ty, expected, expr.id, path_span,
variant, fields, base_expr.is_none());
if let &Some(ref base_expr) = base_expr {
self.check_expr_has_type_or_error(base_expr, struct_ty);
match struct_ty.sty {
ty::TyAdt(adt, substs) if adt.is_struct() => {
let fru_field_types = adt.non_enum_variant().fields.iter().map(|f| {
self.normalize_associated_types_in(expr.span, &f.ty(self.tcx, substs))
}).collect();

self.tables
.borrow_mut()
.fru_field_types_mut()
.insert(expr.hir_id, fru_field_types);
}
_ => {
span_err!(self.tcx.sess, base_expr.span, E0436,
"functional record update syntax requires a struct");
// If check_expr_struct_fields hit an error, do not attempt to populate
// the fields with the base_expr. This could cause us to hit errors later
// when certain fields are assumed to exist that in fact do not.
if !error_happened {
self.check_expr_has_type_or_error(base_expr, struct_ty);
match struct_ty.sty {
ty::TyAdt(adt, substs) if adt.is_struct() => {
let fru_field_types = adt.non_enum_variant().fields.iter().map(|f| {
self.normalize_associated_types_in(expr.span, &f.ty(self.tcx, substs))
}).collect();

self.tables
.borrow_mut()
.fru_field_types_mut()
.insert(expr.hir_id, fru_field_types);
}
_ => {
span_err!(self.tcx.sess, base_expr.span, E0436,
"functional record update syntax requires a struct");
}
}
}
}
Expand Down
29 changes: 29 additions & 0 deletions src/test/ui/issue-50618.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,29 @@
// Copyright 2017 The Rust Project Developers. See the COPYRIGHT
// file at the top-level directory of this distribution and at
// http://rust-lang.org/COPYRIGHT.
//
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
// option. This file may not be copied, modified, or distributed
// except according to those terms.

struct Point {
pub x: u64,
pub y: u64,
}

const TEMPLATE: Point = Point {
x: 0,
y: 0
};

fn main() {
let _ = || {
Point {
nonexistent: 0,
//~^ ERROR struct `Point` has no field named `nonexistent`
..TEMPLATE
}
};
}
11 changes: 11 additions & 0 deletions src/test/ui/issue-50618.stderr
Original file line number Diff line number Diff line change
@@ -0,0 +1,11 @@
error[E0560]: struct `Point` has no field named `nonexistent`
--> $DIR/issue-50618.rs:24:13
|
LL | nonexistent: 0,
| ^^^^^^^^^^^ `Point` does not have this field
|
= note: available fields are: `x`, `y`

error: aborting due to previous error

For more information about this error, try `rustc --explain E0560`.

0 comments on commit 0d48c96

Please sign in to comment.