Second approach - using type contents

This commit is contained in:
Nick Cameron 2014-09-01 09:48:19 +12:00
parent bdfeb65585
commit cc598e6f8e
4 changed files with 11 additions and 28 deletions

View File

@ -167,6 +167,5 @@ register_diagnostics!(
E0155, E0155,
E0156, E0156,
E0157, E0157,
E0158, E0158
E0159
) )

View File

@ -596,8 +596,8 @@ fn check_ty(cx: &mut Context, aty: &Ty) {
Some(ref item_substs) => { Some(ref item_substs) => {
let def_map = cx.tcx.def_map.borrow(); let def_map = cx.tcx.def_map.borrow();
let did = def_map.get_copy(&id).def_id(); let did = def_map.get_copy(&id).def_id();
let ty = ty::lookup_item_type(cx.tcx, did); let generics = ty::lookup_item_type(cx.tcx, did).generics;
for def in ty.generics.types.iter() { for def in generics.types.iter() {
let ty = *item_substs.substs.types.get(def.space, let ty = *item_substs.substs.types.get(def.space,
def.index); def.index);
check_typaram_bounds(cx, aty.span, ty, def); check_typaram_bounds(cx, aty.span, ty, def);
@ -644,20 +644,6 @@ pub fn check_typaram_bounds(cx: &Context,
}); });
} }
// Check that the programmer has not added the `Sized` bound to a trait type
// which would fool the compiler into thinking that trait types are sized, when
// they are really unsized.
fn check_false_sized(cx: &Context, sp: Span, ty: ty::t) {
match ty::get(ty).sty {
ty::ty_trait(..) if ty::type_is_sized(cx.tcx, ty) => {
span_err!(cx.tcx.sess, sp, E0159,
"explicitly adding `Sized` bound to an unsized type `{}`",
ty_to_string(cx.tcx, ty));
}
_ => {}
}
}
fn check_bounds_on_structs_or_enums_in_type_if_possible(cx: &mut Context, fn check_bounds_on_structs_or_enums_in_type_if_possible(cx: &mut Context,
span: Span, span: Span,
ty: ty::t) { ty: ty::t) {
@ -688,7 +674,6 @@ fn check_bounds_on_structs_or_enums_in_type_if_possible(cx: &mut Context,
.types .types
.iter()) { .iter()) {
check_typaram_bounds(cx, span, *ty, type_param_def); check_typaram_bounds(cx, span, *ty, type_param_def);
check_false_sized(cx, span, *ty);
} }
// Check trait bounds. // Check trait bounds.
@ -716,7 +701,6 @@ fn check_bounds_on_structs_or_enums_in_type_if_possible(cx: &mut Context,
cx.tcx)).as_slice()); cx.tcx)).as_slice());
}) })
} }
ty::ty_uniq(ty) => check_false_sized(cx, span, ty),
_ => {} _ => {}
} }
}); });

View File

@ -2362,7 +2362,7 @@ pub fn type_contents(cx: &ctxt, ty: t) -> TypeContents {
} }
ty_trait(box ty::TyTrait { bounds, .. }) => { ty_trait(box ty::TyTrait { bounds, .. }) => {
object_contents(cx, bounds) | TC::ReachesFfiUnsafe object_contents(cx, bounds) | TC::ReachesFfiUnsafe | TC::Nonsized
} }
ty_ptr(ref mt) => { ty_ptr(ref mt) => {

View File

@ -8,18 +8,18 @@
// option. This file may not be copied, modified, or distributed // option. This file may not be copied, modified, or distributed
// except according to those terms. // except according to those terms.
// ignore-tidy-linelength
use std::cell::RefCell; use std::cell::RefCell;
trait Trait {} trait Trait {}
pub fn main() { pub fn main() {
let x: Vec<Trait + Sized> = Vec::new(); let x: Vec<Trait + Sized> = Vec::new();
//~^ ERROR explicitly adding `Sized` bound to an unsized type `Trait+Sized` //~^ ERROR instantiating a type parameter with an incompatible type `Trait+Sized`, which does not fulfill `Sized`
//~^^ ERROR explicitly adding `Sized` bound to an unsized type `Trait+Sized` //~^^ ERROR instantiating a type parameter with an incompatible type `Trait+Sized`, which does not fulfill `Sized`
let x: Vec<Box<Trait + Sized>> = Vec::new(); //~^^^ ERROR instantiating a type parameter with an incompatible type `Trait+Sized`, which does not fulfill `Sized`
//~^ ERROR explicitly adding `Sized` bound to an unsized type `Trait+Sized`
//~^^ ERROR explicitly adding `Sized` bound to an unsized type `Trait+Sized`
let x: Vec<Box<RefCell<Trait + Sized>>> = Vec::new(); let x: Vec<Box<RefCell<Trait + Sized>>> = Vec::new();
//~^ ERROR explicitly adding `Sized` bound to an unsized type `Trait+Sized` //~^ ERROR instantiating a type parameter with an incompatible type `Trait+Sized`, which does not fulfill `Sized`
//~^^ ERROR explicitly adding `Sized` bound to an unsized type `Trait+Sized` //~^^ ERROR instantiating a type parameter with an incompatible type `Trait+Sized`, which does not fulfill `Sized`
} }