2014-12-12 23:39:27 +00:00
|
|
|
// Copyright 2014-2015 The Rust Project Developers. See the COPYRIGHT
|
2014-04-25 02:19:34 -04:00
|
|
|
// 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.
|
|
|
|
|
2015-06-09 11:52:41 -07:00
|
|
|
#![unstable(feature = "heap_api",
|
|
|
|
reason = "the precise API and guarantees it provides may be tweaked \
|
|
|
|
slightly, especially to possibly take into account the \
|
|
|
|
types being stored to make room for a future \
|
2015-08-12 22:19:08 -07:00
|
|
|
tracing garbage collector",
|
|
|
|
issue = "27700")]
|
2015-06-09 11:52:41 -07:00
|
|
|
|
2015-02-07 18:49:54 -05:00
|
|
|
use core::{isize, usize};
|
2016-01-28 23:59:00 +02:00
|
|
|
#[cfg(not(test))]
|
2016-11-11 11:55:47 +01:00
|
|
|
use core::intrinsics::{min_align_of_val, size_of_val};
|
2015-02-07 18:49:54 -05:00
|
|
|
|
2015-06-25 10:07:01 -07:00
|
|
|
#[allow(improper_ctypes)]
|
2015-11-23 15:32:40 +13:00
|
|
|
extern "C" {
|
2015-06-25 10:07:01 -07:00
|
|
|
#[allocator]
|
|
|
|
fn __rust_allocate(size: usize, align: usize) -> *mut u8;
|
2017-03-09 17:53:01 -08:00
|
|
|
fn __rust_allocate_zeroed(size: usize, align: usize) -> *mut u8;
|
2015-06-25 10:07:01 -07:00
|
|
|
fn __rust_deallocate(ptr: *mut u8, old_size: usize, align: usize);
|
2015-09-24 10:00:54 +12:00
|
|
|
fn __rust_reallocate(ptr: *mut u8, old_size: usize, size: usize, align: usize) -> *mut u8;
|
|
|
|
fn __rust_reallocate_inplace(ptr: *mut u8,
|
|
|
|
old_size: usize,
|
|
|
|
size: usize,
|
|
|
|
align: usize)
|
|
|
|
-> usize;
|
2015-06-25 10:07:01 -07:00
|
|
|
fn __rust_usable_size(size: usize, align: usize) -> usize;
|
|
|
|
}
|
|
|
|
|
2015-02-07 18:49:54 -05:00
|
|
|
#[inline(always)]
|
|
|
|
fn check_size_and_alignment(size: usize, align: usize) {
|
|
|
|
debug_assert!(size != 0);
|
2015-09-24 11:32:01 +12:00
|
|
|
debug_assert!(size <= isize::MAX as usize,
|
|
|
|
"Tried to allocate too much: {} bytes",
|
|
|
|
size);
|
|
|
|
debug_assert!(usize::is_power_of_two(align),
|
|
|
|
"Invalid alignment of allocation: {}",
|
|
|
|
align);
|
2015-02-07 18:49:54 -05:00
|
|
|
}
|
|
|
|
|
2014-06-13 23:22:58 -07:00
|
|
|
// FIXME: #13996: mark the `allocate` and `reallocate` return value as `noalias`
|
2014-05-06 17:01:16 -04:00
|
|
|
|
2014-10-28 17:06:06 -04:00
|
|
|
/// Return a pointer to `size` bytes of memory aligned to `align`.
|
|
|
|
///
|
|
|
|
/// On failure, return a null pointer.
|
2014-04-25 02:19:34 -04:00
|
|
|
///
|
2014-06-13 23:22:58 -07:00
|
|
|
/// Behavior is undefined if the requested size is 0 or the alignment is not a
|
|
|
|
/// power of 2. The alignment must be no larger than the largest supported page
|
|
|
|
/// size on the platform.
|
2014-04-25 02:19:34 -04:00
|
|
|
#[inline]
|
2015-02-09 10:00:46 +03:00
|
|
|
pub unsafe fn allocate(size: usize, align: usize) -> *mut u8 {
|
2015-02-07 18:49:54 -05:00
|
|
|
check_size_and_alignment(size, align);
|
2015-06-25 10:07:01 -07:00
|
|
|
__rust_allocate(size, align)
|
2014-04-25 02:19:34 -04:00
|
|
|
}
|
|
|
|
|
2017-03-09 17:53:01 -08:00
|
|
|
/// Return a pointer to `size` bytes of memory aligned to `align` and
|
|
|
|
/// initialized to zeroes.
|
|
|
|
///
|
|
|
|
/// On failure, return a null pointer.
|
|
|
|
///
|
|
|
|
/// Behavior is undefined if the requested size is 0 or the alignment is not a
|
|
|
|
/// power of 2. The alignment must be no larger than the largest supported page
|
|
|
|
/// size on the platform.
|
|
|
|
#[inline]
|
|
|
|
pub unsafe fn allocate_zeroed(size: usize, align: usize) -> *mut u8 {
|
|
|
|
check_size_and_alignment(size, align);
|
|
|
|
__rust_allocate_zeroed(size, align)
|
|
|
|
}
|
|
|
|
|
2014-10-28 17:06:06 -04:00
|
|
|
/// Resize the allocation referenced by `ptr` to `size` bytes.
|
|
|
|
///
|
|
|
|
/// On failure, return a null pointer and leave the original allocation intact.
|
2014-04-25 02:19:34 -04:00
|
|
|
///
|
2015-03-20 15:22:57 -04:00
|
|
|
/// If the allocation was relocated, the memory at the passed-in pointer is
|
|
|
|
/// undefined after the call.
|
|
|
|
///
|
2014-06-13 23:22:58 -07:00
|
|
|
/// Behavior is undefined if the requested size is 0 or the alignment is not a
|
|
|
|
/// power of 2. The alignment must be no larger than the largest supported page
|
|
|
|
/// size on the platform.
|
2014-04-25 02:19:34 -04:00
|
|
|
///
|
2014-06-13 23:22:58 -07:00
|
|
|
/// The `old_size` and `align` parameters are the parameters that were used to
|
2014-10-24 20:11:28 -04:00
|
|
|
/// create the allocation referenced by `ptr`. The `old_size` parameter may be
|
|
|
|
/// any value in range_inclusive(requested_size, usable_size).
|
2014-04-25 02:19:34 -04:00
|
|
|
#[inline]
|
2015-02-09 10:00:46 +03:00
|
|
|
pub unsafe fn reallocate(ptr: *mut u8, old_size: usize, size: usize, align: usize) -> *mut u8 {
|
2015-02-07 18:49:54 -05:00
|
|
|
check_size_and_alignment(size, align);
|
2015-06-25 10:07:01 -07:00
|
|
|
__rust_reallocate(ptr, old_size, size, align)
|
2014-04-25 02:19:34 -04:00
|
|
|
}
|
|
|
|
|
2014-10-28 17:06:06 -04:00
|
|
|
/// Resize the allocation referenced by `ptr` to `size` bytes.
|
2014-04-25 02:19:34 -04:00
|
|
|
///
|
2014-10-24 19:58:26 -04:00
|
|
|
/// If the operation succeeds, it returns `usable_size(size, align)` and if it
|
|
|
|
/// fails (or is a no-op) it returns `usable_size(old_size, align)`.
|
2014-04-25 02:19:34 -04:00
|
|
|
///
|
2014-06-13 23:22:58 -07:00
|
|
|
/// Behavior is undefined if the requested size is 0 or the alignment is not a
|
|
|
|
/// power of 2. The alignment must be no larger than the largest supported page
|
|
|
|
/// size on the platform.
|
2014-04-25 02:19:34 -04:00
|
|
|
///
|
|
|
|
/// The `old_size` and `align` parameters are the parameters that were used to
|
|
|
|
/// create the allocation referenced by `ptr`. The `old_size` parameter may be
|
|
|
|
/// any value in range_inclusive(requested_size, usable_size).
|
|
|
|
#[inline]
|
2015-09-24 10:00:54 +12:00
|
|
|
pub unsafe fn reallocate_inplace(ptr: *mut u8,
|
|
|
|
old_size: usize,
|
|
|
|
size: usize,
|
|
|
|
align: usize)
|
|
|
|
-> usize {
|
2015-02-07 18:49:54 -05:00
|
|
|
check_size_and_alignment(size, align);
|
2015-06-25 10:07:01 -07:00
|
|
|
__rust_reallocate_inplace(ptr, old_size, size, align)
|
2014-04-25 02:19:34 -04:00
|
|
|
}
|
|
|
|
|
2014-08-04 22:48:39 +12:00
|
|
|
/// Deallocates the memory referenced by `ptr`.
|
2014-04-25 02:19:34 -04:00
|
|
|
///
|
|
|
|
/// The `ptr` parameter must not be null.
|
|
|
|
///
|
2014-10-24 20:11:28 -04:00
|
|
|
/// The `old_size` and `align` parameters are the parameters that were used to
|
|
|
|
/// create the allocation referenced by `ptr`. The `old_size` parameter may be
|
|
|
|
/// any value in range_inclusive(requested_size, usable_size).
|
2014-04-25 02:19:34 -04:00
|
|
|
#[inline]
|
2015-02-09 10:00:46 +03:00
|
|
|
pub unsafe fn deallocate(ptr: *mut u8, old_size: usize, align: usize) {
|
2015-06-25 10:07:01 -07:00
|
|
|
__rust_deallocate(ptr, old_size, align)
|
2014-04-25 02:19:34 -04:00
|
|
|
}
|
|
|
|
|
2014-08-04 22:48:39 +12:00
|
|
|
/// Returns the usable size of an allocation created with the specified the
|
2014-06-13 23:22:58 -07:00
|
|
|
/// `size` and `align`.
|
2014-04-25 02:19:34 -04:00
|
|
|
#[inline]
|
2015-02-09 10:00:46 +03:00
|
|
|
pub fn usable_size(size: usize, align: usize) -> usize {
|
2015-06-25 10:07:01 -07:00
|
|
|
unsafe { __rust_usable_size(size, align) }
|
2014-05-11 17:41:15 -04:00
|
|
|
}
|
|
|
|
|
2014-09-15 15:37:01 -04:00
|
|
|
/// An arbitrary non-null address to represent zero-size allocations.
|
|
|
|
///
|
2015-06-25 10:07:01 -07:00
|
|
|
/// This preserves the non-null invariant for types like `Box<T>`. The address
|
|
|
|
/// may overlap with non-zero-size memory allocations.
|
2014-10-06 16:32:00 -07:00
|
|
|
pub const EMPTY: *mut () = 0x1 as *mut ();
|
2014-05-19 23:19:56 -07:00
|
|
|
|
2014-05-06 22:03:14 -04:00
|
|
|
/// The allocator for unique pointers.
|
2016-12-15 18:00:19 -07:00
|
|
|
// This function must not unwind. If it does, MIR trans will fail.
|
2014-05-12 02:51:00 -04:00
|
|
|
#[cfg(not(test))]
|
2015-05-09 14:50:28 -05:00
|
|
|
#[lang = "exchange_malloc"]
|
2014-05-06 22:03:14 -04:00
|
|
|
#[inline]
|
2015-02-09 10:00:46 +03:00
|
|
|
unsafe fn exchange_malloc(size: usize, align: usize) -> *mut u8 {
|
2014-05-06 22:03:14 -04:00
|
|
|
if size == 0 {
|
2014-09-15 15:37:01 -04:00
|
|
|
EMPTY as *mut u8
|
2014-05-06 22:03:14 -04:00
|
|
|
} else {
|
2014-10-28 17:06:06 -04:00
|
|
|
let ptr = allocate(size, align);
|
2015-09-24 10:00:54 +12:00
|
|
|
if ptr.is_null() {
|
|
|
|
::oom()
|
|
|
|
}
|
2014-10-28 17:06:06 -04:00
|
|
|
ptr
|
2014-05-06 22:03:14 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-01-28 23:59:00 +02:00
|
|
|
#[cfg(not(test))]
|
|
|
|
#[lang = "box_free"]
|
|
|
|
#[inline]
|
2016-11-11 11:55:47 +01:00
|
|
|
unsafe fn box_free<T: ?Sized>(ptr: *mut T) {
|
|
|
|
let size = size_of_val(&*ptr);
|
|
|
|
let align = min_align_of_val(&*ptr);
|
2016-01-28 23:59:00 +02:00
|
|
|
// We do not allocate for Box<T> when T is ZST, so deallocation is also not necessary.
|
|
|
|
if size != 0 {
|
2016-11-11 11:55:47 +01:00
|
|
|
deallocate(ptr as *mut u8, size, align);
|
2016-01-28 23:59:00 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-05-06 22:03:14 -04:00
|
|
|
#[cfg(test)]
|
2015-04-24 17:30:41 +02:00
|
|
|
mod tests {
|
2014-05-06 22:03:14 -04:00
|
|
|
extern crate test;
|
|
|
|
use self::test::Bencher;
|
2015-02-17 21:41:32 +01:00
|
|
|
use boxed::Box;
|
2014-10-02 15:12:58 -07:00
|
|
|
use heap;
|
2014-05-06 22:03:14 -04:00
|
|
|
|
2017-03-09 17:53:01 -08:00
|
|
|
#[test]
|
|
|
|
fn allocate_zeroed() {
|
|
|
|
unsafe {
|
|
|
|
let size = 1024;
|
|
|
|
let ptr = heap::allocate_zeroed(size, 1);
|
|
|
|
if ptr.is_null() {
|
|
|
|
::oom()
|
|
|
|
}
|
|
|
|
|
|
|
|
let end = ptr.offset(size as isize);
|
|
|
|
let mut i = ptr;
|
|
|
|
while i < end {
|
|
|
|
assert_eq!(*i, 0);
|
|
|
|
i = i.offset(1);
|
|
|
|
}
|
|
|
|
heap::deallocate(ptr, size, 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-10-02 03:29:39 -04:00
|
|
|
#[test]
|
|
|
|
fn basic_reallocate_inplace_noop() {
|
|
|
|
unsafe {
|
|
|
|
let size = 4000;
|
|
|
|
let ptr = heap::allocate(size, 8);
|
2015-09-24 10:00:54 +12:00
|
|
|
if ptr.is_null() {
|
|
|
|
::oom()
|
|
|
|
}
|
2014-10-08 00:57:29 -04:00
|
|
|
let ret = heap::reallocate_inplace(ptr, size, size, 8);
|
2014-10-02 03:29:39 -04:00
|
|
|
heap::deallocate(ptr, size, 8);
|
2014-10-24 19:58:26 -04:00
|
|
|
assert_eq!(ret, heap::usable_size(size, 8));
|
2014-10-02 03:29:39 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-05-06 22:03:14 -04:00
|
|
|
#[bench]
|
|
|
|
fn alloc_owned_small(b: &mut Bencher) {
|
|
|
|
b.iter(|| {
|
2015-02-17 21:41:32 +01:00
|
|
|
let _: Box<_> = box 10;
|
2014-05-06 22:03:14 -04:00
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|