rust/src/libcore/dvec.rs

368 lines
9.9 KiB
Rust
Raw Normal View History

/*!
Dynamic vector
A growable vector that makes use of unique pointers so that the
result can be sent between tasks and so forth.
Note that recursive use is not permitted.
*/
// NB: transitionary, de-mode-ing.
#[forbid(deprecated_mode)];
#[forbid(deprecated_pattern)];
2012-09-18 17:34:08 -07:00
use cast::reinterpret_cast;
2012-09-04 11:12:17 -07:00
use ptr::null;
2012-05-09 17:30:31 -07:00
2012-08-14 16:54:13 -07:00
export DVec;
export from_elem;
2012-05-09 17:30:31 -07:00
export from_vec;
export extensions;
export unwrap;
/**
* A growable, modifiable vector type that accumulates elements into a
* unique vector.
*
* # Limitations on recursive use
*
* This class works by swapping the unique vector out of the data
* structure whenever it is to be used. Therefore, recursive use is not
* permitted. That is, while iterating through a vector, you cannot
* access the vector in any other way or else the program will fail. If
* you wish, you can use the `swap()` method to gain access to the raw
* vector and transform it or use it any way you like. Eventually, we
* may permit read-only access during iteration or other use.
*
* # WARNING
*
* For maximum performance, this type is implemented using some rather
* unsafe code. In particular, this innocent looking `~[mut A]` pointer
* *may be null!* Therefore, it is important you not reach into the
* data structure manually but instead use the provided extensions.
*
* The reason that I did not use an unsafe pointer in the structure
* itself is that I wanted to ensure that the vector would be freed when
2012-08-20 12:23:37 -07:00
* the dvec is dropped. The reason that I did not use an `Option<T>`
* instead of a nullable pointer is that I found experimentally that it
* becomes approximately 50% slower. This can probably be improved
* through optimization. You can run your own experiments using
* `src/test/bench/vec-append.rs`. My own tests found that using null
* pointers achieved about 103 million pushes/second. Using an option
* type could only produce 47 million pushes/second.
*/
2012-08-14 16:54:13 -07:00
type DVec_<A> = {
mut data: ~[A]
2012-05-09 17:30:31 -07:00
};
2012-08-14 16:54:13 -07:00
enum DVec<A> {
DVec_(DVec_<A>)
}
/// Creates a new, empty dvec
2012-08-27 14:22:25 -07:00
fn DVec<A>() -> DVec<A> {
DVec_({mut data: ~[]})
2012-05-09 17:30:31 -07:00
}
/// Creates a new dvec with a single element
2012-08-14 16:54:13 -07:00
fn from_elem<A>(+e: A) -> DVec<A> {
DVec_({mut data: ~[move e]})
}
/// Creates a new dvec with the contents of a vector
fn from_vec<A>(+v: ~[A]) -> DVec<A> {
2012-09-10 12:14:14 -07:00
DVec_({mut data: move v})
2012-05-09 17:30:31 -07:00
}
/// Consumes the vector and returns its contents
fn unwrap<A>(+d: DVec<A>) -> ~[A] {
2012-08-14 16:54:13 -07:00
let DVec_({data: v}) <- d;
2012-09-10 12:14:14 -07:00
move v
2012-05-09 17:30:31 -07:00
}
2012-08-14 16:54:13 -07:00
priv impl<A> DVec<A> {
pure fn check_not_borrowed() {
2012-05-09 17:30:31 -07:00
unsafe {
2012-09-18 17:34:08 -07:00
let data: *() = cast::reinterpret_cast(&self.data);
2012-05-09 17:30:31 -07:00
if data.is_null() {
fail ~"Recursive use of dvec";
2012-05-09 17:30:31 -07:00
}
}
}
#[inline(always)]
fn check_out<B>(f: fn(-~[A]) -> B) -> B {
2012-05-09 17:30:31 -07:00
unsafe {
2012-09-18 17:34:08 -07:00
let mut data = cast::reinterpret_cast(&null::<()>());
2012-05-09 17:30:31 -07:00
data <-> self.data;
2012-09-18 17:34:08 -07:00
let data_ptr: *() = cast::reinterpret_cast(&data);
if data_ptr.is_null() { fail ~"Recursive use of dvec"; }
2012-09-11 17:17:54 -07:00
return f(move data);
2012-05-09 17:30:31 -07:00
}
}
#[inline(always)]
fn give_back(+data: ~[A]) {
2012-05-09 17:30:31 -07:00
unsafe {
self.data = move data;
2012-05-09 17:30:31 -07:00
}
}
}
// In theory, most everything should work with any A, but in practice
// almost nothing works without the copy bound due to limitations
// around closures.
2012-08-14 16:54:13 -07:00
impl<A> DVec<A> {
/// Reserves space for N elements
fn reserve(count: uint) {
vec::reserve(&mut self.data, count)
}
/**
* Swaps out the current vector and hands it off to a user-provided
* function `f`. The function should transform it however is desired
* and return a new vector to replace it with.
*/
2012-06-14 11:38:45 -07:00
#[inline(always)]
fn swap(f: fn(-~[A]) -> ~[A]) {
2012-09-11 17:17:54 -07:00
self.check_out(|v| self.give_back(f(move v)))
2012-05-09 17:30:31 -07:00
}
/**
* Swaps out the current vector and hands it off to a user-provided
* function `f`. The function should transform it however is desired
* and return a new vector to replace it with.
*/
#[inline(always)]
fn swap_mut(f: fn(-~[mut A]) -> ~[mut A]) {
do self.swap |v| {
vec::from_mut(f(vec::to_mut(move v)))
}
}
/// Returns the number of elements currently in the dvec
pure fn len() -> uint {
unsafe {
do self.check_out |v| {
let l = v.len();
2012-09-11 17:17:54 -07:00
self.give_back(move v);
l
}
2012-05-09 17:30:31 -07:00
}
}
/// Overwrite the current contents
fn set(+w: ~[A]) {
2012-05-09 17:30:31 -07:00
self.check_not_borrowed();
2012-06-15 06:23:58 -07:00
self.data <- w;
2012-05-09 17:30:31 -07:00
}
/// Remove and return the last element
2012-05-09 17:30:31 -07:00
fn pop() -> A {
do self.check_out |v| {
let mut v <- v;
2012-05-09 17:30:31 -07:00
let result = vec::pop(v);
2012-09-11 17:17:54 -07:00
self.give_back(move v);
2012-09-10 12:14:14 -07:00
move result
2012-05-09 17:30:31 -07:00
}
}
/// Insert a single item at the front of the list
fn unshift(-t: A) {
unsafe {
2012-09-18 17:34:08 -07:00
let mut data = cast::reinterpret_cast(&null::<()>());
data <-> self.data;
2012-09-18 17:34:08 -07:00
let data_ptr: *() = cast::reinterpret_cast(&data);
if data_ptr.is_null() { fail ~"Recursive use of dvec"; }
log(error, ~"a");
self.data <- ~[move t];
2012-09-11 17:17:54 -07:00
vec::push_all_move(self.data, move data);
log(error, ~"b");
}
}
/// Append a single item to the end of the list
fn push(+t: A) {
self.check_not_borrowed();
2012-09-10 12:14:14 -07:00
vec::push(self.data, move t);
}
/// Remove and return the first element
fn shift() -> A {
do self.check_out |v| {
let mut v = move v;
let result = vec::shift(v);
self.give_back(move v);
2012-09-10 12:14:14 -07:00
move result
}
}
/// Reverse the elements in the list, in place
fn reverse() {
do self.check_out |v| {
let mut v = move v;
vec::reverse(v);
2012-09-11 17:17:54 -07:00
self.give_back(move v);
}
}
/// Gives access to the vector as a slice with immutable contents
fn borrow<R>(op: fn(x: &[A]) -> R) -> R {
do self.check_out |v| {
let result = op(v);
2012-09-11 17:17:54 -07:00
self.give_back(move v);
2012-09-10 12:14:14 -07:00
move result
}
}
/// Gives access to the vector as a slice with mutable contents
fn borrow_mut<R>(op: fn(x: &[mut A]) -> R) -> R {
do self.check_out |v| {
let mut v = move v;
let result = op(v);
2012-09-11 17:17:54 -07:00
self.give_back(move v);
2012-09-10 12:14:14 -07:00
move result
}
}
}
impl<A: Copy> DVec<A> {
/**
* Append all elements of a vector to the end of the list
*
* Equivalent to `append_iter()` but potentially more efficient.
*/
fn push_all(ts: &[const A]) {
2012-05-09 17:30:31 -07:00
self.push_slice(ts, 0u, vec::len(ts));
}
/// Appends elements from `from_idx` to `to_idx` (exclusive)
fn push_slice(ts: &[const A], from_idx: uint, to_idx: uint) {
2012-06-30 16:19:07 -07:00
do self.swap |v| {
let mut v <- v;
2012-05-09 17:30:31 -07:00
let new_len = vec::len(v) + to_idx - from_idx;
vec::reserve(&mut v, new_len);
2012-05-09 17:30:31 -07:00
let mut i = from_idx;
while i < to_idx {
vec::push(v, ts[i]);
2012-05-09 17:30:31 -07:00
i += 1u;
}
2012-09-10 12:14:14 -07:00
move v
2012-05-09 17:30:31 -07:00
}
}
/**
* Append all elements of an iterable.
*
* Failure will occur if the iterable's `each()` method
* attempts to access this vector.
*/
/*
fn append_iter<A, I:iter::base_iter<A>>(ts: I) {
2012-06-30 16:19:07 -07:00
do self.swap |v| {
2012-08-06 12:34:08 -07:00
let mut v = match ts.size_hint() {
none { v }
2012-08-20 12:23:37 -07:00
Some(h) {
let len = v.len() + h;
let mut v <- v;
vec::reserve(v, len);
v
}
};
for ts.each |t| { vec::push(v, *t) };
v
}
}
*/
2012-05-09 17:30:31 -07:00
/**
* Gets a copy of the current contents.
*
* See `unwrap()` if you do not wish to copy the contents.
*/
pure fn get() -> ~[A] {
unsafe {
do self.check_out |v| {
let w = copy v;
2012-09-11 17:17:54 -07:00
self.give_back(move v);
2012-09-10 12:14:14 -07:00
move w
}
2012-05-09 17:30:31 -07:00
}
}
/// Copy out an individual element
2012-05-15 13:41:15 -07:00
#[inline(always)]
pure fn get_elt(idx: uint) -> A {
2012-05-09 17:30:31 -07:00
self.check_not_borrowed();
2012-08-01 17:30:05 -07:00
return self.data[idx];
2012-05-09 17:30:31 -07:00
}
/// Overwrites the contents of the element at `idx` with `a`
2012-05-09 17:30:31 -07:00
fn set_elt(idx: uint, a: A) {
self.check_not_borrowed();
self.data[idx] = a;
}
/**
* Overwrites the contents of the element at `idx` with `a`,
* growing the vector if necessary. New elements will be initialized
* with `initval`
*/
2012-05-09 17:30:31 -07:00
fn grow_set_elt(idx: uint, initval: A, val: A) {
2012-06-30 16:19:07 -07:00
do self.swap |v| {
let mut v = move v;
vec::grow_set(v, idx, initval, val);
move v
}
2012-05-09 17:30:31 -07:00
}
/// Returns the last element, failing if the vector is empty
#[inline(always)]
pure fn last() -> A {
self.check_not_borrowed();
let length = self.len();
if length == 0u {
fail ~"attempt to retrieve the last element of an empty vector";
}
2012-08-01 17:30:05 -07:00
return self.data[length - 1u];
}
/// Iterates over the elements in reverse order
#[inline(always)]
fn rev_each(f: fn(v: &A) -> bool) {
do self.swap |v| {
// FIXME(#2263)---we should be able to write
// `vec::rev_each(v, f);` but we cannot write now
for vec::rev_each(v) |e| {
if !f(e) { break; }
}
move v
}
}
/// Iterates over the elements and indices in reverse order
#[inline(always)]
fn rev_eachi(f: fn(uint, v: &A) -> bool) {
do self.swap |v| {
// FIXME(#2263)---we should be able to write
// `vec::rev_eachi(v, f);` but we cannot write now
for vec::rev_eachi(v) |i, e| {
if !f(i, e) { break; }
}
move v
}
}
}
impl<A:Copy> DVec<A>: Index<uint,A> {
pure fn index(&&idx: uint) -> A {
self.get_elt(idx)
}
}