2013-05-21 20:24:42 -05:00
|
|
|
// Copyright 2012 The Rust Project Developers. See the COPYRIGHT
|
|
|
|
// 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.
|
|
|
|
|
2013-12-12 15:27:26 -06:00
|
|
|
use libc::{c_void, size_t, free, malloc, realloc};
|
2014-01-17 19:45:48 -06:00
|
|
|
use ptr::{RawPtr, mut_null};
|
2014-02-16 01:49:08 -06:00
|
|
|
use intrinsics::abort;
|
2014-02-16 02:04:33 -06:00
|
|
|
use raw;
|
2013-10-16 20:34:01 -05:00
|
|
|
use mem::size_of;
|
2013-05-21 20:24:42 -05:00
|
|
|
|
2013-06-30 21:41:51 -05:00
|
|
|
#[inline]
|
2013-10-26 03:10:39 -05:00
|
|
|
pub fn get_box_size(body_size: uint, body_align: uint) -> uint {
|
2013-07-21 19:20:52 -05:00
|
|
|
let header_size = size_of::<raw::Box<()>>();
|
2013-05-21 20:24:42 -05:00
|
|
|
let total_size = align_to(header_size, body_align) + body_size;
|
2013-06-29 21:35:04 -05:00
|
|
|
total_size
|
2013-05-21 20:24:42 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
// Rounds |size| to the nearest |alignment|. Invariant: |alignment| is a power
|
|
|
|
// of two.
|
2013-06-30 21:41:51 -05:00
|
|
|
#[inline]
|
2013-05-21 20:24:42 -05:00
|
|
|
fn align_to(size: uint, align: uint) -> uint {
|
|
|
|
assert!(align != 0);
|
|
|
|
(size + align - 1) & !(align - 1)
|
|
|
|
}
|
|
|
|
|
2013-06-29 21:35:04 -05:00
|
|
|
/// A wrapper around libc::malloc, aborting on out-of-memory
|
2014-01-15 03:45:12 -06:00
|
|
|
#[inline]
|
2013-12-12 15:27:26 -06:00
|
|
|
pub unsafe fn malloc_raw(size: uint) -> *mut u8 {
|
2014-01-17 19:45:48 -06:00
|
|
|
// `malloc(0)` may allocate, but it may also return a null pointer
|
|
|
|
// http://pubs.opengroup.org/onlinepubs/9699919799/functions/malloc.html
|
|
|
|
if size == 0 {
|
|
|
|
mut_null()
|
|
|
|
} else {
|
|
|
|
let p = malloc(size as size_t);
|
|
|
|
if p.is_null() {
|
|
|
|
// we need a non-allocating way to print an error here
|
|
|
|
abort();
|
|
|
|
}
|
2013-12-12 15:27:26 -06:00
|
|
|
p as *mut u8
|
2013-06-29 21:35:04 -05:00
|
|
|
}
|
2013-05-21 20:24:42 -05:00
|
|
|
}
|
|
|
|
|
2013-06-30 21:12:26 -05:00
|
|
|
/// A wrapper around libc::realloc, aborting on out-of-memory
|
2014-01-15 03:45:12 -06:00
|
|
|
#[inline]
|
2013-12-12 15:27:26 -06:00
|
|
|
pub unsafe fn realloc_raw(ptr: *mut u8, size: uint) -> *mut u8 {
|
2014-01-17 19:45:48 -06:00
|
|
|
// `realloc(ptr, 0)` may allocate, but it may also return a null pointer
|
|
|
|
// http://pubs.opengroup.org/onlinepubs/9699919799/functions/realloc.html
|
|
|
|
if size == 0 {
|
2014-01-21 08:35:11 -06:00
|
|
|
free(ptr as *mut c_void);
|
2014-01-17 19:45:48 -06:00
|
|
|
mut_null()
|
|
|
|
} else {
|
2013-12-12 15:27:26 -06:00
|
|
|
let p = realloc(ptr as *mut c_void, size as size_t);
|
2014-01-17 19:45:48 -06:00
|
|
|
if p.is_null() {
|
|
|
|
// we need a non-allocating way to print an error here
|
|
|
|
abort();
|
|
|
|
}
|
2013-12-12 15:27:26 -06:00
|
|
|
p as *mut u8
|
2013-06-30 21:12:26 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-04-02 16:23:52 -05:00
|
|
|
// The compiler never calls `exchange_free` on ~ZeroSizeType, so zero-size
|
|
|
|
// allocations can point to this `static`. It would be incorrect to use a null
|
|
|
|
// pointer, due to enums assuming types like unique pointers are never null.
|
|
|
|
static EMPTY: () = ();
|
|
|
|
|
2013-07-08 12:27:55 -05:00
|
|
|
/// The allocator for unique pointers without contained managed pointers.
|
2013-07-21 23:06:29 -05:00
|
|
|
#[cfg(not(test))]
|
2013-06-29 21:35:04 -05:00
|
|
|
#[lang="exchange_malloc"]
|
|
|
|
#[inline]
|
2014-04-02 16:23:52 -05:00
|
|
|
pub unsafe fn exchange_malloc(size: uint) -> *mut u8 {
|
|
|
|
if size == 0 {
|
|
|
|
&EMPTY as *() as *mut u8
|
|
|
|
} else {
|
|
|
|
malloc_raw(size)
|
|
|
|
}
|
2013-06-30 02:22:18 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
// FIXME: #7496
|
2014-02-12 18:40:17 -06:00
|
|
|
#[cfg(not(test))]
|
2014-02-05 22:05:30 -06:00
|
|
|
#[lang="closure_exchange_malloc"]
|
2013-06-30 02:22:18 -05:00
|
|
|
#[inline]
|
2014-02-05 22:05:30 -06:00
|
|
|
pub unsafe fn closure_exchange_malloc_(drop_glue: fn(*mut u8), size: uint, align: uint) -> *u8 {
|
|
|
|
closure_exchange_malloc(drop_glue, size, align)
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
|
|
|
pub unsafe fn closure_exchange_malloc(drop_glue: fn(*mut u8), size: uint, align: uint) -> *u8 {
|
|
|
|
let total_size = get_box_size(size, align);
|
|
|
|
let p = malloc_raw(total_size);
|
|
|
|
|
|
|
|
let alloc = p as *mut raw::Box<()>;
|
|
|
|
(*alloc).drop_glue = drop_glue;
|
|
|
|
|
|
|
|
alloc as *u8
|
|
|
|
}
|
|
|
|
|
2013-06-29 21:35:04 -05:00
|
|
|
// NB: Calls to free CANNOT be allowed to fail, as throwing an exception from
|
|
|
|
// inside a landing pad may corrupt the state of the exception handler.
|
|
|
|
#[cfg(not(test))]
|
|
|
|
#[lang="exchange_free"]
|
2013-07-02 19:36:58 -05:00
|
|
|
#[inline]
|
2013-12-12 15:27:26 -06:00
|
|
|
pub unsafe fn exchange_free_(ptr: *u8) {
|
2013-07-02 19:36:58 -05:00
|
|
|
exchange_free(ptr)
|
2013-05-21 20:24:42 -05:00
|
|
|
}
|
|
|
|
|
2014-01-15 03:45:12 -06:00
|
|
|
#[inline]
|
2013-12-12 15:27:26 -06:00
|
|
|
pub unsafe fn exchange_free(ptr: *u8) {
|
2014-01-21 08:31:32 -06:00
|
|
|
free(ptr as *mut c_void);
|
2013-05-21 20:24:42 -05:00
|
|
|
}
|
2013-07-22 13:42:47 -05:00
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod bench {
|
2014-02-13 19:49:11 -06:00
|
|
|
extern crate test;
|
|
|
|
use self::test::BenchHarness;
|
2013-07-22 13:42:47 -05:00
|
|
|
|
|
|
|
#[bench]
|
|
|
|
fn alloc_owned_small(bh: &mut BenchHarness) {
|
2013-11-21 19:23:21 -06:00
|
|
|
bh.iter(|| {
|
2014-02-12 09:39:21 -06:00
|
|
|
~10
|
2013-11-21 19:23:21 -06:00
|
|
|
})
|
2013-07-22 13:42:47 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
#[bench]
|
|
|
|
fn alloc_owned_big(bh: &mut BenchHarness) {
|
2013-11-21 19:23:21 -06:00
|
|
|
bh.iter(|| {
|
2014-02-12 09:39:21 -06:00
|
|
|
~[10, ..1000]
|
2013-11-21 19:23:21 -06:00
|
|
|
})
|
2013-07-22 13:42:47 -05:00
|
|
|
}
|
|
|
|
}
|