Only mark unions as uninhabited if all of their fields are uninhabited. Fixes #46845.
This commit is contained in:
parent
b76f224af8
commit
c1fe4a22b9
@ -1352,22 +1352,19 @@ impl<'a, 'tcx> LayoutDetails {
|
|||||||
}).collect::<Result<Vec<_>, _>>()
|
}).collect::<Result<Vec<_>, _>>()
|
||||||
}).collect::<Result<Vec<_>, _>>()?;
|
}).collect::<Result<Vec<_>, _>>()?;
|
||||||
|
|
||||||
let (inh_first, inh_second) = {
|
|
||||||
let mut inh_variants = (0..variants.len()).filter(|&v| {
|
|
||||||
variants[v].iter().all(|f| f.abi != Abi::Uninhabited)
|
|
||||||
});
|
|
||||||
(inh_variants.next(), inh_variants.next())
|
|
||||||
};
|
|
||||||
if inh_first.is_none() {
|
|
||||||
// Uninhabited because it has no variants, or only uninhabited ones.
|
|
||||||
return Ok(tcx.intern_layout(LayoutDetails::uninhabited(0)));
|
|
||||||
}
|
|
||||||
|
|
||||||
if def.is_union() {
|
if def.is_union() {
|
||||||
let packed = def.repr.packed();
|
let packed = def.repr.packed();
|
||||||
if packed && def.repr.align > 0 {
|
if packed && def.repr.align > 0 {
|
||||||
bug!("Union cannot be packed and aligned");
|
bug!("Union cannot be packed and aligned");
|
||||||
}
|
}
|
||||||
|
if variants.len() != 1 {
|
||||||
|
bug!("Union must be represented as a single variant");
|
||||||
|
}
|
||||||
|
|
||||||
|
if variants[0].iter().all(|f| f.abi == Abi::Uninhabited) {
|
||||||
|
// Uninhabited because it has only uninhabited variants/fields.
|
||||||
|
return Ok(tcx.intern_layout(LayoutDetails::uninhabited(0)));
|
||||||
|
}
|
||||||
|
|
||||||
let mut align = if def.repr.packed() {
|
let mut align = if def.repr.packed() {
|
||||||
dl.i8_align
|
dl.i8_align
|
||||||
@ -1400,6 +1397,17 @@ impl<'a, 'tcx> LayoutDetails {
|
|||||||
}));
|
}));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
let (inh_first, inh_second) = {
|
||||||
|
let mut inh_variants = (0..variants.len()).filter(|&v| {
|
||||||
|
variants[v].iter().all(|f| f.abi != Abi::Uninhabited)
|
||||||
|
});
|
||||||
|
(inh_variants.next(), inh_variants.next())
|
||||||
|
};
|
||||||
|
if inh_first.is_none() {
|
||||||
|
// Uninhabited because it has no variants, or only uninhabited ones.
|
||||||
|
return Ok(tcx.intern_layout(LayoutDetails::uninhabited(0)));
|
||||||
|
}
|
||||||
|
|
||||||
let is_struct = !def.is_enum() ||
|
let is_struct = !def.is_enum() ||
|
||||||
// Only one variant is inhabited.
|
// Only one variant is inhabited.
|
||||||
(inh_second.is_none() &&
|
(inh_second.is_none() &&
|
||||||
|
39
src/test/run-pass/issue-46845.rs
Normal file
39
src/test/run-pass/issue-46845.rs
Normal file
@ -0,0 +1,39 @@
|
|||||||
|
// 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.
|
||||||
|
|
||||||
|
// To work around #46855
|
||||||
|
// compile-flags: -Z mir-opt-level=0
|
||||||
|
// Regression test for the inhabitedness of unions with uninhabited variants, issue #46845
|
||||||
|
|
||||||
|
use std::mem;
|
||||||
|
|
||||||
|
#[derive(Copy, Clone)]
|
||||||
|
enum Never { }
|
||||||
|
|
||||||
|
// A single uninhabited variant shouldn't make the whole union uninhabited.
|
||||||
|
union Foo {
|
||||||
|
a: u64,
|
||||||
|
_b: Never
|
||||||
|
}
|
||||||
|
|
||||||
|
// If all the variants are uninhabited, however, the union should be uninhabited.
|
||||||
|
union Bar {
|
||||||
|
_a: (Never, u64),
|
||||||
|
_b: (u64, Never)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn main() {
|
||||||
|
assert_eq!(mem::size_of::<Foo>(), 8);
|
||||||
|
assert_eq!(mem::size_of::<Bar>(), 0);
|
||||||
|
|
||||||
|
let f = [Foo { a: 42 }, Foo { a: 10 }];
|
||||||
|
println!("{}", unsafe { f[0].a });
|
||||||
|
assert_eq!(unsafe { f[1].a }, 10);
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user