2011-10-24 15:25:41 -07:00
|
|
|
/*
|
|
|
|
|
Module: deque
|
2010-07-20 18:03:09 -07:00
|
|
|
|
2011-10-24 15:25:41 -07:00
|
|
|
A deque. Untested as of yet. Likely buggy.
|
|
|
|
|
*/
|
2010-07-20 18:03:09 -07:00
|
|
|
|
2011-10-24 15:25:41 -07:00
|
|
|
/*
|
|
|
|
|
Object: t
|
|
|
|
|
*/
|
|
|
|
|
type t<T> = obj {
|
|
|
|
|
// Method: size
|
|
|
|
|
fn size() -> uint;
|
|
|
|
|
// Method: add_front
|
|
|
|
|
fn add_front(T);
|
|
|
|
|
// Method: add_back
|
|
|
|
|
fn add_back(T);
|
|
|
|
|
// Method: pop_front
|
|
|
|
|
fn pop_front() -> T;
|
|
|
|
|
// Method: pop_back
|
|
|
|
|
fn pop_back() -> T;
|
|
|
|
|
// Method: peek_front
|
|
|
|
|
fn peek_front() -> T;
|
|
|
|
|
// Method: peek_back
|
|
|
|
|
fn peek_back() -> T;
|
|
|
|
|
// Method: get
|
|
|
|
|
fn get(int) -> T;
|
2011-06-15 11:19:50 -07:00
|
|
|
};
|
2010-07-20 18:03:09 -07:00
|
|
|
|
2011-10-24 15:25:41 -07:00
|
|
|
/*
|
|
|
|
|
Section: Functions
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
Function: create
|
|
|
|
|
*/
|
2011-10-25 15:56:55 +02:00
|
|
|
fn create<T>() -> t<T> {
|
2011-08-12 06:37:10 -07:00
|
|
|
type cell<T> = option::t<T>;
|
2010-07-20 18:03:09 -07:00
|
|
|
|
2011-07-27 14:19:39 +02:00
|
|
|
let initial_capacity: uint = 32u; // 2^5
|
2011-06-15 11:19:50 -07:00
|
|
|
/**
|
|
|
|
|
* Grow is only called on full elts, so nelts is also len(elts), unlike
|
|
|
|
|
* elsewhere.
|
|
|
|
|
*/
|
2011-10-25 15:56:55 +02:00
|
|
|
fn grow<T>(nelts: uint, lo: uint, elts: [mutable cell<T>]) ->
|
2011-08-10 09:27:11 -07:00
|
|
|
[mutable cell<T>] {
|
2011-08-15 16:38:23 -07:00
|
|
|
assert (nelts == vec::len(elts));
|
2011-08-19 15:16:48 -07:00
|
|
|
let rv = [mutable];
|
2011-06-15 11:19:50 -07:00
|
|
|
|
2011-07-27 14:19:39 +02:00
|
|
|
let i = 0u;
|
|
|
|
|
let nalloc = uint::next_power_of_two(nelts + 1u);
|
|
|
|
|
while i < nalloc {
|
|
|
|
|
if i < nelts {
|
2011-08-19 15:16:48 -07:00
|
|
|
rv += [mutable elts[(lo + i) % nelts]];
|
|
|
|
|
} else { rv += [mutable option::none]; }
|
2011-07-12 14:20:15 -07:00
|
|
|
i += 1u;
|
2010-09-22 15:44:13 -07:00
|
|
|
}
|
2011-07-12 14:20:15 -07:00
|
|
|
|
|
|
|
|
ret rv;
|
2010-07-20 18:03:09 -07:00
|
|
|
}
|
2011-10-25 15:56:55 +02:00
|
|
|
fn get<T>(elts: [mutable cell<T>], i: uint) -> T {
|
2011-08-19 15:16:48 -07:00
|
|
|
ret alt elts[i] { option::some(t) { t } _ { fail } };
|
2010-07-20 18:03:09 -07:00
|
|
|
}
|
2011-10-28 17:00:14 +02:00
|
|
|
obj deque<shar T>(mutable nelts: uint,
|
|
|
|
|
mutable lo: uint,
|
|
|
|
|
mutable hi: uint,
|
|
|
|
|
mutable elts: [mutable cell<T>]) {
|
2011-06-15 11:19:50 -07:00
|
|
|
fn size() -> uint { ret nelts; }
|
2011-09-12 11:27:30 +02:00
|
|
|
fn add_front(t: T) {
|
2011-07-27 14:19:39 +02:00
|
|
|
let oldlo: uint = lo;
|
|
|
|
|
if lo == 0u {
|
2011-08-12 10:56:57 -07:00
|
|
|
lo = vec::len::<cell<T>>(elts) - 1u;
|
2011-06-15 11:19:50 -07:00
|
|
|
} else { lo -= 1u; }
|
2011-07-27 14:19:39 +02:00
|
|
|
if lo == hi {
|
2011-08-12 10:56:57 -07:00
|
|
|
elts = grow::<T>(nelts, oldlo, elts);
|
|
|
|
|
lo = vec::len::<cell<T>>(elts) - 1u;
|
2011-06-15 11:19:50 -07:00
|
|
|
hi = nelts;
|
2010-09-22 15:44:13 -07:00
|
|
|
}
|
2011-08-19 15:16:48 -07:00
|
|
|
elts[lo] = option::some::<T>(t);
|
2011-06-15 11:19:50 -07:00
|
|
|
nelts += 1u;
|
|
|
|
|
}
|
2011-09-12 11:27:30 +02:00
|
|
|
fn add_back(t: T) {
|
2011-07-27 14:19:39 +02:00
|
|
|
if lo == hi && nelts != 0u {
|
2011-08-12 10:56:57 -07:00
|
|
|
elts = grow::<T>(nelts, lo, elts);
|
2011-06-15 11:19:50 -07:00
|
|
|
lo = 0u;
|
|
|
|
|
hi = nelts;
|
2010-09-22 15:44:13 -07:00
|
|
|
}
|
2011-08-19 15:16:48 -07:00
|
|
|
elts[hi] = option::some::<T>(t);
|
2011-08-12 10:56:57 -07:00
|
|
|
hi = (hi + 1u) % vec::len::<cell<T>>(elts);
|
2011-06-15 11:19:50 -07:00
|
|
|
nelts += 1u;
|
2010-09-22 15:44:13 -07:00
|
|
|
}
|
|
|
|
|
|
2011-06-15 11:19:50 -07:00
|
|
|
/**
|
|
|
|
|
* We actually release (turn to none()) the T we're popping so
|
|
|
|
|
* that we don't keep anyone's refcount up unexpectedly.
|
|
|
|
|
*/
|
|
|
|
|
fn pop_front() -> T {
|
2011-08-12 10:56:57 -07:00
|
|
|
let t: T = get::<T>(elts, lo);
|
2011-08-19 15:16:48 -07:00
|
|
|
elts[lo] = option::none::<T>;
|
2011-08-12 10:56:57 -07:00
|
|
|
lo = (lo + 1u) % vec::len::<cell<T>>(elts);
|
2011-06-15 11:19:50 -07:00
|
|
|
nelts -= 1u;
|
|
|
|
|
ret t;
|
|
|
|
|
}
|
|
|
|
|
fn pop_back() -> T {
|
2011-07-27 14:19:39 +02:00
|
|
|
if hi == 0u {
|
2011-08-12 10:56:57 -07:00
|
|
|
hi = vec::len::<cell<T>>(elts) - 1u;
|
2011-07-27 14:19:39 +02:00
|
|
|
} else { hi -= 1u; }
|
2011-08-12 10:56:57 -07:00
|
|
|
let t: T = get::<T>(elts, hi);
|
2011-08-19 15:16:48 -07:00
|
|
|
elts[hi] = option::none::<T>;
|
2011-06-15 11:19:50 -07:00
|
|
|
nelts -= 1u;
|
|
|
|
|
ret t;
|
|
|
|
|
}
|
2011-08-12 10:56:57 -07:00
|
|
|
fn peek_front() -> T { ret get::<T>(elts, lo); }
|
|
|
|
|
fn peek_back() -> T { ret get::<T>(elts, hi - 1u); }
|
2011-07-27 14:19:39 +02:00
|
|
|
fn get(i: int) -> T {
|
2011-08-12 10:56:57 -07:00
|
|
|
let idx: uint = (lo + (i as uint)) % vec::len::<cell<T>>(elts);
|
|
|
|
|
ret get::<T>(elts, idx);
|
2011-06-15 11:19:50 -07:00
|
|
|
}
|
|
|
|
|
}
|
2011-08-10 09:27:11 -07:00
|
|
|
let v: [mutable cell<T>] =
|
2011-08-15 16:38:23 -07:00
|
|
|
vec::init_elt_mut(option::none, initial_capacity);
|
2011-08-12 10:56:57 -07:00
|
|
|
ret deque::<T>(0u, 0u, 0u, v);
|
2010-07-20 18:03:09 -07:00
|
|
|
}
|
2010-09-22 15:44:13 -07:00
|
|
|
// Local Variables:
|
|
|
|
|
// mode: rust;
|
|
|
|
|
// fill-column: 78;
|
|
|
|
|
// indent-tabs-mode: nil
|
|
|
|
|
// c-basic-offset: 4
|
|
|
|
|
// buffer-file-coding-system: utf-8-unix
|
|
|
|
|
// End:
|