about summary refs log tree commit diff
path: root/src/tools/miri/tests/pass/heap_allocator.rs
blob: 2c38dcb49f1cee993951ea62e4bcaed30bffdcff (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
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
#![feature(allocator_api, slice_ptr_get)]

use std::alloc::{Allocator, Global, Layout, System};
use std::ptr::NonNull;
use std::slice;

fn check_alloc<T: Allocator>(allocator: T) {
    unsafe {
        for &align in &[4, 8, 16, 32] {
            let layout_20 = Layout::from_size_align(20, align).unwrap();
            let layout_40 = Layout::from_size_align(40, 4 * align).unwrap();
            let layout_10 = Layout::from_size_align(10, align / 2).unwrap();

            for _ in 0..32 {
                let a = allocator.allocate(layout_20).unwrap().as_non_null_ptr();
                assert_eq!(
                    a.as_ptr() as usize % layout_20.align(),
                    0,
                    "pointer is incorrectly aligned",
                );
                allocator.deallocate(a, layout_20);
            }

            let p1 = allocator.allocate_zeroed(layout_20).unwrap().as_non_null_ptr();
            assert_eq!(
                p1.as_ptr() as usize % layout_20.align(),
                0,
                "pointer is incorrectly aligned",
            );
            assert_eq!(*p1.as_ptr(), 0);

            // old size < new size
            let p2 = allocator.grow(p1, layout_20, layout_40).unwrap().as_non_null_ptr();
            assert_eq!(
                p2.as_ptr() as usize % layout_40.align(),
                0,
                "pointer is incorrectly aligned",
            );
            let slice = slice::from_raw_parts(p2.as_ptr(), 20);
            assert_eq!(&slice, &[0_u8; 20]);

            // old size == new size
            let p3 = allocator.grow(p2, layout_40, layout_40).unwrap().as_non_null_ptr();
            assert_eq!(
                p3.as_ptr() as usize % layout_40.align(),
                0,
                "pointer is incorrectly aligned",
            );
            let slice = slice::from_raw_parts(p3.as_ptr(), 20);
            assert_eq!(&slice, &[0_u8; 20]);

            // old size > new size
            let p4 = allocator.shrink(p3, layout_40, layout_10).unwrap().as_non_null_ptr();
            assert_eq!(
                p4.as_ptr() as usize % layout_10.align(),
                0,
                "pointer is incorrectly aligned",
            );
            let slice = slice::from_raw_parts(p4.as_ptr(), 10);
            assert_eq!(&slice, &[0_u8; 10]);

            allocator.deallocate(p4, layout_10);
        }
    }
}

fn check_align_requests<T: Allocator>(allocator: T) {
    #[rustfmt::skip] // https://github.com/rust-lang/rustfmt/issues/3255
    for &size in &[2, 8, 64] { // size less than and bigger than alignment
        for &align in &[4, 8, 16, 32] { // Be sure to cover less than and bigger than `MIN_ALIGN` for all architectures
            let iterations = 32;
            unsafe {
                let pointers: Vec<_> = (0..iterations)
                    .map(|_| {
                        allocator
                            .allocate(Layout::from_size_align(size, align).unwrap())
                            .unwrap()
                            .as_non_null_ptr()
                    })
                    .collect();
                for &ptr in &pointers {
                    assert_eq!(
                        (ptr.as_ptr() as usize) % align,
                        0,
                        "Got a pointer less aligned than requested",
                    )
                }

                // Clean up.
                for &ptr in &pointers {
                    allocator.deallocate(ptr, Layout::from_size_align(size, align).unwrap())
                }
            }
        }
    };
}

fn global_to_box() {
    type T = [i32; 4];
    let l = Layout::new::<T>();
    // allocate manually with global allocator, then turn into Box and free there
    unsafe {
        let ptr = Global.allocate(l).unwrap().as_non_null_ptr().as_ptr() as *mut T;
        let b = Box::from_raw(ptr);
        drop(b);
    }
}

fn box_to_global() {
    type T = [i32; 4];
    let l = Layout::new::<T>();
    // allocate with the Box, then deallocate manually with global allocator
    unsafe {
        let b = Box::new(T::default());
        let ptr = Box::into_raw(b);
        Global.deallocate(NonNull::new(ptr as *mut u8).unwrap(), l);
    }
}

fn main() {
    check_alloc(System);
    check_alloc(Global);
    check_align_requests(System);
    check_align_requests(Global);
    global_to_box();
    box_to_global();
}