auto merge of #14874 : pcwalton/rust/enum-to-float-casts-part-deux, r=alexcrichton

Closes #14794.

If you're casting from an enum to a float, cast through an integer
first.

[breaking-change]

r? @alexcrichton
This commit is contained in:
bors 2014-06-14 08:46:48 +00:00
commit dbd29ea96e
7 changed files with 32 additions and 18 deletions

View File

@ -503,9 +503,8 @@ fn const_expr_unadjusted(cx: &CrateContext, e: &ast::Expr,
let s = ty::type_is_signed(ety) as Bool;
llvm::LLVMConstIntCast(iv, llty.to_ref(), s)
}
expr::cast_float => llvm::LLVMConstSIToFP(iv, llty.to_ref()),
_ => cx.sess().bug("enum cast destination is not \
integral or float")
integral")
}
}
(expr::cast_pointer, expr::cast_pointer) => {

View File

@ -1685,6 +1685,14 @@ pub fn type_is_scalar(ty: t) -> bool {
}
}
/// Returns true if this type is a floating point type and false otherwise.
pub fn type_is_floating_point(ty: t) -> bool {
match get(ty).sty {
ty_float(_) => true,
_ => false,
}
}
pub fn type_needs_drop(cx: &ctxt, ty: t) -> bool {
type_contents(cx, ty).needs_drop(cx)
}

View File

@ -3042,12 +3042,24 @@ fn check_struct_enum_variant(fcx: &FnCtxt,
let t_1_is_scalar = type_is_scalar(fcx, expr.span, t_1);
let t_1_is_char = type_is_char(fcx, expr.span, t_1);
let t_1_is_bare_fn = type_is_bare_fn(fcx, expr.span, t_1);
let t_1_is_float = type_is_floating_point(fcx,
expr.span,
t_1);
// casts to scalars other than `char` and `bare fn` are trivial
let t_1_is_trivial = t_1_is_scalar &&
!t_1_is_char && !t_1_is_bare_fn;
if type_is_c_like_enum(fcx, expr.span, t_e) && t_1_is_trivial {
if type_is_c_like_enum(fcx, expr.span, t_e) &&
t_1_is_trivial {
if t_1_is_float {
fcx.type_error_message(expr.span, |actual| {
format!("illegal cast; cast through an \
integer first: `{}` as `{}`",
actual,
fcx.infcx().ty_to_str(t_1))
}, t_e, None);
}
// casts from C-like enums are allowed
} else if t_1_is_char {
let te = fcx.infcx().resolve_type_vars_if_possible(te);
@ -4206,6 +4218,11 @@ pub fn type_is_bare_fn(fcx: &FnCtxt, sp: Span, typ: ty::t) -> bool {
return ty::type_is_bare_fn(typ_s);
}
pub fn type_is_floating_point(fcx: &FnCtxt, sp: Span, typ: ty::t) -> bool {
let typ_s = structurally_resolved_type(fcx, sp, typ);
return ty::type_is_floating_point(typ_s);
}
pub fn type_is_unsafe_ptr(fcx: &FnCtxt, sp: Span, typ: ty::t) -> bool {
let typ_s = structurally_resolved_type(fcx, sp, typ);
return ty::type_is_unsafe_ptr(typ_s);

View File

@ -8,7 +8,7 @@
// option. This file may not be copied, modified, or distributed
// except according to those terms.
// Tests that enum-to-float-casts do *signed* integer-to-float conversion.
// Tests that enum-to-float casts are disallowed.
enum E {
L0 = -1,
@ -20,13 +20,13 @@ enum F {
H1 = 0xFFFFFFFFFFFFFFFF
}
static C0: f32 = L0 as f32;
static C1: f32 = H1 as f32;
static C0: f32 = L0 as f32; //~ ERROR illegal cast
static C1: f32 = H1 as f32; //~ ERROR illegal cast
pub fn main() {
let a = L0 as f32;
let a = L0 as f32; //~ ERROR illegal cast
let b = C0;
let c = H1 as f32;
let c = H1 as f32; //~ ERROR illegal cast
let d = C1;
assert_eq!(a, -1.0f32);
assert_eq!(b, -1.0f32);

View File

@ -14,18 +14,10 @@ enum B { B1=0, B2=2 }
pub fn main () {
static c1: int = A2 as int;
static c2: int = B2 as int;
static c3: f64 = A2 as f64;
static c4: f64 = B2 as f64;
let a1 = A2 as int;
let a2 = B2 as int;
let a3 = A2 as f64;
let a4 = B2 as f64;
assert_eq!(c1, 1);
assert_eq!(c2, 2);
assert_eq!(c3, 1.0);
assert_eq!(c4, 2.0);
assert_eq!(a1, 1);
assert_eq!(a2, 2);
assert_eq!(a3, 1.0);
assert_eq!(a4, 2.0);
}

View File

@ -21,5 +21,4 @@ pub fn main() {
fn test_color(color: color, val: int, _name: String) {
assert!(color as int == val);
assert!(color as f64 == val as f64);
}

View File

@ -40,7 +40,6 @@ pub fn main() {
fn test_color(color: color, val: int, name: String) {
//assert!(unsafe::transmute(color) == val);
assert_eq!(color as int, val);
assert_eq!(color as f64, val as f64);
assert!(get_color_alt(color) == name);
assert!(get_color_if(color) == name);
}