Skip to content

Commit

Permalink
Auto merge of #50354 - varkor:initial-field-alignment-c-int, r=eddyb
Browse files Browse the repository at this point in the history
Correct initial field alignment for repr(C)/repr(int)

Fixes #50098 following #50098 (comment).

(I wasn't sure which kind of test was best suited here — I picked run-pass simply because that was convenient, but if codegen is more appropriate, let me know and I'll change it.)

r? @eddyb
  • Loading branch information
bors committed May 2, 2018
2 parents d40a0b3 + 2d0d73e commit 3eadd75
Show file tree
Hide file tree
Showing 4 changed files with 108 additions and 2 deletions.
8 changes: 6 additions & 2 deletions src/librustc/ty/layout.rs
Original file line number Diff line number Diff line change
Expand Up @@ -940,11 +940,15 @@ impl<'a, 'tcx> LayoutCx<'tcx, TyCtxt<'a, 'tcx, 'tcx>> {
// We increase the size of the discriminant to avoid LLVM copying
// padding when it doesn't need to. This normally causes unaligned
// load/stores and excessive memcpy/memset operations. By using a
// bigger integer size, LLVM can be sure about it's contents and
// bigger integer size, LLVM can be sure about its contents and
// won't be so conservative.

// Use the initial field alignment
let mut ity = Integer::for_abi_align(dl, start_align).unwrap_or(min_ity);
let mut ity = if def.repr.c() || def.repr.int.is_some() {
min_ity
} else {
Integer::for_abi_align(dl, start_align).unwrap_or(min_ity)
};

// If the alignment is not larger than the chosen discriminant size,
// don't use the alignment as the final size.
Expand Down
55 changes: 55 additions & 0 deletions src/test/run-pass/repr_c_int_align.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,55 @@
// Copyright 2018 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: -O

#![allow(dead_code)]

#[repr(C, u8)]
enum ReprCu8 {
A(u16),
B,
}

#[repr(u8)]
enum Repru8 {
A(u16),
B,
}

#[repr(C)]
struct ReprC {
tag: u8,
padding: u8,
payload: u16,
}

fn main() {
// Test `repr(C, u8)`.
let r1 = ReprC { tag: 0, padding: 0, payload: 0 };
let r2 = ReprC { tag: 0, padding: 1, payload: 1 };

let t1: &ReprCu8 = unsafe { std::mem::transmute(&r1) };
let t2: &ReprCu8 = unsafe { std::mem::transmute(&r2) };

match (t1, t2) {
(ReprCu8::A(_), ReprCu8::A(_)) => (),
_ => assert!(false)
};

// Test `repr(u8)`.
let t1: &Repru8 = unsafe { std::mem::transmute(&r1) };
let t2: &Repru8 = unsafe { std::mem::transmute(&r2) };

match (t1, t2) {
(Repru8::A(_), Repru8::A(_)) => (),
_ => assert!(false)
};
}
35 changes: 35 additions & 0 deletions src/test/ui/print_type_sizes/repr_int_c.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,35 @@
// Copyright 2018 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: -Z print-type-sizes
// compile-pass

// This test makes sure that the tag is not grown for `repr(C)` or `repr(u8)`
// variants (see https://github.com/rust-lang/rust/issues/50098 for the original bug).

#![feature(start)]
#![allow(dead_code)]

#[repr(C, u8)]
enum ReprCu8 {
A(u16),
B,
}

#[repr(u8)]
enum Repru8 {
A(u16),
B,
}

#[start]
fn start(_: isize, _: *const *const u8) -> isize {
0
}
12 changes: 12 additions & 0 deletions src/test/ui/print_type_sizes/repr_int_c.stdout
Original file line number Diff line number Diff line change
@@ -0,0 +1,12 @@
print-type-size type: `ReprCu8`: 4 bytes, alignment: 2 bytes
print-type-size discriminant: 1 bytes
print-type-size variant `A`: 3 bytes
print-type-size padding: 1 bytes
print-type-size field `.0`: 2 bytes, alignment: 2 bytes
print-type-size variant `B`: 1 bytes
print-type-size type: `Repru8`: 4 bytes, alignment: 2 bytes
print-type-size discriminant: 1 bytes
print-type-size variant `A`: 3 bytes
print-type-size padding: 1 bytes
print-type-size field `.0`: 2 bytes, alignment: 2 bytes
print-type-size variant `B`: 0 bytes

0 comments on commit 3eadd75

Please sign in to comment.