2014-03-14 11:16:10 -07: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.
|
|
|
|
|
|
|
|
//! Library to interface with chunks of memory allocated in C.
|
|
|
|
//!
|
|
|
|
//! It is often desirable to safely interface with memory allocated from C,
|
|
|
|
//! encapsulating the unsafety into allocation and destruction time. Indeed,
|
|
|
|
//! allocating memory externally is currently the only way to give Rust shared
|
|
|
|
//! mut state with C programs that keep their own references; vectors are
|
|
|
|
//! unsuitable because they could be reallocated or moved at any time, and
|
|
|
|
//! importing C memory into a vector takes a one-time snapshot of the memory.
|
|
|
|
//!
|
|
|
|
//! This module simplifies the usage of such external blocks of memory. Memory
|
|
|
|
//! is encapsulated into an opaque object after creation; the lifecycle of the
|
|
|
|
//! memory can be optionally managed by Rust, if an appropriate destructor
|
|
|
|
//! closure is provided. Safety is ensured by bounds-checking accesses, which
|
|
|
|
//! are marshalled through get and set functions.
|
|
|
|
//!
|
|
|
|
//! There are three unsafe functions: the two constructors, and the
|
|
|
|
//! unwrap method. The constructors are unsafe for the
|
|
|
|
//! obvious reason (they act on a pointer that cannot be checked inside the
|
|
|
|
//! method), but `unwrap()` is somewhat more subtle in its unsafety.
|
|
|
|
//! It returns the contained pointer, but at the same time destroys the CVec
|
|
|
|
//! without running its destructor. This can be used to pass memory back to
|
|
|
|
//! C, but care must be taken that the ownership of underlying resources are
|
|
|
|
//! handled correctly, i.e. that allocated memory is eventually freed
|
|
|
|
//! if necessary.
|
|
|
|
|
2014-06-30 17:22:40 -07:00
|
|
|
#![experimental]
|
|
|
|
|
2014-03-08 18:21:49 -08:00
|
|
|
use kinds::Send;
|
core: Remove the cast module
This commit revisits the `cast` module in libcore and libstd, and scrutinizes
all functions inside of it. The result was to remove the `cast` module entirely,
folding all functionality into the `mem` module. Specifically, this is the fate
of each function in the `cast` module.
* transmute - This function was moved to `mem`, but it is now marked as
#[unstable]. This is due to planned changes to the `transmute`
function and how it can be invoked (see the #[unstable] comment).
For more information, see RFC 5 and #12898
* transmute_copy - This function was moved to `mem`, with clarification that is
is not an error to invoke it with T/U that are different
sizes, but rather that it is strongly discouraged. This
function is now #[stable]
* forget - This function was moved to `mem` and marked #[stable]
* bump_box_refcount - This function was removed due to the deprecation of
managed boxes as well as its questionable utility.
* transmute_mut - This function was previously deprecated, and removed as part
of this commit.
* transmute_mut_unsafe - This function doesn't serve much of a purpose when it
can be achieved with an `as` in safe code, so it was
removed.
* transmute_lifetime - This function was removed because it is likely a strong
indication that code is incorrect in the first place.
* transmute_mut_lifetime - This function was removed for the same reasons as
`transmute_lifetime`
* copy_lifetime - This function was moved to `mem`, but it is marked
`#[unstable]` now due to the likelihood of being removed in
the future if it is found to not be very useful.
* copy_mut_lifetime - This function was also moved to `mem`, but had the same
treatment as `copy_lifetime`.
* copy_lifetime_vec - This function was removed because it is not used today,
and its existence is not necessary with DST
(copy_lifetime will suffice).
In summary, the cast module was stripped down to these functions, and then the
functions were moved to the `mem` module.
transmute - #[unstable]
transmute_copy - #[stable]
forget - #[stable]
copy_lifetime - #[unstable]
copy_mut_lifetime - #[unstable]
[breaking-change]
2014-05-09 10:34:51 -07:00
|
|
|
use mem;
|
2014-11-26 08:12:18 -05:00
|
|
|
use ops::{Drop, FnOnce};
|
2014-11-28 11:57:41 -05:00
|
|
|
use option::Option;
|
|
|
|
use option::Option::{Some, None};
|
2014-03-14 11:16:10 -07:00
|
|
|
use ptr::RawPtr;
|
2014-03-08 18:21:49 -08:00
|
|
|
use ptr;
|
2014-03-14 11:16:10 -07:00
|
|
|
use raw;
|
2014-10-05 12:22:42 +13:00
|
|
|
use slice::AsSlice;
|
2014-11-26 08:12:18 -05:00
|
|
|
use thunk::{Thunk};
|
2014-03-14 11:16:10 -07:00
|
|
|
|
|
|
|
/// The type representing a foreign chunk of memory
|
|
|
|
pub struct CVec<T> {
|
2014-03-27 15:09:47 -07:00
|
|
|
base: *mut T,
|
|
|
|
len: uint,
|
2014-11-26 08:12:18 -05:00
|
|
|
dtor: Option<Thunk>,
|
2014-03-14 11:16:10 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
#[unsafe_destructor]
|
|
|
|
impl<T> Drop for CVec<T> {
|
|
|
|
fn drop(&mut self) {
|
|
|
|
match self.dtor.take() {
|
|
|
|
None => (),
|
2014-11-26 08:12:18 -05:00
|
|
|
Some(f) => f.invoke(())
|
2014-03-14 11:16:10 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<T> CVec<T> {
|
|
|
|
/// Create a `CVec` from a raw pointer to a buffer with a given length.
|
|
|
|
///
|
2014-11-08 21:17:51 +05:30
|
|
|
/// Panics if the given pointer is null. The returned vector will not attempt
|
2014-03-14 11:16:10 -07:00
|
|
|
/// to deallocate the vector when dropped.
|
|
|
|
///
|
|
|
|
/// # Arguments
|
|
|
|
///
|
|
|
|
/// * base - A raw pointer to a buffer
|
|
|
|
/// * len - The number of elements in the buffer
|
|
|
|
pub unsafe fn new(base: *mut T, len: uint) -> CVec<T> {
|
2014-09-14 20:27:36 -07:00
|
|
|
assert!(base != ptr::null_mut());
|
2014-03-14 11:16:10 -07:00
|
|
|
CVec {
|
|
|
|
base: base,
|
|
|
|
len: len,
|
|
|
|
dtor: None,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Create a `CVec` from a foreign buffer, with a given length,
|
|
|
|
/// and a function to run upon destruction.
|
|
|
|
///
|
2014-11-08 21:17:51 +05:30
|
|
|
/// Panics if the given pointer is null.
|
2014-03-14 11:16:10 -07:00
|
|
|
///
|
|
|
|
/// # Arguments
|
|
|
|
///
|
|
|
|
/// * base - A foreign pointer to a buffer
|
|
|
|
/// * len - The number of elements in the buffer
|
2014-11-26 08:12:18 -05:00
|
|
|
/// * dtor - A fn to run when the value is destructed, useful
|
2014-03-14 11:16:10 -07:00
|
|
|
/// for freeing the buffer, etc.
|
2014-11-26 08:12:18 -05:00
|
|
|
pub unsafe fn new_with_dtor<F>(base: *mut T,
|
|
|
|
len: uint,
|
|
|
|
dtor: F)
|
|
|
|
-> CVec<T>
|
|
|
|
where F : FnOnce(), F : Send
|
|
|
|
{
|
2014-09-14 20:27:36 -07:00
|
|
|
assert!(base != ptr::null_mut());
|
2014-11-26 08:12:18 -05:00
|
|
|
let dtor: Thunk = Thunk::new(dtor);
|
2014-03-14 11:16:10 -07:00
|
|
|
CVec {
|
|
|
|
base: base,
|
|
|
|
len: len,
|
2014-11-26 08:12:18 -05:00
|
|
|
dtor: Some(dtor)
|
2014-03-14 11:16:10 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// View the stored data as a mutable slice.
|
|
|
|
pub fn as_mut_slice<'a>(&'a mut self) -> &'a mut [T] {
|
|
|
|
unsafe {
|
2014-06-25 12:47:34 -07:00
|
|
|
mem::transmute(raw::Slice { data: self.base as *const T, len: self.len })
|
2014-03-14 11:16:10 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Retrieves an element at a given index, returning `None` if the requested
|
|
|
|
/// index is greater than the length of the vector.
|
|
|
|
pub fn get<'a>(&'a self, ofs: uint) -> Option<&'a T> {
|
|
|
|
if ofs < self.len {
|
|
|
|
Some(unsafe { &*self.base.offset(ofs as int) })
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Retrieves a mutable element at a given index, returning `None` if the
|
|
|
|
/// requested index is greater than the length of the vector.
|
|
|
|
pub fn get_mut<'a>(&'a mut self, ofs: uint) -> Option<&'a mut T> {
|
|
|
|
if ofs < self.len {
|
|
|
|
Some(unsafe { &mut *self.base.offset(ofs as int) })
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Unwrap the pointer without running the destructor
|
|
|
|
///
|
|
|
|
/// This method retrieves the underlying pointer, and in the process
|
|
|
|
/// destroys the CVec but without running the destructor. A use case
|
|
|
|
/// would be transferring ownership of the buffer to a C function, as
|
|
|
|
/// in this case you would not want to run the destructor.
|
|
|
|
///
|
|
|
|
/// Note that if you want to access the underlying pointer without
|
|
|
|
/// cancelling the destructor, you can simply call `transmute` on the return
|
|
|
|
/// value of `get(0)`.
|
2014-11-20 09:23:43 -08:00
|
|
|
pub unsafe fn into_inner(mut self) -> *mut T {
|
2014-03-14 11:16:10 -07:00
|
|
|
self.dtor = None;
|
|
|
|
self.base
|
|
|
|
}
|
2014-10-30 13:43:24 -07:00
|
|
|
|
2014-11-20 09:23:43 -08:00
|
|
|
/// Deprecated, use into_inner() instead
|
|
|
|
#[deprecated = "renamed to into_inner()"]
|
|
|
|
pub unsafe fn unwrap(self) -> *mut T { self.into_inner() }
|
|
|
|
|
2014-10-30 13:43:24 -07:00
|
|
|
/// Returns the number of items in this vector.
|
|
|
|
pub fn len(&self) -> uint { self.len }
|
|
|
|
|
|
|
|
/// Returns whether this vector is empty.
|
|
|
|
pub fn is_empty(&self) -> bool { self.len() == 0 }
|
2014-03-14 11:16:10 -07:00
|
|
|
}
|
|
|
|
|
2014-10-05 12:22:42 +13:00
|
|
|
impl<T> AsSlice<T> for CVec<T> {
|
2014-07-31 15:57:29 +12:00
|
|
|
/// View the stored data as a slice.
|
|
|
|
fn as_slice<'a>(&'a self) -> &'a [T] {
|
|
|
|
unsafe {
|
|
|
|
mem::transmute(raw::Slice { data: self.base as *const T, len: self.len })
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-03-14 11:16:10 -07:00
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
|
|
|
use prelude::*;
|
|
|
|
|
|
|
|
use super::CVec;
|
|
|
|
use libc;
|
|
|
|
use ptr;
|
|
|
|
|
|
|
|
fn malloc(n: uint) -> CVec<u8> {
|
|
|
|
unsafe {
|
2014-10-24 17:34:57 -04:00
|
|
|
let mem = libc::malloc(n as libc::size_t);
|
2014-10-28 17:06:06 -04:00
|
|
|
if mem.is_null() { ::alloc::oom() }
|
2014-03-14 11:16:10 -07:00
|
|
|
|
2014-11-26 08:12:18 -05:00
|
|
|
CVec::new_with_dtor(mem as *mut u8,
|
|
|
|
n,
|
|
|
|
move|| { libc::free(mem as *mut libc::c_void); })
|
2014-03-14 11:16:10 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_basic() {
|
|
|
|
let mut cv = malloc(16);
|
|
|
|
|
|
|
|
*cv.get_mut(3).unwrap() = 8;
|
|
|
|
*cv.get_mut(4).unwrap() = 9;
|
|
|
|
assert_eq!(*cv.get(3).unwrap(), 8);
|
|
|
|
assert_eq!(*cv.get(4).unwrap(), 9);
|
|
|
|
assert_eq!(cv.len(), 16);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
#[should_fail]
|
2014-10-09 15:17:22 -04:00
|
|
|
fn test_panic_at_null() {
|
2014-03-14 11:16:10 -07:00
|
|
|
unsafe {
|
2014-09-14 20:27:36 -07:00
|
|
|
CVec::new(ptr::null_mut::<u8>(), 9);
|
2014-03-14 11:16:10 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_overrun_get() {
|
|
|
|
let cv = malloc(16);
|
|
|
|
|
|
|
|
assert!(cv.get(17).is_none());
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_overrun_set() {
|
|
|
|
let mut cv = malloc(16);
|
|
|
|
|
|
|
|
assert!(cv.get_mut(17).is_none());
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_unwrap() {
|
|
|
|
unsafe {
|
2014-11-26 08:12:18 -05:00
|
|
|
let cv = CVec::new_with_dtor(1 as *mut int,
|
|
|
|
0,
|
|
|
|
move|:| panic!("Don't run this destructor!"));
|
2014-03-14 11:16:10 -07:00
|
|
|
let p = cv.unwrap();
|
|
|
|
assert_eq!(p, 1 as *mut int);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|