Handle structs, unions and enums unsized field/variant separately

This commit is contained in:
Guillaume Gomez 2017-08-16 18:45:54 +02:00
parent 589d994c38
commit c3c99b92c7
7 changed files with 64 additions and 13 deletions

View File

@ -42,6 +42,7 @@ use syntax::ptr::P;
use syntax::symbol::{Symbol, keywords}; use syntax::symbol::{Symbol, keywords};
use syntax::tokenstream::TokenStream; use syntax::tokenstream::TokenStream;
use syntax::util::ThinVec; use syntax::util::ThinVec;
use ty::AdtKind;
use rustc_data_structures::indexed_vec; use rustc_data_structures::indexed_vec;
@ -1789,6 +1790,15 @@ impl Item_ {
ItemDefaultImpl(..) => "item", ItemDefaultImpl(..) => "item",
} }
} }
pub fn adt_kind(&self) -> Option<AdtKind> {
match *self {
ItemStruct(..) => Some(AdtKind::Struct),
ItemUnion(..) => Some(AdtKind::Union),
ItemEnum(..) => Some(AdtKind::Enum),
_ => None,
}
}
} }
/// A reference from an trait to one of its associated items. This /// A reference from an trait to one of its associated items. This

View File

@ -1111,9 +1111,19 @@ impl<'a, 'gcx, 'tcx> InferCtxt<'a, 'gcx, 'tcx> {
ObligationCauseCode::StructInitializerSized => { ObligationCauseCode::StructInitializerSized => {
err.note("structs must have a statically known size to be initialized"); err.note("structs must have a statically known size to be initialized");
} }
ObligationCauseCode::FieldSized => { ObligationCauseCode::FieldSized(ref item) => {
err.note("only the last field of a struct or an union may have a dynamically \ match *item {
sized type"); AdtKind::Struct => {
err.note("only the last field of a struct may have a dynamically \
sized type");
}
AdtKind::Union => {
err.note("no field of a union may have a dynamically sized type");
}
AdtKind::Enum => {
err.note("no field of an enum variant may have a dynamically sized type");
}
}
} }
ObligationCauseCode::ConstSized => { ObligationCauseCode::ConstSized => {
err.note("constant expressions must have a statically known size"); err.note("constant expressions must have a statically known size");
@ -1155,8 +1165,7 @@ impl<'a, 'gcx, 'tcx> InferCtxt<'a, 'gcx, 'tcx> {
fn suggest_new_overflow_limit(&self, err: &mut DiagnosticBuilder) { fn suggest_new_overflow_limit(&self, err: &mut DiagnosticBuilder) {
let current_limit = self.tcx.sess.recursion_limit.get(); let current_limit = self.tcx.sess.recursion_limit.get();
let suggested_limit = current_limit * 2; let suggested_limit = current_limit * 2;
err.help(&format!( err.help(&format!("consider adding a `#![recursion_limit=\"{}\"]` attribute to your crate",
"consider adding a `#![recursion_limit=\"{}\"]` attribute to your crate",
suggested_limit)); suggested_limit));
} }
} }

View File

@ -20,7 +20,7 @@ use hir::def_id::DefId;
use middle::region::RegionMaps; use middle::region::RegionMaps;
use middle::free_region::FreeRegionMap; use middle::free_region::FreeRegionMap;
use ty::subst::Substs; use ty::subst::Substs;
use ty::{self, Ty, TyCtxt, TypeFoldable, ToPredicate}; use ty::{self, AdtKind, Ty, TyCtxt, TypeFoldable, ToPredicate};
use ty::error::{ExpectedFound, TypeError}; use ty::error::{ExpectedFound, TypeError};
use infer::{InferCtxt}; use infer::{InferCtxt};
@ -133,7 +133,7 @@ pub enum ObligationCauseCode<'tcx> {
RepeatVec, RepeatVec,
/// Types of fields (other than the last) in a struct must be sized. /// Types of fields (other than the last) in a struct must be sized.
FieldSized, FieldSized(AdtKind),
/// Constant expressions must be sized. /// Constant expressions must be sized.
ConstSized, ConstSized,

View File

@ -195,7 +195,7 @@ impl<'a, 'tcx> Lift<'tcx> for traits::ObligationCauseCode<'a> {
super::ReturnType(id) => Some(super::ReturnType(id)), super::ReturnType(id) => Some(super::ReturnType(id)),
super::SizedReturnType => Some(super::SizedReturnType), super::SizedReturnType => Some(super::SizedReturnType),
super::RepeatVec => Some(super::RepeatVec), super::RepeatVec => Some(super::RepeatVec),
super::FieldSized => Some(super::FieldSized), super::FieldSized(item) => Some(super::FieldSized(item)),
super::ConstSized => Some(super::ConstSized), super::ConstSized => Some(super::ConstSized),
super::SharedStatic => Some(super::SharedStatic), super::SharedStatic => Some(super::SharedStatic),
super::BuiltinDerivedObligation(ref cause) => { super::BuiltinDerivedObligation(ref cause) => {
@ -484,7 +484,7 @@ impl<'tcx> TypeFoldable<'tcx> for traits::ObligationCauseCode<'tcx> {
super::SizedReturnType | super::SizedReturnType |
super::ReturnNoExpression | super::ReturnNoExpression |
super::RepeatVec | super::RepeatVec |
super::FieldSized | super::FieldSized(_) |
super::ConstSized | super::ConstSized |
super::SharedStatic | super::SharedStatic |
super::BlockTailExpression(_) | super::BlockTailExpression(_) |
@ -532,7 +532,7 @@ impl<'tcx> TypeFoldable<'tcx> for traits::ObligationCauseCode<'tcx> {
super::SizedReturnType | super::SizedReturnType |
super::ReturnNoExpression | super::ReturnNoExpression |
super::RepeatVec | super::RepeatVec |
super::FieldSized | super::FieldSized(_) |
super::ConstSized | super::ConstSized |
super::SharedStatic | super::SharedStatic |
super::BlockTailExpression(_) | super::BlockTailExpression(_) |

View File

@ -233,7 +233,10 @@ impl<'a, 'gcx> CheckTypeWellFormedVisitor<'a, 'gcx> {
fcx.tcx.require_lang_item(lang_items::SizedTraitLangItem), fcx.tcx.require_lang_item(lang_items::SizedTraitLangItem),
traits::ObligationCause::new(field.span, traits::ObligationCause::new(field.span,
fcx.body_id, fcx.body_id,
traits::FieldSized)); traits::FieldSized(match item.node.adt_kind() {
Some(i) => i,
None => bug!(),
})));
} }
// All field types must be well-formed. // All field types must be well-formed.

View File

@ -14,4 +14,13 @@ union Foo<T: ?Sized> {
value: T, value: T,
} }
struct Foo2<T: ?Sized> {
value: T,
t: u32,
}
enum Foo3<T: ?Sized> {
Value(T),
}
fn main() {} fn main() {}

