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 \
|
|
|
|
tracing garbage collector")]
|
|
|
|
|
2015-02-07 18:49:54 -05:00
|
|
|
use core::{isize, usize};
|
|
|
|
|
|
|
|
#[inline(always)]
|
|
|
|
fn check_size_and_alignment(size: usize, align: usize) {
|
|
|
|
debug_assert!(size != 0);
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
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);
|
2014-06-13 23:23:31 -07:00
|
|
|
imp::allocate(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.
|
|
|
|
///
|
|
|
|
/// 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);
|
2014-10-08 00:57:29 -04:00
|
|
|
imp::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-02-09 10:00:46 +03: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);
|
2014-10-08 00:57:29 -04:00
|
|
|
imp::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) {
|
2014-10-24 20:11:28 -04:00
|
|
|
imp::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 {
|
2014-06-13 23:23:31 -07:00
|
|
|
imp::usable_size(size, align)
|
2014-04-25 02:19:34 -04:00
|
|
|
}
|
2014-05-06 22:03:14 -04:00
|
|
|
|
2014-08-04 22:48:39 +12:00
|
|
|
/// Prints implementation-defined allocator statistics.
|
2014-05-11 17:41:15 -04:00
|
|
|
///
|
2014-06-13 23:22:58 -07:00
|
|
|
/// These statistics may be inconsistent if other threads use the allocator
|
|
|
|
/// during the call.
|
2014-05-11 17:41:15 -04:00
|
|
|
pub fn stats_print() {
|
2014-06-13 23:23:31 -07:00
|
|
|
imp::stats_print();
|
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.
|
|
|
|
///
|
|
|
|
/// 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.
|
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);
|
|
|
|
if ptr.is_null() { ::oom() }
|
|
|
|
ptr
|
2014-05-06 22:03:14 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-05-29 17:32:50 -07:00
|
|
|
#[cfg(not(test))]
|
2015-05-09 14:50:28 -05:00
|
|
|
#[lang = "exchange_free"]
|
2014-05-21 00:18:10 -04:00
|
|
|
#[inline]
|
2015-02-09 10:00:46 +03:00
|
|
|
unsafe fn exchange_free(ptr: *mut u8, old_size: usize, align: usize) {
|
2014-10-24 20:11:28 -04:00
|
|
|
deallocate(ptr, old_size, align);
|
2014-05-21 00:18:10 -04:00
|
|
|
}
|
|
|
|
|
2014-09-08 01:46:33 -04:00
|
|
|
// The minimum alignment guaranteed by the architecture. This value is used to
|
|
|
|
// add fast paths for low alignment values. In practice, the alignment is a
|
|
|
|
// constant at the call site and the branch will be optimized out.
|
2014-12-30 06:13:08 +01:00
|
|
|
#[cfg(all(not(feature = "external_funcs"),
|
|
|
|
not(feature = "external_crate"),
|
|
|
|
any(target_arch = "arm",
|
|
|
|
target_arch = "mips",
|
2015-01-09 19:54:55 -08:00
|
|
|
target_arch = "mipsel",
|
|
|
|
target_arch = "powerpc")))]
|
2015-02-09 10:00:46 +03:00
|
|
|
const MIN_ALIGN: usize = 8;
|
2014-12-30 06:13:08 +01:00
|
|
|
#[cfg(all(not(feature = "external_funcs"),
|
|
|
|
not(feature = "external_crate"),
|
|
|
|
any(target_arch = "x86",
|
|
|
|
target_arch = "x86_64",
|
2015-01-07 18:53:58 -08:00
|
|
|
target_arch = "aarch64")))]
|
2015-02-09 10:00:46 +03:00
|
|
|
const MIN_ALIGN: usize = 16;
|
2014-09-08 01:46:33 -04:00
|
|
|
|
2014-12-26 23:28:04 +01:00
|
|
|
#[cfg(feature = "external_funcs")]
|
2014-12-04 15:02:59 -05:00
|
|
|
mod imp {
|
2015-04-16 21:34:12 -05:00
|
|
|
#[allow(improper_ctypes)]
|
2014-12-04 15:02:59 -05:00
|
|
|
extern {
|
2015-02-09 10:00:46 +03:00
|
|
|
fn rust_allocate(size: usize, align: usize) -> *mut u8;
|
|
|
|
fn rust_deallocate(ptr: *mut u8, old_size: usize, align: usize);
|
|
|
|
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;
|
|
|
|
fn rust_usable_size(size: usize, align: usize) -> usize;
|
2014-12-04 15:02:59 -05:00
|
|
|
fn rust_stats_print();
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
2015-02-09 10:00:46 +03:00
|
|
|
pub unsafe fn allocate(size: usize, align: usize) -> *mut u8 {
|
2014-12-04 15:02:59 -05:00
|
|
|
rust_allocate(size, align)
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
2015-02-09 10:00:46 +03:00
|
|
|
pub unsafe fn deallocate(ptr: *mut u8, old_size: usize, align: usize) {
|
2014-12-27 00:34:51 +01:00
|
|
|
rust_deallocate(ptr, old_size, align)
|
2014-12-04 15:02:59 -05: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 {
|
2014-12-27 00:34:51 +01:00
|
|
|
rust_reallocate(ptr, old_size, size, align)
|
2014-12-04 15:02:59 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
2015-02-09 10:00:46 +03:00
|
|
|
pub unsafe fn reallocate_inplace(ptr: *mut u8, old_size: usize, size: usize,
|
|
|
|
align: usize) -> usize {
|
2014-12-04 15:02:59 -05:00
|
|
|
rust_reallocate_inplace(ptr, old_size, size, align)
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
2015-02-09 10:00:46 +03:00
|
|
|
pub fn usable_size(size: usize, align: usize) -> usize {
|
2014-12-04 15:02:59 -05:00
|
|
|
unsafe { rust_usable_size(size, align) }
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
|
|
|
pub fn stats_print() {
|
|
|
|
unsafe { rust_stats_print() }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-12-26 23:28:04 +01:00
|
|
|
#[cfg(feature = "external_crate")]
|
2014-12-04 15:02:59 -05:00
|
|
|
mod imp {
|
|
|
|
extern crate external;
|
|
|
|
pub use self::external::{allocate, deallocate, reallocate_inplace, reallocate};
|
|
|
|
pub use self::external::{usable_size, stats_print};
|
|
|
|
}
|
|
|
|
|
2014-12-27 00:09:04 +01:00
|
|
|
#[cfg(all(not(feature = "external_funcs"),
|
|
|
|
not(feature = "external_crate"),
|
|
|
|
jemalloc))]
|
2014-06-13 23:23:31 -07:00
|
|
|
mod imp {
|
2014-11-28 11:57:41 -05:00
|
|
|
use core::option::Option;
|
|
|
|
use core::option::Option::None;
|
2014-10-28 17:06:06 -04:00
|
|
|
use core::ptr::{null_mut, null};
|
2014-06-13 23:23:31 -07:00
|
|
|
use libc::{c_char, c_int, c_void, size_t};
|
2014-09-08 01:46:33 -04:00
|
|
|
use super::MIN_ALIGN;
|
2014-06-13 23:23:31 -07:00
|
|
|
|
|
|
|
#[link(name = "jemalloc", kind = "static")]
|
2014-07-30 07:44:20 -07:00
|
|
|
#[cfg(not(test))]
|
|
|
|
extern {}
|
|
|
|
|
2014-06-13 23:23:31 -07:00
|
|
|
extern {
|
2015-03-13 03:19:30 +01:00
|
|
|
#[allocator]
|
2014-06-13 23:23:31 -07:00
|
|
|
fn je_mallocx(size: size_t, flags: c_int) -> *mut c_void;
|
2014-10-28 17:06:06 -04:00
|
|
|
fn je_rallocx(ptr: *mut c_void, size: size_t, flags: c_int) -> *mut c_void;
|
|
|
|
fn je_xallocx(ptr: *mut c_void, size: size_t, extra: size_t, flags: c_int) -> size_t;
|
2014-09-05 10:39:36 -04:00
|
|
|
fn je_sdallocx(ptr: *mut c_void, size: size_t, flags: c_int);
|
2014-06-13 23:23:31 -07:00
|
|
|
fn je_nallocx(size: size_t, flags: c_int) -> size_t;
|
2014-06-25 12:47:34 -07:00
|
|
|
fn je_malloc_stats_print(write_cb: Option<extern "C" fn(cbopaque: *mut c_void,
|
|
|
|
*const c_char)>,
|
2014-06-13 23:23:31 -07:00
|
|
|
cbopaque: *mut c_void,
|
2014-06-25 12:47:34 -07:00
|
|
|
opts: *const c_char);
|
2014-06-13 23:23:31 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
// -lpthread needs to occur after -ljemalloc, the earlier argument isn't enough
|
2015-04-21 15:56:55 -07:00
|
|
|
#[cfg(all(not(windows),
|
|
|
|
not(target_os = "android"),
|
|
|
|
not(target_env = "musl")))]
|
2014-06-13 23:23:31 -07:00
|
|
|
#[link(name = "pthread")]
|
|
|
|
extern {}
|
|
|
|
|
|
|
|
// MALLOCX_ALIGN(a) macro
|
|
|
|
#[inline(always)]
|
2015-02-09 10:00:46 +03:00
|
|
|
fn mallocx_align(a: usize) -> c_int { a.trailing_zeros() as c_int }
|
2014-06-13 23:23:31 -07:00
|
|
|
|
2014-09-08 01:46:33 -04:00
|
|
|
#[inline(always)]
|
2015-02-09 10:00:46 +03:00
|
|
|
fn align_to_flags(align: usize) -> c_int {
|
2014-09-08 01:46:33 -04:00
|
|
|
if align <= MIN_ALIGN { 0 } else { mallocx_align(align) }
|
|
|
|
}
|
|
|
|
|
2014-06-13 23:23:31 -07:00
|
|
|
#[inline]
|
2015-02-09 10:00:46 +03:00
|
|
|
pub unsafe fn allocate(size: usize, align: usize) -> *mut u8 {
|
2014-09-08 01:46:33 -04:00
|
|
|
let flags = align_to_flags(align);
|
2014-10-28 17:06:06 -04:00
|
|
|
je_mallocx(size as size_t, flags) as *mut u8
|
2014-06-13 23:23:31 -07: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 {
|
2014-09-08 01:46:33 -04:00
|
|
|
let flags = align_to_flags(align);
|
2014-10-28 17:06:06 -04:00
|
|
|
je_rallocx(ptr as *mut c_void, size as size_t, flags) as *mut u8
|
2014-06-13 23:23:31 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
2015-02-09 10:00:46 +03:00
|
|
|
pub unsafe fn reallocate_inplace(ptr: *mut u8, _old_size: usize, size: usize,
|
|
|
|
align: usize) -> usize {
|
2014-09-08 01:46:33 -04:00
|
|
|
let flags = align_to_flags(align);
|
2015-02-09 10:00:46 +03:00
|
|
|
je_xallocx(ptr as *mut c_void, size as size_t, 0, flags) as usize
|
2014-06-13 23:23:31 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
2015-02-09 10:00:46 +03:00
|
|
|
pub unsafe fn deallocate(ptr: *mut u8, old_size: usize, align: usize) {
|
2014-09-05 10:39:36 -04:00
|
|
|
let flags = align_to_flags(align);
|
2014-10-24 20:11:28 -04:00
|
|
|
je_sdallocx(ptr as *mut c_void, old_size as size_t, flags)
|
2014-09-05 10:39:36 -04:00
|
|
|
}
|
|
|
|
|
2014-06-13 23:23:31 -07:00
|
|
|
#[inline]
|
2015-02-09 10:00:46 +03:00
|
|
|
pub fn usable_size(size: usize, align: usize) -> usize {
|
2014-09-08 01:46:33 -04:00
|
|
|
let flags = align_to_flags(align);
|
2015-02-09 10:00:46 +03:00
|
|
|
unsafe { je_nallocx(size as size_t, flags) as usize }
|
2014-06-13 23:23:31 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn stats_print() {
|
|
|
|
unsafe {
|
2014-09-14 20:27:36 -07:00
|
|
|
je_malloc_stats_print(None, null_mut(), null())
|
2014-06-13 23:23:31 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-12-27 00:09:04 +01:00
|
|
|
#[cfg(all(not(feature = "external_funcs"),
|
|
|
|
not(feature = "external_crate"),
|
|
|
|
not(jemalloc),
|
|
|
|
unix))]
|
2014-06-13 23:23:31 -07:00
|
|
|
mod imp {
|
2014-08-23 17:29:48 +02:00
|
|
|
use core::cmp;
|
2014-06-13 23:23:31 -07:00
|
|
|
use core::ptr;
|
|
|
|
use libc;
|
2014-09-08 01:46:33 -04:00
|
|
|
use super::MIN_ALIGN;
|
2014-06-13 23:23:31 -07:00
|
|
|
|
|
|
|
extern {
|
|
|
|
fn posix_memalign(memptr: *mut *mut libc::c_void,
|
|
|
|
align: libc::size_t,
|
|
|
|
size: libc::size_t) -> libc::c_int;
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
2015-02-09 10:00:46 +03:00
|
|
|
pub unsafe fn allocate(size: usize, align: usize) -> *mut u8 {
|
2014-09-08 01:46:33 -04:00
|
|
|
if align <= MIN_ALIGN {
|
2014-10-28 17:06:06 -04:00
|
|
|
libc::malloc(size as libc::size_t) as *mut u8
|
2014-06-13 23:23:31 -07:00
|
|
|
} else {
|
2015-01-21 09:23:27 -08:00
|
|
|
let mut out = ptr::null_mut();
|
2014-06-13 23:23:31 -07:00
|
|
|
let ret = posix_memalign(&mut out,
|
|
|
|
align as libc::size_t,
|
|
|
|
size as libc::size_t);
|
|
|
|
if ret != 0 {
|
2014-10-28 17:06:06 -04:00
|
|
|
ptr::null_mut()
|
|
|
|
} else {
|
|
|
|
out as *mut u8
|
2014-06-13 23:23:31 -07: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 {
|
2014-09-08 01:46:33 -04:00
|
|
|
if align <= MIN_ALIGN {
|
2014-10-28 17:06:06 -04:00
|
|
|
libc::realloc(ptr as *mut libc::c_void, size as libc::size_t) as *mut u8
|
2014-09-08 01:46:33 -04:00
|
|
|
} else {
|
|
|
|
let new_ptr = allocate(size, align);
|
2015-03-27 11:12:28 -07:00
|
|
|
ptr::copy(ptr, new_ptr, cmp::min(size, old_size));
|
2014-09-08 01:46:33 -04:00
|
|
|
deallocate(ptr, old_size, align);
|
|
|
|
new_ptr
|
|
|
|
}
|
2014-06-13 23:23:31 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
2015-02-09 10:00:46 +03:00
|
|
|
pub unsafe fn reallocate_inplace(_ptr: *mut u8, old_size: usize, _size: usize,
|
|
|
|
_align: usize) -> usize {
|
2014-10-24 19:58:26 -04:00
|
|
|
old_size
|
2014-06-13 23:23:31 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
2015-02-09 10:00:46 +03:00
|
|
|
pub unsafe fn deallocate(ptr: *mut u8, _old_size: usize, _align: usize) {
|
2014-06-13 23:23:31 -07:00
|
|
|
libc::free(ptr as *mut libc::c_void)
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
2015-02-09 10:00:46 +03:00
|
|
|
pub fn usable_size(size: usize, _align: usize) -> usize {
|
2014-06-13 23:23:31 -07:00
|
|
|
size
|
|
|
|
}
|
|
|
|
|
2014-09-08 01:46:33 -04:00
|
|
|
pub fn stats_print() {}
|
2014-06-13 23:23:31 -07:00
|
|
|
}
|
|
|
|
|
2014-12-27 00:09:04 +01:00
|
|
|
#[cfg(all(not(feature = "external_funcs"),
|
|
|
|
not(feature = "external_crate"),
|
|
|
|
not(jemalloc),
|
|
|
|
windows))]
|
2014-06-13 23:23:31 -07:00
|
|
|
mod imp {
|
2015-06-13 15:13:16 -04:00
|
|
|
use core::mem::size_of;
|
|
|
|
use libc::{BOOL, DWORD, HANDLE, LPVOID, SIZE_T, INVALID_HANDLE_VALUE};
|
|
|
|
use libc::{WriteFile};
|
2014-09-08 01:46:33 -04:00
|
|
|
use super::MIN_ALIGN;
|
2014-06-13 23:23:31 -07:00
|
|
|
|
2015-06-13 15:13:16 -04:00
|
|
|
extern "system" {
|
|
|
|
fn GetProcessHeap() -> HANDLE;
|
|
|
|
fn GetStdHandle(nStdHandle: DWORD) -> HANDLE;
|
|
|
|
fn HeapAlloc(hHeap: HANDLE, dwFlags: DWORD, dwBytes: SIZE_T) -> LPVOID;
|
|
|
|
fn HeapReAlloc(hHeap: HANDLE, dwFlags: DWORD, lpMem: LPVOID, dwBytes: SIZE_T) -> LPVOID;
|
|
|
|
fn HeapFree(hHeap: HANDLE, dwFlags: DWORD, lpMem: LPVOID) -> BOOL;
|
|
|
|
fn HeapSummary(hHeap: HANDLE, dwFlags: DWORD, lpSummary: LPHEAP_SUMMARY) -> BOOL;
|
|
|
|
}
|
|
|
|
|
|
|
|
#[repr(C)] #[allow(non_snake_case)]
|
|
|
|
struct HEAP_SUMMARY {
|
|
|
|
cb: DWORD,
|
|
|
|
cbAllocated: SIZE_T,
|
|
|
|
cbCommitted: SIZE_T,
|
|
|
|
cbReserved: SIZE_T,
|
|
|
|
cbMaxReserve: SIZE_T,
|
|
|
|
}
|
|
|
|
#[allow(non_camel_case_types)]
|
|
|
|
type LPHEAP_SUMMARY = *mut HEAP_SUMMARY;
|
|
|
|
|
|
|
|
#[repr(C)]
|
|
|
|
struct Header(*mut u8);
|
|
|
|
|
|
|
|
const HEAP_REALLOC_IN_PLACE_ONLY: DWORD = 0x00000010;
|
|
|
|
const STD_OUTPUT_HANDLE: DWORD = -11i32 as u32;
|
|
|
|
|
|
|
|
#[inline]
|
|
|
|
unsafe fn get_header<'a>(ptr: *mut u8) -> &'a mut Header {
|
|
|
|
&mut *(ptr as *mut Header).offset(-1)
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
|
|
|
unsafe fn align_ptr(ptr: *mut u8, align: usize) -> *mut u8 {
|
|
|
|
let aligned = ptr.offset((align - (ptr as usize & (align - 1))) as isize);
|
|
|
|
*get_header(aligned) = Header(ptr);
|
|
|
|
aligned
|
2014-06-13 23:23:31 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
2015-02-09 10:00:46 +03:00
|
|
|
pub unsafe fn allocate(size: usize, align: usize) -> *mut u8 {
|
2014-09-08 01:46:33 -04:00
|
|
|
if align <= MIN_ALIGN {
|
2015-06-13 15:13:16 -04:00
|
|
|
HeapAlloc(GetProcessHeap(), 0, size as SIZE_T) as *mut u8
|
2014-09-08 01:46:33 -04:00
|
|
|
} else {
|
2015-06-13 15:13:16 -04:00
|
|
|
let ptr = HeapAlloc(GetProcessHeap(), 0, (size + align) as SIZE_T) as *mut u8;
|
|
|
|
if ptr.is_null() { return ptr }
|
|
|
|
align_ptr(ptr, align)
|
2014-06-13 23:23:31 -07: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 {
|
2014-09-08 01:46:33 -04:00
|
|
|
if align <= MIN_ALIGN {
|
2015-06-13 15:13:16 -04:00
|
|
|
HeapReAlloc(GetProcessHeap(), 0, ptr as LPVOID, size as SIZE_T) as *mut u8
|
2014-09-08 01:46:33 -04:00
|
|
|
} else {
|
2015-06-13 15:13:16 -04:00
|
|
|
let header = get_header(ptr);
|
|
|
|
let new = HeapReAlloc(GetProcessHeap(), 0, header.0 as LPVOID,
|
|
|
|
(size + align) as SIZE_T) as *mut u8;
|
|
|
|
if new.is_null() { return new }
|
|
|
|
align_ptr(new, align)
|
2014-06-13 23:23:31 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
2015-06-13 15:13:16 -04:00
|
|
|
pub unsafe fn reallocate_inplace(ptr: *mut u8, old_size: usize, size: usize,
|
|
|
|
align: usize) -> usize {
|
|
|
|
if align <= MIN_ALIGN {
|
|
|
|
let new = HeapReAlloc(GetProcessHeap(), HEAP_REALLOC_IN_PLACE_ONLY, ptr as LPVOID,
|
|
|
|
size as SIZE_T) as *mut u8;
|
|
|
|
if new.is_null() { old_size } else { size }
|
|
|
|
} else {
|
|
|
|
old_size
|
|
|
|
}
|
2014-06-13 23:23:31 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
2015-02-09 10:00:46 +03:00
|
|
|
pub unsafe fn deallocate(ptr: *mut u8, _old_size: usize, align: usize) {
|
2014-09-08 01:46:33 -04:00
|
|
|
if align <= MIN_ALIGN {
|
2015-06-13 15:13:16 -04:00
|
|
|
let err = HeapFree(GetProcessHeap(), 0, ptr as LPVOID);
|
|
|
|
debug_assert!(err != 0);
|
2014-09-08 01:46:33 -04:00
|
|
|
} else {
|
2015-06-13 15:13:16 -04:00
|
|
|
let header = get_header(ptr);
|
|
|
|
let err = HeapFree(GetProcessHeap(), 0, header.0 as LPVOID);
|
|
|
|
debug_assert!(err != 0);
|
2014-09-08 01:46:33 -04:00
|
|
|
}
|
2014-06-13 23:23:31 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
2015-02-09 10:00:46 +03:00
|
|
|
pub fn usable_size(size: usize, _align: usize) -> usize {
|
2014-06-13 23:23:31 -07:00
|
|
|
size
|
|
|
|
}
|
|
|
|
|
2015-06-13 15:13:16 -04:00
|
|
|
pub fn stats_print() {
|
|
|
|
use core::fmt::{Error, Result, Write};
|
|
|
|
use core::ptr::null_mut;
|
|
|
|
use core::raw::Repr;
|
|
|
|
use core::result::Result::{Ok, Err};
|
|
|
|
struct Console(HANDLE);
|
|
|
|
impl Write for Console {
|
|
|
|
fn write_str(&mut self, s: &str) -> Result {
|
|
|
|
let repr = s.repr();
|
|
|
|
let mut written = 0;
|
|
|
|
let err = unsafe { WriteFile(self.0, repr.data as LPVOID, repr.len as DWORD,
|
|
|
|
&mut written, null_mut()) };
|
|
|
|
if written as usize != repr.len { return Err(Error) }
|
|
|
|
if err == 0 { return Err(Error) }
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
let mut hs = HEAP_SUMMARY {
|
|
|
|
cb: size_of::<HEAP_SUMMARY>() as DWORD,
|
|
|
|
cbAllocated: 0,
|
|
|
|
cbCommitted: 0,
|
|
|
|
cbReserved: 0,
|
|
|
|
cbMaxReserve: 0,
|
|
|
|
};
|
|
|
|
let err = unsafe { HeapSummary(GetProcessHeap(), 0, &mut hs) };
|
|
|
|
assert!(err != 0);
|
|
|
|
let handle = unsafe { GetStdHandle(STD_OUTPUT_HANDLE) };
|
|
|
|
if handle.is_null() || handle == INVALID_HANDLE_VALUE { panic!("Failed to open stdout") }
|
|
|
|
let mut out = Console(handle);
|
|
|
|
writeln!(&mut out, "Allocated: {}", hs.cbAllocated).unwrap();
|
|
|
|
writeln!(&mut out, "Committed: {}", hs.cbCommitted).unwrap();
|
|
|
|
writeln!(&mut out, "Reserved: {}", hs.cbReserved).unwrap();
|
|
|
|
writeln!(&mut out, "MaxReserve: {}", hs.cbMaxReserve).unwrap();
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn alignment_header_size() {
|
|
|
|
assert!(size_of::<Header>() <= MIN_ALIGN);
|
|
|
|
}
|
2014-06-13 23:23:31 -07: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
|
|
|
|
2014-10-02 03:29:39 -04:00
|
|
|
#[test]
|
|
|
|
fn basic_reallocate_inplace_noop() {
|
|
|
|
unsafe {
|
|
|
|
let size = 4000;
|
|
|
|
let ptr = heap::allocate(size, 8);
|
2014-10-28 17:06:06 -04: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
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|