Skip to content

Commit dd570d6

Browse files
committed
For size of struct P<T>(Q<T>), don't double-count the prefix added by Q.
Fix #36278. Fix #36294.
1 parent 0b02ae0 commit dd570d6

File tree

2 files changed

+40
-1
lines changed

2 files changed

+40
-1
lines changed

src/librustc_trans/glue.rs

+12-1
Original file line numberDiff line numberDiff line change
@@ -348,9 +348,20 @@ pub fn size_and_align_of_dst<'blk, 'tcx>(bcx: &BlockAndBuilder<'blk, 'tcx>,
348348
let layout = ccx.layout_of(t);
349349
debug!("DST {} layout: {:?}", t, layout);
350350

351+
// Returns size in bytes of all fields except the last one
352+
// (we will be recursing on the last one).
353+
fn local_prefix_bytes(variant: &ty::layout::Struct) -> u64 {
354+
let fields = variant.offset_after_field.len();
355+
if fields > 1 {
356+
variant.offset_after_field[fields - 2].bytes()
357+
} else {
358+
0
359+
}
360+
}
361+
351362
let (sized_size, sized_align) = match *layout {
352363
ty::layout::Layout::Univariant { ref variant, .. } => {
353-
(variant.min_size().bytes(), variant.align.abi())
364+
(local_prefix_bytes(variant), variant.align.abi())
354365
}
355366
_ => {
356367
bug!("size_and_align_of_dst: expcted Univariant for `{}`, found {:#?}",
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,28 @@
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+
// Issue 36278: On an unsized struct with >1 level of nontrivial
12+
// nesting, ensure we are computing dynamic size of prefix correctly.
13+
14+
use std::mem;
15+
16+
const SZ: usize = 100;
17+
struct P<T: ?Sized>([u8; SZ], T);
18+
19+
type Ack<T> = P<P<T>>;
20+
21+
fn main() {
22+
let size_of_sized; let size_of_unsized;
23+
let x: Box<Ack<[u8; 0]>> = Box::new(P([0; SZ], P([0; SZ], [0; 0])));
24+
size_of_sized = mem::size_of_val::<Ack<_>>(&x);
25+
let y: Box<Ack<[u8 ]>> = x;
26+
size_of_unsized = mem::size_of_val::<Ack<_>>(&y);
27+
assert_eq!(size_of_sized, size_of_unsized);
28+
}

0 commit comments

Comments
 (0)