View File

@ -6,7 +6,27 @@ error[E0277]: the trait bound `T: std::marker::Sized` is not satisfied
| |
= help: the trait `std::marker::Sized` is not implemented for `T` = help: the trait `std::marker::Sized` is not implemented for `T`
= help: consider adding a `where T: std::marker::Sized` bound = help: consider adding a `where T: std::marker::Sized` bound
= note: only the last field of a struct or an union may have a dynamically sized type = note: no field of a union may have a dynamically sized type
error: aborting due to previous error error[E0277]: the trait bound `T: std::marker::Sized` is not satisfied
--> $DIR/union-sized-field.rs:18:5
|
18 | value: T,
| ^^^^^^^^ `T` does not have a constant size known at compile-time
|
= help: the trait `std::marker::Sized` is not implemented for `T`
= help: consider adding a `where T: std::marker::Sized` bound
= note: only the last field of a struct may have a dynamically sized type
error[E0277]: the trait bound `T: std::marker::Sized` is not satisfied
--> $DIR/union-sized-field.rs:23:11
|
23 | Value(T),
| ^^ `T` does not have a constant size known at compile-time
|
= help: the trait `std::marker::Sized` is not implemented for `T`
= help: consider adding a `where T: std::marker::Sized` bound
= note: no field of an enum variant may have a dynamically sized type
error: aborting due to 3 previous errors