mirror of
https://github.com/rust-lang/rust.git
synced 2025-04-28 02:57:37 +00:00
Auto merge of #43901 - GuillaumeGomez:unsized-union-field, r=petrochenkov
udpdate error message for unsized union field Fixes #36312.
This commit is contained in:
commit
b8ce1a3d2e
@ -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
|
||||||
|
@ -1111,8 +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 may have a dynamically sized type");
|
match *item {
|
||||||
|
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");
|
||||||
@ -1154,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));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -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,
|
||||||
|
@ -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(_) |
|
||||||
|
@ -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.
|
||||||
|
26
src/test/ui/union-sized-field.rs
Normal file
26
src/test/ui/union-sized-field.rs
Normal file
@ -0,0 +1,26 @@
|
|||||||
|
// 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.
|
||||||
|
|
||||||
|
#![feature(untagged_unions)]
|
||||||
|
|
||||||
|
union Foo<T: ?Sized> {
|
||||||
|
value: T,
|
||||||
|
}
|
||||||
|
|
||||||
|
struct Foo2<T: ?Sized> {
|
||||||
|
value: T,
|
||||||
|
t: u32,
|
||||||
|
}
|
||||||
|
|
||||||
|
enum Foo3<T: ?Sized> {
|
||||||
|
Value(T),
|
||||||
|
}
|
||||||
|
|
||||||
|
fn main() {}
|
32
src/test/ui/union-sized-field.stderr
Normal file
32
src/test/ui/union-sized-field.stderr
Normal file
@ -0,0 +1,32 @@
|
|||||||
|
error[E0277]: the trait bound `T: std::marker::Sized` is not satisfied
|
||||||
|
--> $DIR/union-sized-field.rs:14:5
|
||||||
|
|
|
||||||
|
14 | 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 a union may have a dynamically sized type
|
||||||
|
|
||||||
|
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
|
||||||
|
|
Loading…
Reference in New Issue
Block a user