Skip to content

Commit

Permalink
Auto merge of #47009 - eddyb:issue-46855, r=arielb1
Browse files Browse the repository at this point in the history
rustc_trans: support ZST indexing involving uninhabited types.

Fixes #46855 in a minimal way. I decided against supporting non-memory `Rvalue::Len` in this PR (see #46855 (comment)), as `PlaceContext::Inspect` is also used for `Rvalue::Discriminant`.

r? @arielb1
  • Loading branch information
bors committed Dec 27, 2017
2 parents 71ed31f + 57bb8ab commit 63efff5
Show file tree
Hide file tree
Showing 2 changed files with 50 additions and 3 deletions.
19 changes: 16 additions & 3 deletions src/librustc_trans/mir/operand.rs
Original file line number Diff line number Diff line change
Expand Up @@ -267,9 +267,22 @@ impl<'a, 'tcx> MirContext<'a, 'tcx> {

// Moves out of scalar and scalar pair fields are trivial.
if let &mir::Place::Projection(ref proj) = place {
if let mir::ProjectionElem::Field(ref f, _) = proj.elem {
if let Some(o) = self.maybe_trans_consume_direct(bcx, &proj.base) {
return Some(o.extract_field(bcx, f.index()));
if let Some(o) = self.maybe_trans_consume_direct(bcx, &proj.base) {
match proj.elem {
mir::ProjectionElem::Field(ref f, _) => {
return Some(o.extract_field(bcx, f.index()));
}
mir::ProjectionElem::Index(_) |
mir::ProjectionElem::ConstantIndex { .. } => {
// ZSTs don't require any actual memory access.
// FIXME(eddyb) deduplicate this with the identical
// checks in `trans_consume` and `extract_field`.
let elem = o.layout.field(bcx.ccx, 0);
if elem.is_zst() {
return Some(OperandRef::new_zst(bcx.ccx, elem));
}
}
_ => {}
}
}
}
Expand Down
34 changes: 34 additions & 0 deletions src/test/run-pass/issue-46855.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,34 @@
// 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.

// compile-flags: -Zmir-opt-level=1

#![feature(slice_patterns)]

use std::mem;

#[derive(Copy, Clone)]
enum Never {}

union Foo {
a: u64,
b: Never
}

fn foo(xs: [(Never, u32); 1]) -> u32 { xs[0].1 }

fn bar([(_, x)]: [(Never, u32); 1]) -> u32 { x }

fn main() {
println!("{}", mem::size_of::<Foo>());

let f = [Foo { a: 42 }, Foo { a: 10 }];
println!("{:?}", unsafe { f[0].a });
}

0 comments on commit 63efff5

Please sign in to comment.