From c1fe4a22b9f35f326038b3a2745102ce7bd86fc1 Mon Sep 17 00:00:00 2001 From: Jonathan S Date: Tue, 19 Dec 2017 17:24:38 -0600 Subject: [PATCH 1/3] Only mark unions as uninhabited if all of their fields are uninhabited. Fixes #46845. --- src/librustc/ty/layout.rs | 30 +++++++++++++++--------- src/test/run-pass/issue-46845.rs | 39 ++++++++++++++++++++++++++++++++ 2 files changed, 58 insertions(+), 11 deletions(-) create mode 100644 src/test/run-pass/issue-46845.rs diff --git a/src/librustc/ty/layout.rs b/src/librustc/ty/layout.rs index a2692fb8f5a1e..68aa553c529a6 100644 --- a/src/librustc/ty/layout.rs +++ b/src/librustc/ty/layout.rs @@ -1352,22 +1352,19 @@ impl<'a, 'tcx> LayoutDetails { }).collect::, _>>() }).collect::, _>>()?; - 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() { let packed = def.repr.packed(); if packed && def.repr.align > 0 { 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() { 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() || // Only one variant is inhabited. (inh_second.is_none() && diff --git a/src/test/run-pass/issue-46845.rs b/src/test/run-pass/issue-46845.rs new file mode 100644 index 0000000000000..235d3982a9c0c --- /dev/null +++ b/src/test/run-pass/issue-46845.rs @@ -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 or the MIT license +// , 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::(), 8); + assert_eq!(mem::size_of::(), 0); + + let f = [Foo { a: 42 }, Foo { a: 10 }]; + println!("{}", unsafe { f[0].a }); + assert_eq!(unsafe { f[1].a }, 10); +} From cbcf2ffe8041ae29f51e3a378c86734d6b555c05 Mon Sep 17 00:00:00 2001 From: Jonathan S Date: Wed, 20 Dec 2017 19:36:41 -0600 Subject: [PATCH 2/3] Never mark unions as uninhabited. Although I think this is wrong, it is certainly sound, and the general consensus seems to value not having footguns over some sort of aesthetic consistency. --- src/librustc/ty/layout.rs | 5 ----- 1 file changed, 5 deletions(-) diff --git a/src/librustc/ty/layout.rs b/src/librustc/ty/layout.rs index 68aa553c529a6..b46a7585945c7 100644 --- a/src/librustc/ty/layout.rs +++ b/src/librustc/ty/layout.rs @@ -1361,11 +1361,6 @@ impl<'a, 'tcx> LayoutDetails { 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() { dl.i8_align } else { From da9791767bf9d4c334e22a1c4718e325279be984 Mon Sep 17 00:00:00 2001 From: Jonathan S Date: Sat, 23 Dec 2017 20:38:36 -0600 Subject: [PATCH 3/3] Remove unnecessary assert that unions have only one variant --- src/librustc/ty/layout.rs | 3 --- 1 file changed, 3 deletions(-) diff --git a/src/librustc/ty/layout.rs b/src/librustc/ty/layout.rs index b46a7585945c7..b6871147da81d 100644 --- a/src/librustc/ty/layout.rs +++ b/src/librustc/ty/layout.rs @@ -1357,9 +1357,6 @@ impl<'a, 'tcx> LayoutDetails { if packed && def.repr.align > 0 { bug!("Union cannot be packed and aligned"); } - if variants.len() != 1 { - bug!("Union must be represented as a single variant"); - } let mut align = if def.repr.packed() { dl.i8_align