summaryrefslogtreecommitdiffstats
path: root/src/test/ui/consts/const-size_of-align_of.rs
blob: 0c63dc84a37060632c2d5929bbd87ee432b5d740 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
// run-pass
#![allow(dead_code)]

use std::mem;

// Get around the limitations of CTFE in today's Rust.
const fn choice_u64(c: bool, a: u64, b: u64) -> u64 {
    (-(c as i64) as u64) & a | (-(!c as i64) as u64) & b
}

const fn max_usize(a: usize, b: usize) -> usize {
    choice_u64(a > b, a as u64, b as u64) as usize
}

const fn align_to(size: usize, align: usize) -> usize {
    (size + (align - 1)) & !(align - 1)
}

const fn packed_union_size_of<A, B>() -> usize {
    max_usize(mem::size_of::<A>(), mem::size_of::<B>())
}

const fn union_align_of<A, B>() -> usize {
    max_usize(mem::align_of::<A>(), mem::align_of::<B>())
}

const fn union_size_of<A, B>() -> usize {
    align_to(packed_union_size_of::<A, B>(), union_align_of::<A, B>())
}

macro_rules! fake_union {
    ($name:ident { $a:ty, $b:ty }) => (
        struct $name {
            _align: ([$a; 0], [$b; 0]),
            _bytes: [u8; union_size_of::<$a, $b>()]
        }
    )
}

// Check that we can (poorly) emulate unions by
// calling size_of and align_of at compile-time.
fake_union!(U { u16, [u8; 3] });

fn test(u: U) {
    assert_eq!(mem::size_of_val(&u._bytes), 4);
}

fn main() {
    assert_eq!(mem::size_of::<U>(), 4);
    assert_eq!(mem::align_of::<U>(), 2);
}