Skip to content

Commit 49e6b46

Browse files
committed
Mark enums with non-zero discriminant as non-zero
1 parent 6dc035e commit 49e6b46

File tree

3 files changed

+58
-15
lines changed

3 files changed

+58
-15
lines changed

src/librustc/ty/layout.rs

+17-15
Original file line numberDiff line numberDiff line change
@@ -597,7 +597,8 @@ impl<'a, 'gcx, 'tcx> Struct {
597597
-> Result<Option<FieldPath>, LayoutError<'gcx>> {
598598
let tcx = infcx.tcx.global_tcx();
599599
match (ty.layout(infcx)?, &ty.sty) {
600-
(&Scalar { non_zero: true, .. }, _) => Ok(Some(vec![])),
600+
(&Scalar { non_zero: true, .. }, _) |
601+
(&CEnum { non_zero: true, .. }, _) => Ok(Some(vec![])),
601602
(&FatPointer { non_zero: true, .. }, _) => {
602603
Ok(Some(vec![FAT_PTR_ADDR as u32]))
603604
}
@@ -769,6 +770,7 @@ pub enum Layout {
769770
CEnum {
770771
discr: Integer,
771772
signed: bool,
773+
non_zero: bool,
772774
// Inclusive discriminant range.
773775
// If min > max, it represents min...u64::MAX followed by 0...max.
774776
// FIXME(eddyb) always use the shortest range, e.g. by finding
@@ -1002,9 +1004,10 @@ impl<'a, 'gcx, 'tcx> Layout {
10021004

10031005
if def.is_enum() && def.variants.iter().all(|v| v.fields.is_empty()) {
10041006
// All bodies empty -> intlike
1005-
let (mut min, mut max) = (i64::MAX, i64::MIN);
1007+
let (mut min, mut max, mut non_zero) = (i64::MAX, i64::MIN, true);
10061008
for v in &def.variants {
10071009
let x = v.disr_val.to_u64_unchecked() as i64;
1010+
if x == 0 { non_zero = false; }
10081011
if x < min { min = x; }
10091012
if x > max { max = x; }
10101013
}
@@ -1013,6 +1016,7 @@ impl<'a, 'gcx, 'tcx> Layout {
10131016
return success(CEnum {
10141017
discr: discr,
10151018
signed: signed,
1019+
non_zero: non_zero,
10161020
min: min as u64,
10171021
max: max as u64
10181022
});
@@ -1069,19 +1073,17 @@ impl<'a, 'gcx, 'tcx> Layout {
10691073

10701074
// FIXME(eddyb) should take advantage of a newtype.
10711075
if path == &[0] && variants[discr].len() == 1 {
1072-
match *variants[discr][0].layout(infcx)? {
1073-
Scalar { value, .. } => {
1074-
return success(RawNullablePointer {
1075-
nndiscr: discr as u64,
1076-
value: value
1077-
});
1078-
}
1079-
_ => {
1080-
bug!("Layout::compute: `{}`'s non-zero \
1081-
`{}` field not scalar?!",
1082-
ty, variants[discr][0])
1083-
}
1084-
}
1076+
let value = match *variants[discr][0].layout(infcx)? {
1077+
Scalar { value, .. } => value,
1078+
CEnum { discr, .. } => Int(discr),
1079+
_ => bug!("Layout::compute: `{}`'s non-zero \
1080+
`{}` field not scalar?!",
1081+
ty, variants[discr][0])
1082+
};
1083+
return success(RawNullablePointer {
1084+
nndiscr: discr as u64,
1085+
value: value,
1086+
});
10851087
}
10861088

10871089
path.push(0); // For GEP through a pointer.

src/test/run-pass/nonzero-enum.rs

+39
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,39 @@
1+
// Copyright 2016 The Rust Project Developers. See the COPYRIGHT
2+
// file at the top-level directory of this distribution and at
3+
// http://rust-lang.org/COPYRIGHT.
4+
//
5+
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
6+
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
7+
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
8+
// option. This file may not be copied, modified, or distributed
9+
// except according to those terms.
10+
11+
use std::mem::size_of;
12+
13+
enum E {
14+
A = 1,
15+
B = 2,
16+
C = 3,
17+
}
18+
19+
struct S {
20+
a: u16,
21+
b: u8,
22+
e: E,
23+
}
24+
25+
fn main() {
26+
assert_eq!(size_of::<E>(), 1);
27+
assert_eq!(size_of::<Option<E>>(), 1);
28+
assert_eq!(size_of::<Result<E, ()>>(), 1);
29+
assert_eq!(size_of::<S>(), 4);
30+
assert_eq!(size_of::<Option<S>>(), 4);
31+
let enone = None::<E>;
32+
let esome = Some(E::A);
33+
if let Some(..) = enone {
34+
panic!();
35+
}
36+
if let None = esome {
37+
panic!();
38+
}
39+
}

src/tools/compiletest/src/main.rs

+2
Original file line numberDiff line numberDiff line change
@@ -15,6 +15,8 @@
1515
#![feature(test)]
1616
#![feature(libc)]
1717

18+
#![cfg_attr(stage0, feature(question_mark))]
19+
1820
#![deny(warnings)]
1921

2022
extern crate libc;

0 commit comments

Comments
 (0)