2020-08-05 13:38:15 +02:00
|
|
|
#![feature(allocator_api, slice_ptr_get)]
|
2019-02-13 19:47:51 +01:00
|
|
|
|
2019-02-25 09:47:34 +01:00
|
|
|
use std::ptr::NonNull;
|
2020-12-04 23:52:13 +01:00
|
|
|
use std::alloc::{Global, Allocator, Layout, System};
|
2019-04-21 13:43:53 +02:00
|
|
|
use std::slice;
|
|
|
|
|
2020-12-04 23:52:13 +01:00
|
|
|
fn check_alloc<T: Allocator>(allocator: T) { unsafe {
|
2019-06-30 21:08:17 +02:00
|
|
|
for &align in &[4, 8, 16, 32] {
|
2020-08-27 09:27:58 +02:00
|
|
|
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();
|
2019-06-30 21:08:17 +02:00
|
|
|
|
|
|
|
for _ in 0..32 {
|
2020-12-04 23:52:13 +01:00
|
|
|
let a = allocator.allocate(layout_20).unwrap().as_non_null_ptr();
|
2020-08-27 09:27:58 +02:00
|
|
|
assert_eq!(a.as_ptr() as usize % layout_20.align(), 0, "pointer is incorrectly aligned");
|
2020-12-04 23:52:13 +01:00
|
|
|
allocator.deallocate(a, layout_20);
|
2019-06-30 21:08:17 +02:00
|
|
|
}
|
|
|
|
|
2020-12-04 23:52:13 +01:00
|
|
|
let p1 = allocator.allocate_zeroed(layout_20).unwrap().as_non_null_ptr();
|
2020-08-27 09:27:58 +02:00
|
|
|
assert_eq!(p1.as_ptr() as usize % layout_20.align(), 0, "pointer is incorrectly aligned");
|
2020-08-05 13:38:15 +02:00
|
|
|
assert_eq!(*p1.as_ptr(), 0);
|
2019-06-30 21:08:17 +02:00
|
|
|
|
2020-04-02 11:45:17 +02:00
|
|
|
// old size < new size
|
2020-08-27 09:27:58 +02:00
|
|
|
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");
|
2019-06-30 21:08:17 +02:00
|
|
|
let slice = slice::from_raw_parts(p2.as_ptr(), 20);
|
|
|
|
assert_eq!(&slice, &[0_u8; 20]);
|
|
|
|
|
|
|
|
// old size == new size
|
2020-08-27 09:27:58 +02:00
|
|
|
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");
|
2019-06-30 21:08:17 +02:00
|
|
|
let slice = slice::from_raw_parts(p3.as_ptr(), 20);
|
|
|
|
assert_eq!(&slice, &[0_u8; 20]);
|
|
|
|
|
|
|
|
// old size > new size
|
2020-08-27 09:27:58 +02:00
|
|
|
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");
|
2019-06-30 21:08:17 +02:00
|
|
|
let slice = slice::from_raw_parts(p4.as_ptr(), 10);
|
|
|
|
assert_eq!(&slice, &[0_u8; 10]);
|
|
|
|
|
2020-12-04 23:52:13 +01:00
|
|
|
allocator.deallocate(p4, layout_10);
|
2019-06-30 21:08:17 +02:00
|
|
|
}
|
2019-04-21 13:43:53 +02:00
|
|
|
} }
|
2019-02-13 19:47:51 +01:00
|
|
|
|
2020-12-04 23:52:13 +01:00
|
|
|
fn check_align_requests<T: Allocator>(allocator: T) {
|
2019-06-30 21:23:48 +02:00
|
|
|
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(|_| {
|
2020-12-04 23:52:13 +01:00
|
|
|
allocator.allocate(Layout::from_size_align(size, align).unwrap()).unwrap().as_non_null_ptr()
|
2019-06-30 21:23:48 +02:00
|
|
|
}).collect();
|
|
|
|
for &ptr in &pointers {
|
|
|
|
assert_eq!((ptr.as_ptr() as usize) % align, 0,
|
|
|
|
"Got a pointer less aligned than requested")
|
|
|
|
}
|
2019-06-30 17:02:20 +02:00
|
|
|
|
2019-06-30 21:23:48 +02:00
|
|
|
// Clean up.
|
|
|
|
for &ptr in &pointers {
|
2020-12-04 23:52:13 +01:00
|
|
|
allocator.deallocate(ptr, Layout::from_size_align(size, align).unwrap())
|
2019-06-30 21:23:48 +02:00
|
|
|
}
|
|
|
|
}
|
2019-02-13 19:47:51 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-02-25 09:47:34 +01:00
|
|
|
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 {
|
2020-12-04 23:52:13 +01:00
|
|
|
let ptr = Global.allocate(l).unwrap().as_non_null_ptr().as_ptr() as *mut T;
|
2019-02-25 09:47:34 +01:00
|
|
|
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);
|
2020-12-04 23:52:13 +01:00
|
|
|
Global.deallocate(NonNull::new(ptr as *mut u8).unwrap(), l);
|
2019-02-25 09:47:34 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-02-13 19:47:51 +01:00
|
|
|
fn main() {
|
2019-04-21 13:43:53 +02:00
|
|
|
check_alloc(System);
|
|
|
|
check_alloc(Global);
|
2019-07-02 13:02:48 +02:00
|
|
|
check_align_requests(System);
|
|
|
|
check_align_requests(Global);
|
2019-02-25 09:47:34 +01:00
|
|
|
global_to_box();
|
|
|
|
box_to_global();
|
2019-02-13 19:47:51 +01:00
|
|
|
}
|