Deprecate the rev_iter pattern in all places where a DoubleEndedIterator is provided (everywhere but treemap)
This commit deprecates rev_iter, mut_rev_iter, move_rev_iter everywhere (except treemap) and also
deprecates related functions like rsplit, rev_components, and rev_str_components. In every case,
these functions can be replaced with the non-reversed form followed by a call to .rev(). To make this
more concrete, a translation table for all functional changes necessary follows:
* container.rev_iter() -> container.iter().rev()
* container.mut_rev_iter() -> container.mut_iter().rev()
* container.move_rev_iter() -> container.move_iter().rev()
* sliceorstr.rsplit(sep) -> sliceorstr.split(sep).rev()
* path.rev_components() -> path.components().rev()
* path.rev_str_components() -> path.str_components().rev()
In terms of the type system, this change also deprecates any specialized reversed iterator types (except
in treemap), opting instead to use Rev directly if any type annotations are needed. However, since
methods directly returning reversed iterators are now discouraged, the need for such annotations should
be small. However, in those cases, the general pattern for conversion is to take whatever follows Rev in
the original reversed name and surround it with Rev<>:
* RevComponents<'a> -> Rev<Components<'a>>
* RevStrComponents<'a> -> Rev<StrComponents<'a>>
* RevItems<'a, T> -> Rev<Items<'a, T>>
* etc.
The reasoning behind this change is that it makes the standard API much simpler without reducing readability,
performance, or power. The presence of functions such as rev_iter adds more boilerplate code to libraries
(all of which simply call .iter().rev()), clutters up the documentation, and only helps code by saving two
characters. Additionally, the numerous type synonyms that were used to make the type signatures look nice
like RevItems add even more boilerplate and clutter up the docs even more. With this change, all that cruft
goes away.
[breaking-change]
2014-04-20 23:59:12 -05:00
|
|
|
// Copyright 2012-2014 The Rust Project Developers. See the COPYRIGHT
|
2012-12-03 16:48:01 -08: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-02-17 23:44:55 -08:00
|
|
|
//! VecDeque is a double-ended queue, which is implemented with the help of a
|
|
|
|
//! growing ring buffer.
|
2015-02-17 13:45:35 -05:00
|
|
|
//!
|
2015-02-17 19:00:20 -08:00
|
|
|
//! This queue has `O(1)` amortized inserts and removals from both ends of the
|
|
|
|
//! container. It also has `O(1)` indexing like a vector. The contained elements
|
|
|
|
//! are not required to be copyable, and the queue will be sendable if the
|
|
|
|
//! contained type is sendable.
|
2013-05-17 15:28:44 -07:00
|
|
|
|
2015-01-23 21:48:20 -08:00
|
|
|
#![stable(feature = "rust1", since = "1.0.0")]
|
2015-01-04 16:35:20 -08:00
|
|
|
|
2014-12-22 09:04:23 -08:00
|
|
|
use core::cmp::Ordering;
|
std: Recreate a `collections` module
As with the previous commit with `librand`, this commit shuffles around some
`collections` code. The new state of the world is similar to that of librand:
* The libcollections crate now only depends on libcore and liballoc.
* The standard library has a new module, `std::collections`. All functionality
of libcollections is reexported through this module.
I would like to stress that this change is purely cosmetic. There are very few
alterations to these primitives.
There are a number of notable points about the new organization:
* std::{str, slice, string, vec} all moved to libcollections. There is no reason
that these primitives shouldn't be necessarily usable in a freestanding
context that has allocation. These are all reexported in their usual places in
the standard library.
* The `hashmap`, and transitively the `lru_cache`, modules no longer reside in
`libcollections`, but rather in libstd. The reason for this is because the
`HashMap::new` contructor requires access to the OSRng for initially seeding
the hash map. Beyond this requirement, there is no reason that the hashmap
could not move to libcollections.
I do, however, have a plan to move the hash map to the collections module. The
`HashMap::new` function could be altered to require that the `H` hasher
parameter ascribe to the `Default` trait, allowing the entire `hashmap` module
to live in libcollections. The key idea would be that the default hasher would
be different in libstd. Something along the lines of:
// src/libstd/collections/mod.rs
pub type HashMap<K, V, H = RandomizedSipHasher> =
core_collections::HashMap<K, V, H>;
This is not possible today because you cannot invoke static methods through
type aliases. If we modified the compiler, however, to allow invocation of
static methods through type aliases, then this type definition would
essentially be switching the default hasher from `SipHasher` in libcollections
to a libstd-defined `RandomizedSipHasher` type. This type's `Default`
implementation would randomly seed the `SipHasher` instance, and otherwise
perform the same as `SipHasher`.
This future state doesn't seem incredibly far off, but until that time comes,
the hashmap module will live in libstd to not compromise on functionality.
* In preparation for the hashmap moving to libcollections, the `hash` module has
moved from libstd to libcollections. A previously snapshotted commit enables a
distinct `Writer` trait to live in the `hash` module which `Hash`
implementations are now parameterized over.
Due to using a custom trait, the `SipHasher` implementation has lost its
specialized methods for writing integers. These can be re-added
backwards-compatibly in the future via default methods if necessary, but the
FNV hashing should satisfy much of the need for speedier hashing.
A list of breaking changes:
* HashMap::{get, get_mut} no longer fails with the key formatted into the error
message with `{:?}`, instead, a generic message is printed. With backtraces,
it should still be not-too-hard to track down errors.
* The HashMap, HashSet, and LruCache types are now available through
std::collections instead of the collections crate.
* Manual implementations of hash should be parameterized over `hash::Writer`
instead of just `Writer`.
[breaking-change]
2014-05-29 18:50:12 -07:00
|
|
|
use core::fmt;
|
2015-08-23 21:59:07 -07:00
|
|
|
use core::iter::{repeat, FromIterator};
|
2015-09-18 16:32:52 +02:00
|
|
|
use core::mem;
|
2014-12-22 09:04:23 -08:00
|
|
|
use core::ops::{Index, IndexMut};
|
2015-07-09 21:57:21 -07:00
|
|
|
use core::ptr;
|
2015-03-13 09:56:18 +01:00
|
|
|
use core::slice;
|
2014-10-19 16:19:07 -04:00
|
|
|
|
2015-02-17 20:48:07 -08:00
|
|
|
use core::hash::{Hash, Hasher};
|
2014-09-07 14:57:26 -07:00
|
|
|
use core::cmp;
|
2013-05-06 00:42:54 -04:00
|
|
|
|
2015-07-09 21:57:21 -07:00
|
|
|
use alloc::raw_vec::RawVec;
|
2013-07-10 15:27:14 +02:00
|
|
|
|
2015-08-06 19:03:14 -07:00
|
|
|
use super::range::RangeArgument;
|
|
|
|
|
2015-02-27 15:36:53 +01:00
|
|
|
const INITIAL_CAPACITY: usize = 7; // 2^3 - 1
|
|
|
|
const MINIMUM_CAPACITY: usize = 1; // 2 - 1
|
2016-01-15 10:07:52 -08:00
|
|
|
#[cfg(target_pointer_width = "32")]
|
|
|
|
const MAXIMUM_ZST_CAPACITY: usize = 1 << (32 - 1); // Largest possible power of two
|
|
|
|
#[cfg(target_pointer_width = "64")]
|
|
|
|
const MAXIMUM_ZST_CAPACITY: usize = 1 << (64 - 1); // Largest possible power of two
|
2014-11-06 12:24:47 -05:00
|
|
|
|
std: Stabilize library APIs for 1.5
This commit stabilizes and deprecates library APIs whose FCP has closed in the
last cycle, specifically:
Stabilized APIs:
* `fs::canonicalize`
* `Path::{metadata, symlink_metadata, canonicalize, read_link, read_dir, exists,
is_file, is_dir}` - all moved to inherent methods from the `PathExt` trait.
* `Formatter::fill`
* `Formatter::width`
* `Formatter::precision`
* `Formatter::sign_plus`
* `Formatter::sign_minus`
* `Formatter::alternate`
* `Formatter::sign_aware_zero_pad`
* `string::ParseError`
* `Utf8Error::valid_up_to`
* `Iterator::{cmp, partial_cmp, eq, ne, lt, le, gt, ge}`
* `<[T]>::split_{first,last}{,_mut}`
* `Condvar::wait_timeout` - note that `wait_timeout_ms` is not yet deprecated
but will be once 1.5 is released.
* `str::{R,}MatchIndices`
* `str::{r,}match_indices`
* `char::from_u32_unchecked`
* `VecDeque::insert`
* `VecDeque::shrink_to_fit`
* `VecDeque::as_slices`
* `VecDeque::as_mut_slices`
* `VecDeque::swap_remove_front` - (renamed from `swap_front_remove`)
* `VecDeque::swap_remove_back` - (renamed from `swap_back_remove`)
* `Vec::resize`
* `str::slice_mut_unchecked`
* `FileTypeExt`
* `FileTypeExt::{is_block_device, is_char_device, is_fifo, is_socket}`
* `BinaryHeap::from` - `from_vec` deprecated in favor of this
* `BinaryHeap::into_vec` - plus a `Into` impl
* `BinaryHeap::into_sorted_vec`
Deprecated APIs
* `slice::ref_slice`
* `slice::mut_ref_slice`
* `iter::{range_inclusive, RangeInclusive}`
* `std::dynamic_lib`
Closes #27706
Closes #27725
cc #27726 (align not stabilized yet)
Closes #27734
Closes #27737
Closes #27742
Closes #27743
Closes #27772
Closes #27774
Closes #27777
Closes #27781
cc #27788 (a few remaining methods though)
Closes #27790
Closes #27793
Closes #27796
Closes #27810
cc #28147 (not all parts stabilized)
2015-10-22 16:28:45 -07:00
|
|
|
/// `VecDeque` is a growable ring buffer, which can be used as a double-ended
|
|
|
|
/// queue efficiently.
|
2015-07-05 12:18:57 -04:00
|
|
|
///
|
std: Stabilize library APIs for 1.5
This commit stabilizes and deprecates library APIs whose FCP has closed in the
last cycle, specifically:
Stabilized APIs:
* `fs::canonicalize`
* `Path::{metadata, symlink_metadata, canonicalize, read_link, read_dir, exists,
is_file, is_dir}` - all moved to inherent methods from the `PathExt` trait.
* `Formatter::fill`
* `Formatter::width`
* `Formatter::precision`
* `Formatter::sign_plus`
* `Formatter::sign_minus`
* `Formatter::alternate`
* `Formatter::sign_aware_zero_pad`
* `string::ParseError`
* `Utf8Error::valid_up_to`
* `Iterator::{cmp, partial_cmp, eq, ne, lt, le, gt, ge}`
* `<[T]>::split_{first,last}{,_mut}`
* `Condvar::wait_timeout` - note that `wait_timeout_ms` is not yet deprecated
but will be once 1.5 is released.
* `str::{R,}MatchIndices`
* `str::{r,}match_indices`
* `char::from_u32_unchecked`
* `VecDeque::insert`
* `VecDeque::shrink_to_fit`
* `VecDeque::as_slices`
* `VecDeque::as_mut_slices`
* `VecDeque::swap_remove_front` - (renamed from `swap_front_remove`)
* `VecDeque::swap_remove_back` - (renamed from `swap_back_remove`)
* `Vec::resize`
* `str::slice_mut_unchecked`
* `FileTypeExt`
* `FileTypeExt::{is_block_device, is_char_device, is_fifo, is_socket}`
* `BinaryHeap::from` - `from_vec` deprecated in favor of this
* `BinaryHeap::into_vec` - plus a `Into` impl
* `BinaryHeap::into_sorted_vec`
Deprecated APIs
* `slice::ref_slice`
* `slice::mut_ref_slice`
* `iter::{range_inclusive, RangeInclusive}`
* `std::dynamic_lib`
Closes #27706
Closes #27725
cc #27726 (align not stabilized yet)
Closes #27734
Closes #27737
Closes #27742
Closes #27743
Closes #27772
Closes #27774
Closes #27777
Closes #27781
cc #27788 (a few remaining methods though)
Closes #27790
Closes #27793
Closes #27796
Closes #27810
cc #28147 (not all parts stabilized)
2015-10-22 16:28:45 -07:00
|
|
|
/// The "default" usage of this type as a queue is to use `push_back` to add to
|
|
|
|
/// the queue, and `pop_front` to remove from the queue. `extend` and `append`
|
|
|
|
/// push onto the back in this manner, and iterating over `VecDeque` goes front
|
|
|
|
/// to back.
|
2015-01-23 21:48:20 -08:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2015-02-17 23:44:55 -08:00
|
|
|
pub struct VecDeque<T> {
|
2014-10-19 16:19:07 -04:00
|
|
|
// tail and head are pointers into the buffer. Tail always points
|
|
|
|
// to the first element that could be read, Head always points
|
|
|
|
// to where data should be written.
|
2015-07-05 03:49:36 -04:00
|
|
|
// If tail == head the buffer is empty. The length of the ringbuffer
|
2014-10-19 16:19:07 -04:00
|
|
|
// is defined as the distance between the two.
|
2015-02-04 21:17:19 -05:00
|
|
|
tail: usize,
|
|
|
|
head: usize,
|
2015-07-09 21:57:21 -07:00
|
|
|
buf: RawVec<T>,
|
2014-10-19 16:19:07 -04:00
|
|
|
}
|
|
|
|
|
2015-01-23 21:48:20 -08:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2015-02-17 23:44:55 -08:00
|
|
|
impl<T: Clone> Clone for VecDeque<T> {
|
|
|
|
fn clone(&self) -> VecDeque<T> {
|
2015-02-04 21:17:19 -05:00
|
|
|
self.iter().cloned().collect()
|
2014-10-19 16:19:07 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-23 21:48:20 -08:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2015-02-17 23:44:55 -08:00
|
|
|
impl<T> Drop for VecDeque<T> {
|
2015-07-17 16:12:35 +02:00
|
|
|
#[unsafe_destructor_blind_to_params]
|
2014-10-19 16:19:07 -04:00
|
|
|
fn drop(&mut self) {
|
2016-03-02 17:54:43 +01:00
|
|
|
let (front, back) = self.as_mut_slices();
|
|
|
|
unsafe {
|
|
|
|
// use drop for [T]
|
|
|
|
ptr::drop_in_place(front);
|
|
|
|
ptr::drop_in_place(back);
|
|
|
|
}
|
2015-07-09 21:57:21 -07:00
|
|
|
// RawVec handles deallocation
|
2014-10-19 16:19:07 -04:00
|
|
|
}
|
2012-01-11 12:49:33 +01:00
|
|
|
}
|
2010-07-20 18:03:09 -07:00
|
|
|
|
2015-01-23 21:48:20 -08:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2015-02-17 23:44:55 -08:00
|
|
|
impl<T> Default for VecDeque<T> {
|
2014-06-09 00:30:04 -07:00
|
|
|
#[inline]
|
2015-11-24 11:23:48 +13:00
|
|
|
fn default() -> VecDeque<T> {
|
|
|
|
VecDeque::new()
|
|
|
|
}
|
2014-06-09 00:30:04 -07:00
|
|
|
}
|
|
|
|
|
2015-02-17 23:44:55 -08:00
|
|
|
impl<T> VecDeque<T> {
|
2015-07-09 21:57:21 -07:00
|
|
|
/// Marginally more convenient
|
|
|
|
#[inline]
|
|
|
|
fn ptr(&self) -> *mut T {
|
|
|
|
self.buf.ptr()
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Marginally more convenient
|
|
|
|
#[inline]
|
|
|
|
fn cap(&self) -> usize {
|
2015-09-18 16:32:52 +02:00
|
|
|
if mem::size_of::<T>() == 0 {
|
|
|
|
// For zero sized types, we are always at maximum capacity
|
|
|
|
MAXIMUM_ZST_CAPACITY
|
|
|
|
} else {
|
|
|
|
self.buf.cap()
|
|
|
|
}
|
2015-07-09 21:57:21 -07:00
|
|
|
}
|
|
|
|
|
2014-10-19 16:19:07 -04:00
|
|
|
/// Turn ptr into a slice
|
|
|
|
#[inline]
|
2014-12-30 19:07:53 -05:00
|
|
|
unsafe fn buffer_as_slice(&self) -> &[T] {
|
2015-07-09 21:57:21 -07:00
|
|
|
slice::from_raw_parts(self.ptr(), self.cap())
|
2014-12-15 23:01:58 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Turn ptr into a mut slice
|
|
|
|
#[inline]
|
2014-12-30 19:07:53 -05:00
|
|
|
unsafe fn buffer_as_mut_slice(&mut self) -> &mut [T] {
|
2015-07-09 21:57:21 -07:00
|
|
|
slice::from_raw_parts_mut(self.ptr(), self.cap())
|
2014-10-19 16:19:07 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Moves an element out of the buffer
|
|
|
|
#[inline]
|
2015-02-04 21:17:19 -05:00
|
|
|
unsafe fn buffer_read(&mut self, off: usize) -> T {
|
2015-07-09 21:57:21 -07:00
|
|
|
ptr::read(self.ptr().offset(off as isize))
|
2014-10-19 16:19:07 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Writes an element into the buffer, moving it.
|
|
|
|
#[inline]
|
2015-07-17 21:28:25 +02:00
|
|
|
unsafe fn buffer_write(&mut self, off: usize, value: T) {
|
|
|
|
ptr::write(self.ptr().offset(off as isize), value);
|
2014-10-19 16:19:07 -04:00
|
|
|
}
|
|
|
|
|
2015-07-08 21:17:13 -05:00
|
|
|
/// Returns true if and only if the buffer is at capacity
|
2014-10-19 16:19:07 -04:00
|
|
|
#[inline]
|
2015-11-24 11:23:48 +13:00
|
|
|
fn is_full(&self) -> bool {
|
|
|
|
self.cap() - self.len() == 1
|
|
|
|
}
|
2014-11-11 20:22:07 -05:00
|
|
|
|
2015-02-17 19:00:20 -08:00
|
|
|
/// Returns the index in the underlying buffer for a given logical element
|
|
|
|
/// index.
|
2014-11-11 20:22:07 -05:00
|
|
|
#[inline]
|
2015-11-24 11:23:48 +13:00
|
|
|
fn wrap_index(&self, idx: usize) -> usize {
|
|
|
|
wrap_index(idx, self.cap())
|
|
|
|
}
|
2014-12-03 11:12:30 -06:00
|
|
|
|
2015-02-19 08:33:32 +01:00
|
|
|
/// Returns the index in the underlying buffer for a given logical element
|
|
|
|
/// index + addend.
|
|
|
|
#[inline]
|
|
|
|
fn wrap_add(&self, idx: usize, addend: usize) -> usize {
|
2015-07-09 21:57:21 -07:00
|
|
|
wrap_index(idx.wrapping_add(addend), self.cap())
|
2015-02-19 08:33:32 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the index in the underlying buffer for a given logical element
|
|
|
|
/// index - subtrahend.
|
|
|
|
#[inline]
|
|
|
|
fn wrap_sub(&self, idx: usize, subtrahend: usize) -> usize {
|
2015-07-09 21:57:21 -07:00
|
|
|
wrap_index(idx.wrapping_sub(subtrahend), self.cap())
|
2015-02-19 08:33:32 +01:00
|
|
|
}
|
|
|
|
|
2014-12-03 11:12:30 -06:00
|
|
|
/// Copies a contiguous block of memory len long from src to dst
|
|
|
|
#[inline]
|
2015-02-04 21:17:19 -05:00
|
|
|
unsafe fn copy(&self, dst: usize, src: usize, len: usize) {
|
2015-11-24 11:23:48 +13:00
|
|
|
debug_assert!(dst + len <= self.cap(),
|
|
|
|
"cpy dst={} src={} len={} cap={}",
|
|
|
|
dst,
|
|
|
|
src,
|
|
|
|
len,
|
2015-07-09 21:57:21 -07:00
|
|
|
self.cap());
|
2015-11-24 11:23:48 +13:00
|
|
|
debug_assert!(src + len <= self.cap(),
|
|
|
|
"cpy dst={} src={} len={} cap={}",
|
|
|
|
dst,
|
|
|
|
src,
|
|
|
|
len,
|
2015-07-09 21:57:21 -07:00
|
|
|
self.cap());
|
2015-11-24 11:23:48 +13:00
|
|
|
ptr::copy(self.ptr().offset(src as isize),
|
|
|
|
self.ptr().offset(dst as isize),
|
|
|
|
len);
|
2015-01-05 15:48:58 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Copies a contiguous block of memory len long from src to dst
|
|
|
|
#[inline]
|
2015-02-04 21:17:19 -05:00
|
|
|
unsafe fn copy_nonoverlapping(&self, dst: usize, src: usize, len: usize) {
|
2015-11-24 11:23:48 +13:00
|
|
|
debug_assert!(dst + len <= self.cap(),
|
|
|
|
"cno dst={} src={} len={} cap={}",
|
|
|
|
dst,
|
|
|
|
src,
|
|
|
|
len,
|
2015-07-09 21:57:21 -07:00
|
|
|
self.cap());
|
2015-11-24 11:23:48 +13:00
|
|
|
debug_assert!(src + len <= self.cap(),
|
|
|
|
"cno dst={} src={} len={} cap={}",
|
|
|
|
dst,
|
|
|
|
src,
|
|
|
|
len,
|
2015-07-09 21:57:21 -07:00
|
|
|
self.cap());
|
2015-11-24 11:23:48 +13:00
|
|
|
ptr::copy_nonoverlapping(self.ptr().offset(src as isize),
|
|
|
|
self.ptr().offset(dst as isize),
|
|
|
|
len);
|
2014-12-03 11:12:30 -06:00
|
|
|
}
|
2015-07-09 21:57:21 -07:00
|
|
|
|
2015-08-06 19:03:14 -07:00
|
|
|
/// Copies a potentially wrapping block of memory len long from src to dest.
|
|
|
|
/// (abs(dst - src) + len) must be no larger than cap() (There must be at
|
|
|
|
/// most one continuous overlapping region between src and dest).
|
|
|
|
unsafe fn wrap_copy(&self, dst: usize, src: usize, len: usize) {
|
2015-11-03 13:03:36 -05:00
|
|
|
#[allow(dead_code)]
|
2015-11-24 11:23:48 +13:00
|
|
|
fn diff(a: usize, b: usize) -> usize {
|
|
|
|
if a <= b {
|
|
|
|
b - a
|
|
|
|
} else {
|
|
|
|
a - b
|
|
|
|
}
|
|
|
|
}
|
|
|
|
debug_assert!(cmp::min(diff(dst, src), self.cap() - diff(dst, src)) + len <= self.cap(),
|
|
|
|
"wrc dst={} src={} len={} cap={}",
|
|
|
|
dst,
|
|
|
|
src,
|
|
|
|
len,
|
|
|
|
self.cap());
|
2015-08-06 19:03:14 -07:00
|
|
|
|
2015-11-24 11:23:48 +13:00
|
|
|
if src == dst || len == 0 {
|
|
|
|
return;
|
|
|
|
}
|
2015-08-06 19:03:14 -07:00
|
|
|
|
|
|
|
let dst_after_src = self.wrap_sub(dst, src) < len;
|
|
|
|
|
|
|
|
let src_pre_wrap_len = self.cap() - src;
|
|
|
|
let dst_pre_wrap_len = self.cap() - dst;
|
|
|
|
let src_wraps = src_pre_wrap_len < len;
|
|
|
|
let dst_wraps = dst_pre_wrap_len < len;
|
|
|
|
|
|
|
|
match (dst_after_src, src_wraps, dst_wraps) {
|
|
|
|
(_, false, false) => {
|
|
|
|
// src doesn't wrap, dst doesn't wrap
|
|
|
|
//
|
|
|
|
// S . . .
|
|
|
|
// 1 [_ _ A A B B C C _]
|
|
|
|
// 2 [_ _ A A A A B B _]
|
|
|
|
// D . . .
|
|
|
|
//
|
|
|
|
self.copy(dst, src, len);
|
|
|
|
}
|
|
|
|
(false, false, true) => {
|
|
|
|
// dst before src, src doesn't wrap, dst wraps
|
|
|
|
//
|
|
|
|
// S . . .
|
|
|
|
// 1 [A A B B _ _ _ C C]
|
|
|
|
// 2 [A A B B _ _ _ A A]
|
|
|
|
// 3 [B B B B _ _ _ A A]
|
|
|
|
// . . D .
|
|
|
|
//
|
|
|
|
self.copy(dst, src, dst_pre_wrap_len);
|
|
|
|
self.copy(0, src + dst_pre_wrap_len, len - dst_pre_wrap_len);
|
|
|
|
}
|
|
|
|
(true, false, true) => {
|
|
|
|
// src before dst, src doesn't wrap, dst wraps
|
|
|
|
//
|
|
|
|
// S . . .
|
|
|
|
// 1 [C C _ _ _ A A B B]
|
|
|
|
// 2 [B B _ _ _ A A B B]
|
|
|
|
// 3 [B B _ _ _ A A A A]
|
|
|
|
// . . D .
|
|
|
|
//
|
|
|
|
self.copy(0, src + dst_pre_wrap_len, len - dst_pre_wrap_len);
|
|
|
|
self.copy(dst, src, dst_pre_wrap_len);
|
|
|
|
}
|
|
|
|
(false, true, false) => {
|
|
|
|
// dst before src, src wraps, dst doesn't wrap
|
|
|
|
//
|
|
|
|
// . . S .
|
|
|
|
// 1 [C C _ _ _ A A B B]
|
|
|
|
// 2 [C C _ _ _ B B B B]
|
|
|
|
// 3 [C C _ _ _ B B C C]
|
|
|
|
// D . . .
|
|
|
|
//
|
|
|
|
self.copy(dst, src, src_pre_wrap_len);
|
|
|
|
self.copy(dst + src_pre_wrap_len, 0, len - src_pre_wrap_len);
|
|
|
|
}
|
|
|
|
(true, true, false) => {
|
|
|
|
// src before dst, src wraps, dst doesn't wrap
|
|
|
|
//
|
|
|
|
// . . S .
|
|
|
|
// 1 [A A B B _ _ _ C C]
|
|
|
|
// 2 [A A A A _ _ _ C C]
|
|
|
|
// 3 [C C A A _ _ _ C C]
|
|
|
|
// D . . .
|
|
|
|
//
|
|
|
|
self.copy(dst + src_pre_wrap_len, 0, len - src_pre_wrap_len);
|
|
|
|
self.copy(dst, src, src_pre_wrap_len);
|
|
|
|
}
|
|
|
|
(false, true, true) => {
|
|
|
|
// dst before src, src wraps, dst wraps
|
|
|
|
//
|
|
|
|
// . . . S .
|
|
|
|
// 1 [A B C D _ E F G H]
|
|
|
|
// 2 [A B C D _ E G H H]
|
|
|
|
// 3 [A B C D _ E G H A]
|
|
|
|
// 4 [B C C D _ E G H A]
|
|
|
|
// . . D . .
|
|
|
|
//
|
|
|
|
debug_assert!(dst_pre_wrap_len > src_pre_wrap_len);
|
|
|
|
let delta = dst_pre_wrap_len - src_pre_wrap_len;
|
|
|
|
self.copy(dst, src, src_pre_wrap_len);
|
|
|
|
self.copy(dst + src_pre_wrap_len, 0, delta);
|
|
|
|
self.copy(0, delta, len - dst_pre_wrap_len);
|
|
|
|
}
|
|
|
|
(true, true, true) => {
|
|
|
|
// src before dst, src wraps, dst wraps
|
|
|
|
//
|
|
|
|
// . . S . .
|
|
|
|
// 1 [A B C D _ E F G H]
|
|
|
|
// 2 [A A B D _ E F G H]
|
|
|
|
// 3 [H A B D _ E F G H]
|
|
|
|
// 4 [H A B D _ E F F G]
|
|
|
|
// . . . D .
|
|
|
|
//
|
|
|
|
debug_assert!(src_pre_wrap_len > dst_pre_wrap_len);
|
|
|
|
let delta = src_pre_wrap_len - dst_pre_wrap_len;
|
|
|
|
self.copy(delta, 0, len - src_pre_wrap_len);
|
|
|
|
self.copy(0, self.cap() - delta, delta);
|
|
|
|
self.copy(dst, src, dst_pre_wrap_len);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-07-09 21:57:21 -07:00
|
|
|
/// Frobs the head and tail sections around to handle the fact that we
|
|
|
|
/// just reallocated. Unsafe because it trusts old_cap.
|
|
|
|
#[inline]
|
|
|
|
unsafe fn handle_cap_increase(&mut self, old_cap: usize) {
|
|
|
|
let new_cap = self.cap();
|
|
|
|
|
|
|
|
// Move the shortest contiguous section of the ring buffer
|
|
|
|
// T H
|
|
|
|
// [o o o o o o o . ]
|
|
|
|
// T H
|
|
|
|
// A [o o o o o o o . . . . . . . . . ]
|
|
|
|
// H T
|
|
|
|
// [o o . o o o o o ]
|
|
|
|
// T H
|
|
|
|
// B [. . . o o o o o o o . . . . . . ]
|
|
|
|
// H T
|
|
|
|
// [o o o o o . o o ]
|
|
|
|
// H T
|
|
|
|
// C [o o o o o . . . . . . . . . o o ]
|
|
|
|
|
2015-11-24 11:23:48 +13:00
|
|
|
if self.tail <= self.head {
|
|
|
|
// A
|
2015-07-09 21:57:21 -07:00
|
|
|
// Nop
|
2015-11-24 11:23:48 +13:00
|
|
|
} else if self.head < old_cap - self.tail {
|
|
|
|
// B
|
2015-07-09 21:57:21 -07:00
|
|
|
self.copy_nonoverlapping(old_cap, 0, self.head);
|
|
|
|
self.head += old_cap;
|
|
|
|
debug_assert!(self.head > self.tail);
|
2015-11-24 11:23:48 +13:00
|
|
|
} else {
|
|
|
|
// C
|
2015-07-09 21:57:21 -07:00
|
|
|
let new_tail = new_cap - (old_cap - self.tail);
|
|
|
|
self.copy_nonoverlapping(new_tail, self.tail, old_cap - self.tail);
|
|
|
|
self.tail = new_tail;
|
|
|
|
debug_assert!(self.head < self.tail);
|
|
|
|
}
|
|
|
|
debug_assert!(self.head < self.cap());
|
|
|
|
debug_assert!(self.tail < self.cap());
|
|
|
|
debug_assert!(self.cap().count_ones() == 1);
|
|
|
|
}
|
2014-10-19 16:19:07 -04:00
|
|
|
}
|
|
|
|
|
2015-02-17 23:44:55 -08:00
|
|
|
impl<T> VecDeque<T> {
|
|
|
|
/// Creates an empty `VecDeque`.
|
2015-01-23 21:48:20 -08:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2015-02-17 23:44:55 -08:00
|
|
|
pub fn new() -> VecDeque<T> {
|
|
|
|
VecDeque::with_capacity(INITIAL_CAPACITY)
|
2013-07-10 15:27:14 +02:00
|
|
|
}
|
|
|
|
|
2015-02-17 23:44:55 -08:00
|
|
|
/// Creates an empty `VecDeque` with space for at least `n` elements.
|
2015-01-23 21:48:20 -08:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2015-02-17 23:44:55 -08:00
|
|
|
pub fn with_capacity(n: usize) -> VecDeque<T> {
|
2014-10-19 16:19:07 -04:00
|
|
|
// +1 since the ringbuffer always leaves one space empty
|
2015-01-05 15:48:58 +01:00
|
|
|
let cap = cmp::max(n + 1, MINIMUM_CAPACITY + 1).next_power_of_two();
|
|
|
|
assert!(cap > n, "capacity overflow");
|
2014-11-09 22:34:53 -05:00
|
|
|
|
2015-02-17 23:44:55 -08:00
|
|
|
VecDeque {
|
2014-10-19 16:19:07 -04:00
|
|
|
tail: 0,
|
|
|
|
head: 0,
|
2015-07-09 21:57:21 -07:00
|
|
|
buf: RawVec::with_capacity(cap),
|
2014-10-19 16:19:07 -04:00
|
|
|
}
|
2013-07-10 15:27:14 +02:00
|
|
|
}
|
|
|
|
|
2015-02-17 23:44:55 -08:00
|
|
|
/// Retrieves an element in the `VecDeque` by index.
|
2013-07-10 15:27:14 +02:00
|
|
|
///
|
2014-12-08 23:28:07 -06:00
|
|
|
/// # Examples
|
2014-11-06 12:24:47 -05:00
|
|
|
///
|
2015-03-12 22:42:38 -04:00
|
|
|
/// ```
|
2015-02-17 23:44:55 -08:00
|
|
|
/// use std::collections::VecDeque;
|
2014-11-06 12:24:47 -05:00
|
|
|
///
|
2015-02-17 23:44:55 -08:00
|
|
|
/// let mut buf = VecDeque::new();
|
2015-01-25 22:05:03 +01:00
|
|
|
/// buf.push_back(3);
|
2014-11-06 12:24:47 -05:00
|
|
|
/// buf.push_back(4);
|
|
|
|
/// buf.push_back(5);
|
2015-07-17 23:49:13 +02:00
|
|
|
/// assert_eq!(buf.get(1), Some(&4));
|
2014-11-06 12:24:47 -05:00
|
|
|
/// ```
|
2015-01-23 21:48:20 -08:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2015-07-17 21:28:25 +02:00
|
|
|
pub fn get(&self, index: usize) -> Option<&T> {
|
|
|
|
if index < self.len() {
|
|
|
|
let idx = self.wrap_add(self.tail, index);
|
2015-07-09 21:57:21 -07:00
|
|
|
unsafe { Some(&*self.ptr().offset(idx as isize)) }
|
2014-10-19 16:19:07 -04:00
|
|
|
} else {
|
|
|
|
None
|
2014-11-06 12:24:47 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-02-17 23:44:55 -08:00
|
|
|
/// Retrieves an element in the `VecDeque` mutably by index.
|
2014-07-17 19:19:51 -04:00
|
|
|
///
|
2014-12-08 23:28:07 -06:00
|
|
|
/// # Examples
|
2014-07-17 19:19:51 -04:00
|
|
|
///
|
2015-03-12 22:42:38 -04:00
|
|
|
/// ```
|
2015-02-17 23:44:55 -08:00
|
|
|
/// use std::collections::VecDeque;
|
2014-07-17 19:19:51 -04:00
|
|
|
///
|
2015-02-17 23:44:55 -08:00
|
|
|
/// let mut buf = VecDeque::new();
|
2015-01-25 22:05:03 +01:00
|
|
|
/// buf.push_back(3);
|
2014-11-06 12:24:47 -05:00
|
|
|
/// buf.push_back(4);
|
|
|
|
/// buf.push_back(5);
|
2015-04-18 12:45:05 -04:00
|
|
|
/// if let Some(elem) = buf.get_mut(1) {
|
|
|
|
/// *elem = 7;
|
2014-11-06 12:24:47 -05:00
|
|
|
/// }
|
|
|
|
///
|
2014-08-02 18:39:39 +12:00
|
|
|
/// assert_eq!(buf[1], 7);
|
2014-07-17 19:19:51 -04:00
|
|
|
/// ```
|
2015-01-23 21:48:20 -08:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2015-07-17 21:28:25 +02:00
|
|
|
pub fn get_mut(&mut self, index: usize) -> Option<&mut T> {
|
|
|
|
if index < self.len() {
|
|
|
|
let idx = self.wrap_add(self.tail, index);
|
2015-07-09 21:57:21 -07:00
|
|
|
unsafe { Some(&mut *self.ptr().offset(idx as isize)) }
|
2014-10-19 16:19:07 -04:00
|
|
|
} else {
|
|
|
|
None
|
2014-11-06 12:24:47 -05:00
|
|
|
}
|
2013-07-10 15:27:14 +02:00
|
|
|
}
|
|
|
|
|
2014-08-04 22:48:39 +12:00
|
|
|
/// Swaps elements at indices `i` and `j`.
|
2013-09-26 09:19:26 +02:00
|
|
|
///
|
|
|
|
/// `i` and `j` may be equal.
|
|
|
|
///
|
2014-08-04 22:48:39 +12:00
|
|
|
/// Fails if there is no element with either index.
|
2014-07-17 19:19:51 -04:00
|
|
|
///
|
2014-12-08 23:28:07 -06:00
|
|
|
/// # Examples
|
2014-07-17 19:19:51 -04:00
|
|
|
///
|
2015-03-12 22:42:38 -04:00
|
|
|
/// ```
|
2015-02-17 23:44:55 -08:00
|
|
|
/// use std::collections::VecDeque;
|
2014-07-17 19:19:51 -04:00
|
|
|
///
|
2015-02-17 23:44:55 -08:00
|
|
|
/// let mut buf = VecDeque::new();
|
2015-01-25 22:05:03 +01:00
|
|
|
/// buf.push_back(3);
|
2014-11-06 12:24:47 -05:00
|
|
|
/// buf.push_back(4);
|
|
|
|
/// buf.push_back(5);
|
2014-07-17 19:19:51 -04:00
|
|
|
/// buf.swap(0, 2);
|
2014-08-02 18:39:39 +12:00
|
|
|
/// assert_eq!(buf[0], 5);
|
|
|
|
/// assert_eq!(buf[2], 3);
|
2014-07-17 19:19:51 -04:00
|
|
|
/// ```
|
2015-01-23 21:48:20 -08:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2015-02-04 21:17:19 -05:00
|
|
|
pub fn swap(&mut self, i: usize, j: usize) {
|
2013-09-26 09:19:26 +02:00
|
|
|
assert!(i < self.len());
|
|
|
|
assert!(j < self.len());
|
2015-02-19 08:33:32 +01:00
|
|
|
let ri = self.wrap_add(self.tail, i);
|
|
|
|
let rj = self.wrap_add(self.tail, j);
|
2014-10-19 16:19:07 -04:00
|
|
|
unsafe {
|
2015-11-24 11:23:48 +13:00
|
|
|
ptr::swap(self.ptr().offset(ri as isize),
|
|
|
|
self.ptr().offset(rj as isize))
|
2014-10-19 16:19:07 -04:00
|
|
|
}
|
2013-07-10 15:27:14 +02:00
|
|
|
}
|
|
|
|
|
2015-02-17 23:44:55 -08:00
|
|
|
/// Returns the number of elements the `VecDeque` can hold without
|
2014-11-06 12:24:47 -05:00
|
|
|
/// reallocating.
|
|
|
|
///
|
2014-12-08 23:28:07 -06:00
|
|
|
/// # Examples
|
2014-11-06 12:24:47 -05:00
|
|
|
///
|
|
|
|
/// ```
|
2015-02-17 23:44:55 -08:00
|
|
|
/// use std::collections::VecDeque;
|
2014-11-06 12:24:47 -05:00
|
|
|
///
|
2015-02-17 23:44:55 -08:00
|
|
|
/// let buf: VecDeque<i32> = VecDeque::with_capacity(10);
|
2014-10-19 16:19:07 -04:00
|
|
|
/// assert!(buf.capacity() >= 10);
|
2014-11-06 12:24:47 -05:00
|
|
|
/// ```
|
|
|
|
#[inline]
|
2015-01-23 21:48:20 -08:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2015-11-24 11:23:48 +13:00
|
|
|
pub fn capacity(&self) -> usize {
|
|
|
|
self.cap() - 1
|
|
|
|
}
|
2013-05-27 11:47:38 -07:00
|
|
|
|
2014-11-06 12:24:47 -05:00
|
|
|
/// Reserves the minimum capacity for exactly `additional` more elements to be inserted in the
|
2015-02-17 23:44:55 -08:00
|
|
|
/// given `VecDeque`. Does nothing if the capacity is already sufficient.
|
2014-11-06 12:24:47 -05:00
|
|
|
///
|
|
|
|
/// Note that the allocator may give the collection more space than it requests. Therefore
|
|
|
|
/// capacity can not be relied upon to be precisely minimal. Prefer `reserve` if future
|
|
|
|
/// insertions are expected.
|
|
|
|
///
|
|
|
|
/// # Panics
|
|
|
|
///
|
2015-02-04 21:17:19 -05:00
|
|
|
/// Panics if the new capacity overflows `usize`.
|
2013-05-27 11:47:38 -07:00
|
|
|
///
|
2014-12-08 23:28:07 -06:00
|
|
|
/// # Examples
|
2014-11-06 12:24:47 -05:00
|
|
|
///
|
|
|
|
/// ```
|
2015-02-17 23:44:55 -08:00
|
|
|
/// use std::collections::VecDeque;
|
2014-11-06 12:24:47 -05:00
|
|
|
///
|
2015-02-17 23:44:55 -08:00
|
|
|
/// let mut buf: VecDeque<i32> = vec![1].into_iter().collect();
|
2014-11-06 12:24:47 -05:00
|
|
|
/// buf.reserve_exact(10);
|
|
|
|
/// assert!(buf.capacity() >= 11);
|
|
|
|
/// ```
|
2015-01-23 21:48:20 -08:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2015-02-04 21:17:19 -05:00
|
|
|
pub fn reserve_exact(&mut self, additional: usize) {
|
2014-10-19 16:19:07 -04:00
|
|
|
self.reserve(additional);
|
2014-11-06 12:24:47 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Reserves capacity for at least `additional` more elements to be inserted in the given
|
2015-07-05 03:49:36 -04:00
|
|
|
/// `VecDeque`. The collection may reserve more space to avoid frequent reallocations.
|
2014-11-06 12:24:47 -05:00
|
|
|
///
|
|
|
|
/// # Panics
|
|
|
|
///
|
2015-02-04 21:17:19 -05:00
|
|
|
/// Panics if the new capacity overflows `usize`.
|
2014-11-06 12:24:47 -05:00
|
|
|
///
|
2014-12-08 23:28:07 -06:00
|
|
|
/// # Examples
|
2014-11-06 12:24:47 -05:00
|
|
|
///
|
|
|
|
/// ```
|
2015-02-17 23:44:55 -08:00
|
|
|
/// use std::collections::VecDeque;
|
2014-11-06 12:24:47 -05:00
|
|
|
///
|
2015-02-17 23:44:55 -08:00
|
|
|
/// let mut buf: VecDeque<i32> = vec![1].into_iter().collect();
|
2014-11-06 12:24:47 -05:00
|
|
|
/// buf.reserve(10);
|
|
|
|
/// assert!(buf.capacity() >= 11);
|
|
|
|
/// ```
|
2015-01-23 21:48:20 -08:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2015-02-04 21:17:19 -05:00
|
|
|
pub fn reserve(&mut self, additional: usize) {
|
2015-07-09 21:57:21 -07:00
|
|
|
let old_cap = self.cap();
|
|
|
|
let used_cap = self.len() + 1;
|
2015-11-24 11:23:48 +13:00
|
|
|
let new_cap = used_cap.checked_add(additional)
|
|
|
|
.and_then(|needed_cap| needed_cap.checked_next_power_of_two())
|
|
|
|
.expect("capacity overflow");
|
2015-07-09 21:57:21 -07:00
|
|
|
|
|
|
|
if new_cap > self.capacity() {
|
|
|
|
self.buf.reserve_exact(used_cap, new_cap - used_cap);
|
2015-11-24 11:23:48 +13:00
|
|
|
unsafe {
|
|
|
|
self.handle_cap_increase(old_cap);
|
|
|
|
}
|
2015-01-05 15:48:58 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-07-05 03:49:36 -04:00
|
|
|
/// Shrinks the capacity of the `VecDeque` as much as possible.
|
2015-01-05 15:48:58 +01:00
|
|
|
///
|
|
|
|
/// It will drop down as close as possible to the length but the allocator may still inform the
|
2015-07-05 03:49:36 -04:00
|
|
|
/// `VecDeque` that there is space for a few more elements.
|
2015-01-05 15:48:58 +01:00
|
|
|
///
|
|
|
|
/// # Examples
|
|
|
|
///
|
|
|
|
/// ```
|
2015-02-17 23:44:55 -08:00
|
|
|
/// use std::collections::VecDeque;
|
2015-01-05 15:48:58 +01:00
|
|
|
///
|
2015-02-17 23:44:55 -08:00
|
|
|
/// let mut buf = VecDeque::with_capacity(15);
|
2015-02-06 13:57:13 -05:00
|
|
|
/// buf.extend(0..4);
|
2015-01-05 15:48:58 +01:00
|
|
|
/// assert_eq!(buf.capacity(), 15);
|
|
|
|
/// buf.shrink_to_fit();
|
|
|
|
/// assert!(buf.capacity() >= 4);
|
|
|
|
/// ```
|
std: Stabilize library APIs for 1.5
This commit stabilizes and deprecates library APIs whose FCP has closed in the
last cycle, specifically:
Stabilized APIs:
* `fs::canonicalize`
* `Path::{metadata, symlink_metadata, canonicalize, read_link, read_dir, exists,
is_file, is_dir}` - all moved to inherent methods from the `PathExt` trait.
* `Formatter::fill`
* `Formatter::width`
* `Formatter::precision`
* `Formatter::sign_plus`
* `Formatter::sign_minus`
* `Formatter::alternate`
* `Formatter::sign_aware_zero_pad`
* `string::ParseError`
* `Utf8Error::valid_up_to`
* `Iterator::{cmp, partial_cmp, eq, ne, lt, le, gt, ge}`
* `<[T]>::split_{first,last}{,_mut}`
* `Condvar::wait_timeout` - note that `wait_timeout_ms` is not yet deprecated
but will be once 1.5 is released.
* `str::{R,}MatchIndices`
* `str::{r,}match_indices`
* `char::from_u32_unchecked`
* `VecDeque::insert`
* `VecDeque::shrink_to_fit`
* `VecDeque::as_slices`
* `VecDeque::as_mut_slices`
* `VecDeque::swap_remove_front` - (renamed from `swap_front_remove`)
* `VecDeque::swap_remove_back` - (renamed from `swap_back_remove`)
* `Vec::resize`
* `str::slice_mut_unchecked`
* `FileTypeExt`
* `FileTypeExt::{is_block_device, is_char_device, is_fifo, is_socket}`
* `BinaryHeap::from` - `from_vec` deprecated in favor of this
* `BinaryHeap::into_vec` - plus a `Into` impl
* `BinaryHeap::into_sorted_vec`
Deprecated APIs
* `slice::ref_slice`
* `slice::mut_ref_slice`
* `iter::{range_inclusive, RangeInclusive}`
* `std::dynamic_lib`
Closes #27706
Closes #27725
cc #27726 (align not stabilized yet)
Closes #27734
Closes #27737
Closes #27742
Closes #27743
Closes #27772
Closes #27774
Closes #27777
Closes #27781
cc #27788 (a few remaining methods though)
Closes #27790
Closes #27793
Closes #27796
Closes #27810
cc #28147 (not all parts stabilized)
2015-10-22 16:28:45 -07:00
|
|
|
#[stable(feature = "deque_extras_15", since = "1.5.0")]
|
2015-01-05 15:48:58 +01:00
|
|
|
pub fn shrink_to_fit(&mut self) {
|
|
|
|
// +1 since the ringbuffer always leaves one space empty
|
2015-07-05 03:49:36 -04:00
|
|
|
// len + 1 can't overflow for an existing, well-formed ringbuffer.
|
2015-01-05 15:48:58 +01:00
|
|
|
let target_cap = cmp::max(self.len() + 1, MINIMUM_CAPACITY + 1).next_power_of_two();
|
2015-07-09 21:57:21 -07:00
|
|
|
if target_cap < self.cap() {
|
2015-01-05 15:48:58 +01:00
|
|
|
// There are three cases of interest:
|
|
|
|
// All elements are out of desired bounds
|
|
|
|
// Elements are contiguous, and head is out of desired bounds
|
|
|
|
// Elements are discontiguous, and tail is out of desired bounds
|
|
|
|
//
|
|
|
|
// At all other times, element positions are unaffected.
|
|
|
|
//
|
|
|
|
// Indicates that elements at the head should be moved.
|
|
|
|
let head_outside = self.head == 0 || self.head >= target_cap;
|
|
|
|
// Move elements from out of desired bounds (positions after target_cap)
|
|
|
|
if self.tail >= target_cap && head_outside {
|
|
|
|
// T H
|
|
|
|
// [. . . . . . . . o o o o o o o . ]
|
|
|
|
// T H
|
|
|
|
// [o o o o o o o . ]
|
|
|
|
unsafe {
|
|
|
|
self.copy_nonoverlapping(0, self.tail, self.len());
|
|
|
|
}
|
|
|
|
self.head = self.len();
|
|
|
|
self.tail = 0;
|
|
|
|
} else if self.tail != 0 && self.tail < target_cap && head_outside {
|
|
|
|
// T H
|
|
|
|
// [. . . o o o o o o o . . . . . . ]
|
|
|
|
// H T
|
|
|
|
// [o o . o o o o o ]
|
2015-02-19 08:33:32 +01:00
|
|
|
let len = self.wrap_sub(self.head, target_cap);
|
2015-01-05 15:48:58 +01:00
|
|
|
unsafe {
|
|
|
|
self.copy_nonoverlapping(0, target_cap, len);
|
|
|
|
}
|
|
|
|
self.head = len;
|
|
|
|
debug_assert!(self.head < self.tail);
|
|
|
|
} else if self.tail >= target_cap {
|
|
|
|
// H T
|
|
|
|
// [o o o o o . . . . . . . . . o o ]
|
|
|
|
// H T
|
|
|
|
// [o o o o o . o o ]
|
2015-02-19 08:33:32 +01:00
|
|
|
debug_assert!(self.wrap_sub(self.head, 1) < target_cap);
|
2015-07-09 21:57:21 -07:00
|
|
|
let len = self.cap() - self.tail;
|
2015-01-05 15:48:58 +01:00
|
|
|
let new_tail = target_cap - len;
|
|
|
|
unsafe {
|
|
|
|
self.copy_nonoverlapping(new_tail, self.tail, len);
|
|
|
|
}
|
|
|
|
self.tail = new_tail;
|
2014-11-10 21:16:29 -05:00
|
|
|
debug_assert!(self.head < self.tail);
|
2014-10-19 16:19:07 -04:00
|
|
|
}
|
2015-01-05 15:48:58 +01:00
|
|
|
|
2015-07-09 21:57:21 -07:00
|
|
|
self.buf.shrink_to_fit(target_cap);
|
|
|
|
|
|
|
|
debug_assert!(self.head < self.cap());
|
|
|
|
debug_assert!(self.tail < self.cap());
|
|
|
|
debug_assert!(self.cap().count_ones() == 1);
|
2014-10-19 16:19:07 -04:00
|
|
|
}
|
2013-05-27 11:47:38 -07:00
|
|
|
}
|
2013-06-25 15:08:47 -04:00
|
|
|
|
2015-07-05 03:49:36 -04:00
|
|
|
/// Shortens a `VecDeque`, dropping excess elements from the back.
|
2015-01-05 15:48:58 +01:00
|
|
|
///
|
2015-07-05 03:49:36 -04:00
|
|
|
/// If `len` is greater than the `VecDeque`'s current length, this has no
|
2015-01-05 15:48:58 +01:00
|
|
|
/// effect.
|
|
|
|
///
|
|
|
|
/// # Examples
|
|
|
|
///
|
|
|
|
/// ```
|
2015-07-27 10:50:19 -04:00
|
|
|
/// #![feature(deque_extras)]
|
|
|
|
///
|
2015-02-17 23:44:55 -08:00
|
|
|
/// use std::collections::VecDeque;
|
2015-01-05 15:48:58 +01:00
|
|
|
///
|
2015-02-17 23:44:55 -08:00
|
|
|
/// let mut buf = VecDeque::new();
|
2015-01-25 22:05:03 +01:00
|
|
|
/// buf.push_back(5);
|
|
|
|
/// buf.push_back(10);
|
2015-01-05 15:48:58 +01:00
|
|
|
/// buf.push_back(15);
|
|
|
|
/// buf.truncate(1);
|
|
|
|
/// assert_eq!(buf.len(), 1);
|
|
|
|
/// assert_eq!(Some(&5), buf.get(0));
|
|
|
|
/// ```
|
2015-06-09 14:39:23 -07:00
|
|
|
#[unstable(feature = "deque_extras",
|
2015-08-12 22:19:51 -07:00
|
|
|
reason = "matches collection reform specification; waiting on panic semantics",
|
|
|
|
issue = "27788")]
|
2015-02-04 21:17:19 -05:00
|
|
|
pub fn truncate(&mut self, len: usize) {
|
2015-01-26 16:05:07 -05:00
|
|
|
for _ in len..self.len() {
|
2015-01-05 15:48:58 +01:00
|
|
|
self.pop_back();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-08-04 22:48:39 +12:00
|
|
|
/// Returns a front-to-back iterator.
|
2014-07-17 19:19:51 -04:00
|
|
|
///
|
2014-12-08 23:28:07 -06:00
|
|
|
/// # Examples
|
2014-07-17 19:19:51 -04:00
|
|
|
///
|
2015-03-12 22:42:38 -04:00
|
|
|
/// ```
|
2015-02-17 23:44:55 -08:00
|
|
|
/// use std::collections::VecDeque;
|
2014-07-17 19:19:51 -04:00
|
|
|
///
|
2015-02-17 23:44:55 -08:00
|
|
|
/// let mut buf = VecDeque::new();
|
2015-01-25 22:05:03 +01:00
|
|
|
/// buf.push_back(5);
|
2014-11-06 12:24:47 -05:00
|
|
|
/// buf.push_back(3);
|
|
|
|
/// buf.push_back(4);
|
2014-08-06 11:59:40 +02:00
|
|
|
/// let b: &[_] = &[&5, &3, &4];
|
2015-03-30 19:22:46 +03:00
|
|
|
/// let c: Vec<&i32> = buf.iter().collect();
|
|
|
|
/// assert_eq!(&c[..], b);
|
2014-07-17 19:19:51 -04:00
|
|
|
/// ```
|
2015-01-23 21:48:20 -08:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2014-12-19 21:52:10 +01:00
|
|
|
pub fn iter(&self) -> Iter<T> {
|
|
|
|
Iter {
|
2014-10-19 16:19:07 -04:00
|
|
|
tail: self.tail,
|
|
|
|
head: self.head,
|
2015-11-24 11:23:48 +13:00
|
|
|
ring: unsafe { self.buffer_as_slice() },
|
2014-10-19 16:19:07 -04:00
|
|
|
}
|
2013-06-25 15:08:47 -04:00
|
|
|
}
|
2013-06-26 18:14:35 -04:00
|
|
|
|
2014-12-15 10:22:49 +01:00
|
|
|
/// Returns a front-to-back iterator that returns mutable references.
|
2014-07-17 19:19:51 -04:00
|
|
|
///
|
2014-12-08 23:28:07 -06:00
|
|
|
/// # Examples
|
2014-07-17 19:19:51 -04:00
|
|
|
///
|
2015-03-12 22:42:38 -04:00
|
|
|
/// ```
|
2015-02-17 23:44:55 -08:00
|
|
|
/// use std::collections::VecDeque;
|
2014-07-17 19:19:51 -04:00
|
|
|
///
|
2015-02-17 23:44:55 -08:00
|
|
|
/// let mut buf = VecDeque::new();
|
2015-01-25 22:05:03 +01:00
|
|
|
/// buf.push_back(5);
|
2014-11-06 12:24:47 -05:00
|
|
|
/// buf.push_back(3);
|
|
|
|
/// buf.push_back(4);
|
2014-09-14 20:27:36 -07:00
|
|
|
/// for num in buf.iter_mut() {
|
2014-07-17 19:19:51 -04:00
|
|
|
/// *num = *num - 2;
|
|
|
|
/// }
|
2014-08-06 11:59:40 +02:00
|
|
|
/// let b: &[_] = &[&mut 3, &mut 1, &mut 2];
|
2015-03-26 09:57:58 -07:00
|
|
|
/// assert_eq!(&buf.iter_mut().collect::<Vec<&mut i32>>()[..], b);
|
2014-07-17 19:19:51 -04:00
|
|
|
/// ```
|
2015-01-23 21:48:20 -08:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2015-02-05 13:48:20 -05:00
|
|
|
pub fn iter_mut(&mut self) -> IterMut<T> {
|
2014-12-19 21:52:10 +01:00
|
|
|
IterMut {
|
2014-10-19 16:19:07 -04:00
|
|
|
tail: self.tail,
|
|
|
|
head: self.head,
|
2015-02-25 18:11:23 +08:00
|
|
|
ring: unsafe { self.buffer_as_mut_slice() },
|
2013-11-16 17:29:39 -05:00
|
|
|
}
|
2013-06-25 15:08:47 -04:00
|
|
|
}
|
2014-10-30 13:43:24 -07:00
|
|
|
|
2014-12-15 23:01:58 -05:00
|
|
|
/// Returns a pair of slices which contain, in order, the contents of the
|
2015-02-17 23:44:55 -08:00
|
|
|
/// `VecDeque`.
|
2014-12-15 23:01:58 -05:00
|
|
|
#[inline]
|
std: Stabilize library APIs for 1.5
This commit stabilizes and deprecates library APIs whose FCP has closed in the
last cycle, specifically:
Stabilized APIs:
* `fs::canonicalize`
* `Path::{metadata, symlink_metadata, canonicalize, read_link, read_dir, exists,
is_file, is_dir}` - all moved to inherent methods from the `PathExt` trait.
* `Formatter::fill`
* `Formatter::width`
* `Formatter::precision`
* `Formatter::sign_plus`
* `Formatter::sign_minus`
* `Formatter::alternate`
* `Formatter::sign_aware_zero_pad`
* `string::ParseError`
* `Utf8Error::valid_up_to`
* `Iterator::{cmp, partial_cmp, eq, ne, lt, le, gt, ge}`
* `<[T]>::split_{first,last}{,_mut}`
* `Condvar::wait_timeout` - note that `wait_timeout_ms` is not yet deprecated
but will be once 1.5 is released.
* `str::{R,}MatchIndices`
* `str::{r,}match_indices`
* `char::from_u32_unchecked`
* `VecDeque::insert`
* `VecDeque::shrink_to_fit`
* `VecDeque::as_slices`
* `VecDeque::as_mut_slices`
* `VecDeque::swap_remove_front` - (renamed from `swap_front_remove`)
* `VecDeque::swap_remove_back` - (renamed from `swap_back_remove`)
* `Vec::resize`
* `str::slice_mut_unchecked`
* `FileTypeExt`
* `FileTypeExt::{is_block_device, is_char_device, is_fifo, is_socket}`
* `BinaryHeap::from` - `from_vec` deprecated in favor of this
* `BinaryHeap::into_vec` - plus a `Into` impl
* `BinaryHeap::into_sorted_vec`
Deprecated APIs
* `slice::ref_slice`
* `slice::mut_ref_slice`
* `iter::{range_inclusive, RangeInclusive}`
* `std::dynamic_lib`
Closes #27706
Closes #27725
cc #27726 (align not stabilized yet)
Closes #27734
Closes #27737
Closes #27742
Closes #27743
Closes #27772
Closes #27774
Closes #27777
Closes #27781
cc #27788 (a few remaining methods though)
Closes #27790
Closes #27793
Closes #27796
Closes #27810
cc #28147 (not all parts stabilized)
2015-10-22 16:28:45 -07:00
|
|
|
#[stable(feature = "deque_extras_15", since = "1.5.0")]
|
2015-02-05 13:48:20 -05:00
|
|
|
pub fn as_slices(&self) -> (&[T], &[T]) {
|
2014-12-15 23:01:58 -05:00
|
|
|
unsafe {
|
|
|
|
let contiguous = self.is_contiguous();
|
|
|
|
let buf = self.buffer_as_slice();
|
|
|
|
if contiguous {
|
|
|
|
let (empty, buf) = buf.split_at(0);
|
2015-01-07 11:58:31 -05:00
|
|
|
(&buf[self.tail..self.head], empty)
|
2014-12-15 23:01:58 -05:00
|
|
|
} else {
|
|
|
|
let (mid, right) = buf.split_at(self.tail);
|
|
|
|
let (left, _) = mid.split_at(self.head);
|
|
|
|
(right, left)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns a pair of slices which contain, in order, the contents of the
|
2015-02-17 23:44:55 -08:00
|
|
|
/// `VecDeque`.
|
2014-12-15 23:01:58 -05:00
|
|
|
#[inline]
|
std: Stabilize library APIs for 1.5
This commit stabilizes and deprecates library APIs whose FCP has closed in the
last cycle, specifically:
Stabilized APIs:
* `fs::canonicalize`
* `Path::{metadata, symlink_metadata, canonicalize, read_link, read_dir, exists,
is_file, is_dir}` - all moved to inherent methods from the `PathExt` trait.
* `Formatter::fill`
* `Formatter::width`
* `Formatter::precision`
* `Formatter::sign_plus`
* `Formatter::sign_minus`
* `Formatter::alternate`
* `Formatter::sign_aware_zero_pad`
* `string::ParseError`
* `Utf8Error::valid_up_to`
* `Iterator::{cmp, partial_cmp, eq, ne, lt, le, gt, ge}`
* `<[T]>::split_{first,last}{,_mut}`
* `Condvar::wait_timeout` - note that `wait_timeout_ms` is not yet deprecated
but will be once 1.5 is released.
* `str::{R,}MatchIndices`
* `str::{r,}match_indices`
* `char::from_u32_unchecked`
* `VecDeque::insert`
* `VecDeque::shrink_to_fit`
* `VecDeque::as_slices`
* `VecDeque::as_mut_slices`
* `VecDeque::swap_remove_front` - (renamed from `swap_front_remove`)
* `VecDeque::swap_remove_back` - (renamed from `swap_back_remove`)
* `Vec::resize`
* `str::slice_mut_unchecked`
* `FileTypeExt`
* `FileTypeExt::{is_block_device, is_char_device, is_fifo, is_socket}`
* `BinaryHeap::from` - `from_vec` deprecated in favor of this
* `BinaryHeap::into_vec` - plus a `Into` impl
* `BinaryHeap::into_sorted_vec`
Deprecated APIs
* `slice::ref_slice`
* `slice::mut_ref_slice`
* `iter::{range_inclusive, RangeInclusive}`
* `std::dynamic_lib`
Closes #27706
Closes #27725
cc #27726 (align not stabilized yet)
Closes #27734
Closes #27737
Closes #27742
Closes #27743
Closes #27772
Closes #27774
Closes #27777
Closes #27781
cc #27788 (a few remaining methods though)
Closes #27790
Closes #27793
Closes #27796
Closes #27810
cc #28147 (not all parts stabilized)
2015-10-22 16:28:45 -07:00
|
|
|
#[stable(feature = "deque_extras_15", since = "1.5.0")]
|
2015-02-05 13:48:20 -05:00
|
|
|
pub fn as_mut_slices(&mut self) -> (&mut [T], &mut [T]) {
|
2014-12-15 23:01:58 -05:00
|
|
|
unsafe {
|
|
|
|
let contiguous = self.is_contiguous();
|
|
|
|
let head = self.head;
|
|
|
|
let tail = self.tail;
|
|
|
|
let buf = self.buffer_as_mut_slice();
|
|
|
|
|
|
|
|
if contiguous {
|
|
|
|
let (empty, buf) = buf.split_at_mut(0);
|
2015-11-24 11:23:48 +13:00
|
|
|
(&mut buf[tail..head], empty)
|
2014-12-15 23:01:58 -05:00
|
|
|
} else {
|
|
|
|
let (mid, right) = buf.split_at_mut(tail);
|
|
|
|
let (left, _) = mid.split_at_mut(head);
|
|
|
|
|
|
|
|
(right, left)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-02-17 23:44:55 -08:00
|
|
|
/// Returns the number of elements in the `VecDeque`.
|
2014-10-30 13:43:24 -07:00
|
|
|
///
|
2014-12-08 23:28:07 -06:00
|
|
|
/// # Examples
|
2014-10-30 13:43:24 -07:00
|
|
|
///
|
|
|
|
/// ```
|
2015-02-17 23:44:55 -08:00
|
|
|
/// use std::collections::VecDeque;
|
2014-10-30 13:43:24 -07:00
|
|
|
///
|
2015-02-17 23:44:55 -08:00
|
|
|
/// let mut v = VecDeque::new();
|
2014-10-30 13:43:24 -07:00
|
|
|
/// assert_eq!(v.len(), 0);
|
2015-01-25 22:05:03 +01:00
|
|
|
/// v.push_back(1);
|
2014-10-30 13:43:24 -07:00
|
|
|
/// assert_eq!(v.len(), 1);
|
|
|
|
/// ```
|
2015-01-23 21:48:20 -08:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2015-11-24 11:23:48 +13:00
|
|
|
pub fn len(&self) -> usize {
|
|
|
|
count(self.tail, self.head, self.cap())
|
|
|
|
}
|
2014-10-30 13:43:24 -07:00
|
|
|
|
|
|
|
/// Returns true if the buffer contains no elements
|
|
|
|
///
|
2014-12-08 23:28:07 -06:00
|
|
|
/// # Examples
|
2014-10-30 13:43:24 -07:00
|
|
|
///
|
|
|
|
/// ```
|
2015-02-17 23:44:55 -08:00
|
|
|
/// use std::collections::VecDeque;
|
2014-10-30 13:43:24 -07:00
|
|
|
///
|
2015-02-17 23:44:55 -08:00
|
|
|
/// let mut v = VecDeque::new();
|
2014-10-30 13:43:24 -07:00
|
|
|
/// assert!(v.is_empty());
|
2015-01-25 22:05:03 +01:00
|
|
|
/// v.push_front(1);
|
2014-10-30 13:43:24 -07:00
|
|
|
/// assert!(!v.is_empty());
|
|
|
|
/// ```
|
2015-01-23 21:48:20 -08:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2015-11-24 11:23:48 +13:00
|
|
|
pub fn is_empty(&self) -> bool {
|
|
|
|
self.len() == 0
|
|
|
|
}
|
2014-10-30 13:43:24 -07:00
|
|
|
|
2015-08-06 19:03:14 -07:00
|
|
|
/// Create a draining iterator that removes the specified range in the
|
2015-12-09 00:04:54 +02:00
|
|
|
/// `VecDeque` and yields the removed items.
|
2015-08-06 19:03:14 -07:00
|
|
|
///
|
2015-12-09 00:04:54 +02:00
|
|
|
/// Note 1: The element range is removed even if the iterator is not
|
|
|
|
/// consumed until the end.
|
|
|
|
///
|
|
|
|
/// Note 2: It is unspecified how many elements are removed from the deque,
|
2015-08-06 19:03:14 -07:00
|
|
|
/// if the `Drain` value is not dropped, but the borrow it holds expires
|
|
|
|
/// (eg. due to mem::forget).
|
|
|
|
///
|
|
|
|
/// # Panics
|
|
|
|
///
|
|
|
|
/// Panics if the starting point is greater than the end point or if
|
|
|
|
/// the end point is greater than the length of the vector.
|
2014-12-16 17:45:03 -05:00
|
|
|
///
|
|
|
|
/// # Examples
|
|
|
|
///
|
|
|
|
/// ```
|
2015-02-17 23:44:55 -08:00
|
|
|
/// use std::collections::VecDeque;
|
2015-12-09 00:04:54 +02:00
|
|
|
|
|
|
|
/// let mut v: VecDeque<_> = vec![1, 2, 3].into_iter().collect();
|
|
|
|
/// assert_eq!(vec![3].into_iter().collect::<VecDeque<_>>(), v.drain(2..).collect());
|
|
|
|
/// assert_eq!(vec![1, 2].into_iter().collect::<VecDeque<_>>(), v);
|
2014-12-16 17:45:03 -05:00
|
|
|
///
|
2015-12-09 00:04:54 +02:00
|
|
|
/// // A full range clears all contents
|
|
|
|
/// v.drain(..);
|
2014-12-16 17:45:03 -05:00
|
|
|
/// assert!(v.is_empty());
|
|
|
|
/// ```
|
|
|
|
#[inline]
|
2015-12-02 17:31:49 -08:00
|
|
|
#[stable(feature = "drain", since = "1.6.0")]
|
2015-11-24 11:23:48 +13:00
|
|
|
pub fn drain<R>(&mut self, range: R) -> Drain<T>
|
|
|
|
where R: RangeArgument<usize>
|
|
|
|
{
|
2015-08-06 19:03:14 -07:00
|
|
|
// Memory safety
|
|
|
|
//
|
|
|
|
// When the Drain is first created, the source deque is shortened to
|
|
|
|
// make sure no uninitialized or moved-from elements are accessible at
|
|
|
|
// all if the Drain's destructor never gets to run.
|
|
|
|
//
|
|
|
|
// Drain will ptr::read out the values to remove.
|
|
|
|
// When finished, the remaining data will be copied back to cover the hole,
|
|
|
|
// and the head/tail values will be restored correctly.
|
|
|
|
//
|
|
|
|
let len = self.len();
|
|
|
|
let start = *range.start().unwrap_or(&0);
|
|
|
|
let end = *range.end().unwrap_or(&len);
|
|
|
|
assert!(start <= end, "drain lower bound was too large");
|
|
|
|
assert!(end <= len, "drain upper bound was too large");
|
|
|
|
|
|
|
|
// The deque's elements are parted into three segments:
|
|
|
|
// * self.tail -> drain_tail
|
|
|
|
// * drain_tail -> drain_head
|
|
|
|
// * drain_head -> self.head
|
|
|
|
//
|
|
|
|
// T = self.tail; H = self.head; t = drain_tail; h = drain_head
|
|
|
|
//
|
|
|
|
// We store drain_tail as self.head, and drain_head and self.head as
|
|
|
|
// after_tail and after_head respectively on the Drain. This also
|
|
|
|
// truncates the effective array such that if the Drain is leaked, we
|
|
|
|
// have forgotten about the potentially moved values after the start of
|
|
|
|
// the drain.
|
|
|
|
//
|
|
|
|
// T t h H
|
|
|
|
// [. . . o o x x o o . . .]
|
|
|
|
//
|
|
|
|
let drain_tail = self.wrap_add(self.tail, start);
|
|
|
|
let drain_head = self.wrap_add(self.tail, end);
|
|
|
|
let head = self.head;
|
|
|
|
|
|
|
|
// "forget" about the values after the start of the drain until after
|
|
|
|
// the drain is complete and the Drain destructor is run.
|
|
|
|
self.head = drain_tail;
|
|
|
|
|
2014-12-16 17:45:03 -05:00
|
|
|
Drain {
|
2015-08-06 19:03:14 -07:00
|
|
|
deque: self as *mut _,
|
|
|
|
after_tail: drain_head,
|
|
|
|
after_head: head,
|
|
|
|
iter: Iter {
|
|
|
|
tail: drain_tail,
|
|
|
|
head: drain_head,
|
|
|
|
ring: unsafe { self.buffer_as_mut_slice() },
|
|
|
|
},
|
2014-12-16 17:45:03 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-10-30 13:43:24 -07:00
|
|
|
/// Clears the buffer, removing all values.
|
|
|
|
///
|
2014-12-08 23:28:07 -06:00
|
|
|
/// # Examples
|
2014-10-30 13:43:24 -07:00
|
|
|
///
|
|
|
|
/// ```
|
2015-02-17 23:44:55 -08:00
|
|
|
/// use std::collections::VecDeque;
|
2014-10-30 13:43:24 -07:00
|
|
|
///
|
2015-02-17 23:44:55 -08:00
|
|
|
/// let mut v = VecDeque::new();
|
2015-01-25 22:05:03 +01:00
|
|
|
/// v.push_back(1);
|
2014-10-30 13:43:24 -07:00
|
|
|
/// v.clear();
|
|
|
|
/// assert!(v.is_empty());
|
|
|
|
/// ```
|
2015-01-23 21:48:20 -08:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2014-12-16 17:45:03 -05:00
|
|
|
#[inline]
|
2014-10-30 13:43:24 -07:00
|
|
|
pub fn clear(&mut self) {
|
2015-08-06 19:03:14 -07:00
|
|
|
self.drain(..);
|
2014-10-30 13:43:24 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Provides a reference to the front element, or `None` if the sequence is
|
|
|
|
/// empty.
|
|
|
|
///
|
2014-12-08 23:28:07 -06:00
|
|
|
/// # Examples
|
2014-10-30 13:43:24 -07:00
|
|
|
///
|
|
|
|
/// ```
|
2015-02-17 23:44:55 -08:00
|
|
|
/// use std::collections::VecDeque;
|
2014-10-30 13:43:24 -07:00
|
|
|
///
|
2015-02-17 23:44:55 -08:00
|
|
|
/// let mut d = VecDeque::new();
|
2014-10-30 13:43:24 -07:00
|
|
|
/// assert_eq!(d.front(), None);
|
|
|
|
///
|
2015-01-25 22:05:03 +01:00
|
|
|
/// d.push_back(1);
|
|
|
|
/// d.push_back(2);
|
|
|
|
/// assert_eq!(d.front(), Some(&1));
|
2014-10-30 13:43:24 -07:00
|
|
|
/// ```
|
2015-01-23 21:48:20 -08:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2014-10-30 13:43:24 -07:00
|
|
|
pub fn front(&self) -> Option<&T> {
|
2015-11-24 11:23:48 +13:00
|
|
|
if !self.is_empty() {
|
|
|
|
Some(&self[0])
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
2014-10-30 13:43:24 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Provides a mutable reference to the front element, or `None` if the
|
|
|
|
/// sequence is empty.
|
|
|
|
///
|
2014-12-08 23:28:07 -06:00
|
|
|
/// # Examples
|
2014-10-30 13:43:24 -07:00
|
|
|
///
|
|
|
|
/// ```
|
2015-02-17 23:44:55 -08:00
|
|
|
/// use std::collections::VecDeque;
|
2014-10-30 13:43:24 -07:00
|
|
|
///
|
2015-02-17 23:44:55 -08:00
|
|
|
/// let mut d = VecDeque::new();
|
2014-10-30 13:43:24 -07:00
|
|
|
/// assert_eq!(d.front_mut(), None);
|
|
|
|
///
|
2015-01-25 22:05:03 +01:00
|
|
|
/// d.push_back(1);
|
|
|
|
/// d.push_back(2);
|
2014-10-30 13:43:24 -07:00
|
|
|
/// match d.front_mut() {
|
2015-01-25 22:05:03 +01:00
|
|
|
/// Some(x) => *x = 9,
|
2014-10-30 13:43:24 -07:00
|
|
|
/// None => (),
|
|
|
|
/// }
|
2015-01-25 22:05:03 +01:00
|
|
|
/// assert_eq!(d.front(), Some(&9));
|
2014-10-30 13:43:24 -07:00
|
|
|
/// ```
|
2015-01-23 21:48:20 -08:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2014-10-30 13:43:24 -07:00
|
|
|
pub fn front_mut(&mut self) -> Option<&mut T> {
|
2015-11-24 11:23:48 +13:00
|
|
|
if !self.is_empty() {
|
|
|
|
Some(&mut self[0])
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
2014-10-30 13:43:24 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Provides a reference to the back element, or `None` if the sequence is
|
|
|
|
/// empty.
|
|
|
|
///
|
2014-12-08 23:28:07 -06:00
|
|
|
/// # Examples
|
2014-10-30 13:43:24 -07:00
|
|
|
///
|
|
|
|
/// ```
|
2015-02-17 23:44:55 -08:00
|
|
|
/// use std::collections::VecDeque;
|
2014-10-30 13:43:24 -07:00
|
|
|
///
|
2015-02-17 23:44:55 -08:00
|
|
|
/// let mut d = VecDeque::new();
|
2014-10-30 13:43:24 -07:00
|
|
|
/// assert_eq!(d.back(), None);
|
|
|
|
///
|
2015-01-25 22:05:03 +01:00
|
|
|
/// d.push_back(1);
|
|
|
|
/// d.push_back(2);
|
|
|
|
/// assert_eq!(d.back(), Some(&2));
|
2014-10-30 13:43:24 -07:00
|
|
|
/// ```
|
2015-01-23 21:48:20 -08:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2014-10-30 13:43:24 -07:00
|
|
|
pub fn back(&self) -> Option<&T> {
|
2015-11-24 11:23:48 +13:00
|
|
|
if !self.is_empty() {
|
|
|
|
Some(&self[self.len() - 1])
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
2014-10-30 13:43:24 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Provides a mutable reference to the back element, or `None` if the
|
|
|
|
/// sequence is empty.
|
|
|
|
///
|
2014-12-08 23:28:07 -06:00
|
|
|
/// # Examples
|
2014-10-30 13:43:24 -07:00
|
|
|
///
|
|
|
|
/// ```
|
2015-02-17 23:44:55 -08:00
|
|
|
/// use std::collections::VecDeque;
|
2014-10-30 13:43:24 -07:00
|
|
|
///
|
2015-02-17 23:44:55 -08:00
|
|
|
/// let mut d = VecDeque::new();
|
2014-10-30 13:43:24 -07:00
|
|
|
/// assert_eq!(d.back(), None);
|
|
|
|
///
|
2015-01-25 22:05:03 +01:00
|
|
|
/// d.push_back(1);
|
|
|
|
/// d.push_back(2);
|
2014-10-30 13:43:24 -07:00
|
|
|
/// match d.back_mut() {
|
2015-01-25 22:05:03 +01:00
|
|
|
/// Some(x) => *x = 9,
|
2014-10-30 13:43:24 -07:00
|
|
|
/// None => (),
|
|
|
|
/// }
|
2015-01-25 22:05:03 +01:00
|
|
|
/// assert_eq!(d.back(), Some(&9));
|
2014-10-30 13:43:24 -07:00
|
|
|
/// ```
|
2015-01-23 21:48:20 -08:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2014-10-30 13:43:24 -07:00
|
|
|
pub fn back_mut(&mut self) -> Option<&mut T> {
|
2014-10-19 16:19:07 -04:00
|
|
|
let len = self.len();
|
2015-11-24 11:23:48 +13:00
|
|
|
if !self.is_empty() {
|
|
|
|
Some(&mut self[len - 1])
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
2014-10-30 13:43:24 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Removes the first element and returns it, or `None` if the sequence is
|
|
|
|
/// empty.
|
|
|
|
///
|
2014-12-08 23:28:07 -06:00
|
|
|
/// # Examples
|
2014-10-30 13:43:24 -07:00
|
|
|
///
|
|
|
|
/// ```
|
2015-02-17 23:44:55 -08:00
|
|
|
/// use std::collections::VecDeque;
|
2014-10-30 13:43:24 -07:00
|
|
|
///
|
2015-02-17 23:44:55 -08:00
|
|
|
/// let mut d = VecDeque::new();
|
2015-01-25 22:05:03 +01:00
|
|
|
/// d.push_back(1);
|
|
|
|
/// d.push_back(2);
|
2014-10-30 13:43:24 -07:00
|
|
|
///
|
2015-01-25 22:05:03 +01:00
|
|
|
/// assert_eq!(d.pop_front(), Some(1));
|
|
|
|
/// assert_eq!(d.pop_front(), Some(2));
|
2014-10-30 13:43:24 -07:00
|
|
|
/// assert_eq!(d.pop_front(), None);
|
|
|
|
/// ```
|
2015-01-23 21:48:20 -08:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2014-10-30 13:43:24 -07:00
|
|
|
pub fn pop_front(&mut self) -> Option<T> {
|
2014-10-19 16:19:07 -04:00
|
|
|
if self.is_empty() {
|
|
|
|
None
|
|
|
|
} else {
|
|
|
|
let tail = self.tail;
|
2015-02-19 08:33:32 +01:00
|
|
|
self.tail = self.wrap_add(self.tail, 1);
|
2014-10-19 16:19:07 -04:00
|
|
|
unsafe { Some(self.buffer_read(tail)) }
|
2014-10-30 13:43:24 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Inserts an element first in the sequence.
|
|
|
|
///
|
2014-12-08 23:28:07 -06:00
|
|
|
/// # Examples
|
2014-10-30 13:43:24 -07:00
|
|
|
///
|
|
|
|
/// ```
|
2015-02-17 23:44:55 -08:00
|
|
|
/// use std::collections::VecDeque;
|
2014-10-30 13:43:24 -07:00
|
|
|
///
|
2015-02-17 23:44:55 -08:00
|
|
|
/// let mut d = VecDeque::new();
|
2015-01-25 22:05:03 +01:00
|
|
|
/// d.push_front(1);
|
|
|
|
/// d.push_front(2);
|
|
|
|
/// assert_eq!(d.front(), Some(&2));
|
2014-10-30 13:43:24 -07:00
|
|
|
/// ```
|
2015-01-23 21:48:20 -08:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2015-07-17 21:28:25 +02:00
|
|
|
pub fn push_front(&mut self, value: T) {
|
2014-11-10 21:16:29 -05:00
|
|
|
if self.is_full() {
|
2015-07-09 21:57:21 -07:00
|
|
|
let old_cap = self.cap();
|
|
|
|
self.buf.double();
|
2015-11-24 11:23:48 +13:00
|
|
|
unsafe {
|
|
|
|
self.handle_cap_increase(old_cap);
|
|
|
|
}
|
2014-11-10 21:16:29 -05:00
|
|
|
debug_assert!(!self.is_full());
|
|
|
|
}
|
2014-10-19 16:19:07 -04:00
|
|
|
|
2015-02-19 08:33:32 +01:00
|
|
|
self.tail = self.wrap_sub(self.tail, 1);
|
2014-10-19 16:19:07 -04:00
|
|
|
let tail = self.tail;
|
2015-11-24 11:23:48 +13:00
|
|
|
unsafe {
|
|
|
|
self.buffer_write(tail, value);
|
|
|
|
}
|
2014-10-30 13:43:24 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Appends an element to the back of a buffer
|
|
|
|
///
|
2014-12-08 23:28:07 -06:00
|
|
|
/// # Examples
|
2014-10-30 13:43:24 -07:00
|
|
|
///
|
2015-03-12 22:42:38 -04:00
|
|
|
/// ```
|
2015-02-17 23:44:55 -08:00
|
|
|
/// use std::collections::VecDeque;
|
2014-10-30 13:43:24 -07:00
|
|
|
///
|
2015-02-17 23:44:55 -08:00
|
|
|
/// let mut buf = VecDeque::new();
|
2015-01-25 22:05:03 +01:00
|
|
|
/// buf.push_back(1);
|
2014-11-06 12:24:47 -05:00
|
|
|
/// buf.push_back(3);
|
2014-10-30 13:43:24 -07:00
|
|
|
/// assert_eq!(3, *buf.back().unwrap());
|
|
|
|
/// ```
|
2015-01-23 21:48:20 -08:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2015-07-17 21:28:25 +02:00
|
|
|
pub fn push_back(&mut self, value: T) {
|
2014-11-10 21:16:29 -05:00
|
|
|
if self.is_full() {
|
2015-07-09 21:57:21 -07:00
|
|
|
let old_cap = self.cap();
|
|
|
|
self.buf.double();
|
2015-11-24 11:23:48 +13:00
|
|
|
unsafe {
|
|
|
|
self.handle_cap_increase(old_cap);
|
|
|
|
}
|
2014-11-10 21:16:29 -05:00
|
|
|
debug_assert!(!self.is_full());
|
|
|
|
}
|
2014-10-19 16:19:07 -04:00
|
|
|
|
|
|
|
let head = self.head;
|
2015-02-19 08:33:32 +01:00
|
|
|
self.head = self.wrap_add(self.head, 1);
|
2015-07-17 21:28:25 +02:00
|
|
|
unsafe { self.buffer_write(head, value) }
|
2014-10-30 13:43:24 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Removes the last element from a buffer and returns it, or `None` if
|
|
|
|
/// it is empty.
|
|
|
|
///
|
2014-12-08 23:28:07 -06:00
|
|
|
/// # Examples
|
2014-10-30 13:43:24 -07:00
|
|
|
///
|
2015-03-12 22:42:38 -04:00
|
|
|
/// ```
|
2015-02-17 23:44:55 -08:00
|
|
|
/// use std::collections::VecDeque;
|
2014-10-30 13:43:24 -07:00
|
|
|
///
|
2015-02-17 23:44:55 -08:00
|
|
|
/// let mut buf = VecDeque::new();
|
2014-11-06 12:24:47 -05:00
|
|
|
/// assert_eq!(buf.pop_back(), None);
|
2015-01-25 22:05:03 +01:00
|
|
|
/// buf.push_back(1);
|
2014-11-06 12:24:47 -05:00
|
|
|
/// buf.push_back(3);
|
|
|
|
/// assert_eq!(buf.pop_back(), Some(3));
|
2014-10-30 13:43:24 -07:00
|
|
|
/// ```
|
2015-01-23 21:48:20 -08:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2014-11-06 12:24:47 -05:00
|
|
|
pub fn pop_back(&mut self) -> Option<T> {
|
2014-10-19 16:19:07 -04:00
|
|
|
if self.is_empty() {
|
2014-10-30 13:43:24 -07:00
|
|
|
None
|
2014-10-19 16:19:07 -04:00
|
|
|
} else {
|
2015-02-19 08:33:32 +01:00
|
|
|
self.head = self.wrap_sub(self.head, 1);
|
2014-10-19 16:19:07 -04:00
|
|
|
let head = self.head;
|
|
|
|
unsafe { Some(self.buffer_read(head)) }
|
2014-10-30 13:43:24 -07:00
|
|
|
}
|
|
|
|
}
|
2014-12-03 11:12:30 -06:00
|
|
|
|
2014-12-15 23:01:58 -05:00
|
|
|
#[inline]
|
|
|
|
fn is_contiguous(&self) -> bool {
|
|
|
|
self.tail <= self.head
|
|
|
|
}
|
|
|
|
|
2015-07-05 03:49:36 -04:00
|
|
|
/// Removes an element from anywhere in the `VecDeque` and returns it, replacing it with the
|
|
|
|
/// last element.
|
2015-01-05 15:48:58 +01:00
|
|
|
///
|
|
|
|
/// This does not preserve ordering, but is O(1).
|
|
|
|
///
|
|
|
|
/// Returns `None` if `index` is out of bounds.
|
|
|
|
///
|
|
|
|
/// # Examples
|
|
|
|
///
|
|
|
|
/// ```
|
2015-02-17 23:44:55 -08:00
|
|
|
/// use std::collections::VecDeque;
|
2015-01-05 15:48:58 +01:00
|
|
|
///
|
2015-02-17 23:44:55 -08:00
|
|
|
/// let mut buf = VecDeque::new();
|
std: Stabilize library APIs for 1.5
This commit stabilizes and deprecates library APIs whose FCP has closed in the
last cycle, specifically:
Stabilized APIs:
* `fs::canonicalize`
* `Path::{metadata, symlink_metadata, canonicalize, read_link, read_dir, exists,
is_file, is_dir}` - all moved to inherent methods from the `PathExt` trait.
* `Formatter::fill`
* `Formatter::width`
* `Formatter::precision`
* `Formatter::sign_plus`
* `Formatter::sign_minus`
* `Formatter::alternate`
* `Formatter::sign_aware_zero_pad`
* `string::ParseError`
* `Utf8Error::valid_up_to`
* `Iterator::{cmp, partial_cmp, eq, ne, lt, le, gt, ge}`
* `<[T]>::split_{first,last}{,_mut}`
* `Condvar::wait_timeout` - note that `wait_timeout_ms` is not yet deprecated
but will be once 1.5 is released.
* `str::{R,}MatchIndices`
* `str::{r,}match_indices`
* `char::from_u32_unchecked`
* `VecDeque::insert`
* `VecDeque::shrink_to_fit`
* `VecDeque::as_slices`
* `VecDeque::as_mut_slices`
* `VecDeque::swap_remove_front` - (renamed from `swap_front_remove`)
* `VecDeque::swap_remove_back` - (renamed from `swap_back_remove`)
* `Vec::resize`
* `str::slice_mut_unchecked`
* `FileTypeExt`
* `FileTypeExt::{is_block_device, is_char_device, is_fifo, is_socket}`
* `BinaryHeap::from` - `from_vec` deprecated in favor of this
* `BinaryHeap::into_vec` - plus a `Into` impl
* `BinaryHeap::into_sorted_vec`
Deprecated APIs
* `slice::ref_slice`
* `slice::mut_ref_slice`
* `iter::{range_inclusive, RangeInclusive}`
* `std::dynamic_lib`
Closes #27706
Closes #27725
cc #27726 (align not stabilized yet)
Closes #27734
Closes #27737
Closes #27742
Closes #27743
Closes #27772
Closes #27774
Closes #27777
Closes #27781
cc #27788 (a few remaining methods though)
Closes #27790
Closes #27793
Closes #27796
Closes #27810
cc #28147 (not all parts stabilized)
2015-10-22 16:28:45 -07:00
|
|
|
/// assert_eq!(buf.swap_remove_back(0), None);
|
2015-07-17 23:49:13 +02:00
|
|
|
/// buf.push_back(1);
|
|
|
|
/// buf.push_back(2);
|
|
|
|
/// buf.push_back(3);
|
|
|
|
///
|
std: Stabilize library APIs for 1.5
This commit stabilizes and deprecates library APIs whose FCP has closed in the
last cycle, specifically:
Stabilized APIs:
* `fs::canonicalize`
* `Path::{metadata, symlink_metadata, canonicalize, read_link, read_dir, exists,
is_file, is_dir}` - all moved to inherent methods from the `PathExt` trait.
* `Formatter::fill`
* `Formatter::width`
* `Formatter::precision`
* `Formatter::sign_plus`
* `Formatter::sign_minus`
* `Formatter::alternate`
* `Formatter::sign_aware_zero_pad`
* `string::ParseError`
* `Utf8Error::valid_up_to`
* `Iterator::{cmp, partial_cmp, eq, ne, lt, le, gt, ge}`
* `<[T]>::split_{first,last}{,_mut}`
* `Condvar::wait_timeout` - note that `wait_timeout_ms` is not yet deprecated
but will be once 1.5 is released.
* `str::{R,}MatchIndices`
* `str::{r,}match_indices`
* `char::from_u32_unchecked`
* `VecDeque::insert`
* `VecDeque::shrink_to_fit`
* `VecDeque::as_slices`
* `VecDeque::as_mut_slices`
* `VecDeque::swap_remove_front` - (renamed from `swap_front_remove`)
* `VecDeque::swap_remove_back` - (renamed from `swap_back_remove`)
* `Vec::resize`
* `str::slice_mut_unchecked`
* `FileTypeExt`
* `FileTypeExt::{is_block_device, is_char_device, is_fifo, is_socket}`
* `BinaryHeap::from` - `from_vec` deprecated in favor of this
* `BinaryHeap::into_vec` - plus a `Into` impl
* `BinaryHeap::into_sorted_vec`
Deprecated APIs
* `slice::ref_slice`
* `slice::mut_ref_slice`
* `iter::{range_inclusive, RangeInclusive}`
* `std::dynamic_lib`
Closes #27706
Closes #27725
cc #27726 (align not stabilized yet)
Closes #27734
Closes #27737
Closes #27742
Closes #27743
Closes #27772
Closes #27774
Closes #27777
Closes #27781
cc #27788 (a few remaining methods though)
Closes #27790
Closes #27793
Closes #27796
Closes #27810
cc #28147 (not all parts stabilized)
2015-10-22 16:28:45 -07:00
|
|
|
/// assert_eq!(buf.swap_remove_back(0), Some(1));
|
2015-07-17 23:49:13 +02:00
|
|
|
/// assert_eq!(buf.len(), 2);
|
|
|
|
/// assert_eq!(buf[0], 3);
|
|
|
|
/// assert_eq!(buf[1], 2);
|
2015-01-05 15:48:58 +01:00
|
|
|
/// ```
|
std: Stabilize library APIs for 1.5
This commit stabilizes and deprecates library APIs whose FCP has closed in the
last cycle, specifically:
Stabilized APIs:
* `fs::canonicalize`
* `Path::{metadata, symlink_metadata, canonicalize, read_link, read_dir, exists,
is_file, is_dir}` - all moved to inherent methods from the `PathExt` trait.
* `Formatter::fill`
* `Formatter::width`
* `Formatter::precision`
* `Formatter::sign_plus`
* `Formatter::sign_minus`
* `Formatter::alternate`
* `Formatter::sign_aware_zero_pad`
* `string::ParseError`
* `Utf8Error::valid_up_to`
* `Iterator::{cmp, partial_cmp, eq, ne, lt, le, gt, ge}`
* `<[T]>::split_{first,last}{,_mut}`
* `Condvar::wait_timeout` - note that `wait_timeout_ms` is not yet deprecated
but will be once 1.5 is released.
* `str::{R,}MatchIndices`
* `str::{r,}match_indices`
* `char::from_u32_unchecked`
* `VecDeque::insert`
* `VecDeque::shrink_to_fit`
* `VecDeque::as_slices`
* `VecDeque::as_mut_slices`
* `VecDeque::swap_remove_front` - (renamed from `swap_front_remove`)
* `VecDeque::swap_remove_back` - (renamed from `swap_back_remove`)
* `Vec::resize`
* `str::slice_mut_unchecked`
* `FileTypeExt`
* `FileTypeExt::{is_block_device, is_char_device, is_fifo, is_socket}`
* `BinaryHeap::from` - `from_vec` deprecated in favor of this
* `BinaryHeap::into_vec` - plus a `Into` impl
* `BinaryHeap::into_sorted_vec`
Deprecated APIs
* `slice::ref_slice`
* `slice::mut_ref_slice`
* `iter::{range_inclusive, RangeInclusive}`
* `std::dynamic_lib`
Closes #27706
Closes #27725
cc #27726 (align not stabilized yet)
Closes #27734
Closes #27737
Closes #27742
Closes #27743
Closes #27772
Closes #27774
Closes #27777
Closes #27781
cc #27788 (a few remaining methods though)
Closes #27790
Closes #27793
Closes #27796
Closes #27810
cc #28147 (not all parts stabilized)
2015-10-22 16:28:45 -07:00
|
|
|
#[stable(feature = "deque_extras_15", since = "1.5.0")]
|
|
|
|
pub fn swap_remove_back(&mut self, index: usize) -> Option<T> {
|
2015-01-05 15:48:58 +01:00
|
|
|
let length = self.len();
|
|
|
|
if length > 0 && index < length - 1 {
|
|
|
|
self.swap(index, length - 1);
|
|
|
|
} else if index >= length {
|
|
|
|
return None;
|
|
|
|
}
|
|
|
|
self.pop_back()
|
|
|
|
}
|
|
|
|
|
2015-07-05 03:49:36 -04:00
|
|
|
/// Removes an element from anywhere in the `VecDeque` and returns it,
|
2015-06-10 13:33:52 -07:00
|
|
|
/// replacing it with the first element.
|
2015-01-05 15:48:58 +01:00
|
|
|
///
|
|
|
|
/// This does not preserve ordering, but is O(1).
|
|
|
|
///
|
|
|
|
/// Returns `None` if `index` is out of bounds.
|
|
|
|
///
|
|
|
|
/// # Examples
|
|
|
|
///
|
|
|
|
/// ```
|
2015-02-17 23:44:55 -08:00
|
|
|
/// use std::collections::VecDeque;
|
2015-01-05 15:48:58 +01:00
|
|
|
///
|
2015-02-17 23:44:55 -08:00
|
|
|
/// let mut buf = VecDeque::new();
|
std: Stabilize library APIs for 1.5
This commit stabilizes and deprecates library APIs whose FCP has closed in the
last cycle, specifically:
Stabilized APIs:
* `fs::canonicalize`
* `Path::{metadata, symlink_metadata, canonicalize, read_link, read_dir, exists,
is_file, is_dir}` - all moved to inherent methods from the `PathExt` trait.
* `Formatter::fill`
* `Formatter::width`
* `Formatter::precision`
* `Formatter::sign_plus`
* `Formatter::sign_minus`
* `Formatter::alternate`
* `Formatter::sign_aware_zero_pad`
* `string::ParseError`
* `Utf8Error::valid_up_to`
* `Iterator::{cmp, partial_cmp, eq, ne, lt, le, gt, ge}`
* `<[T]>::split_{first,last}{,_mut}`
* `Condvar::wait_timeout` - note that `wait_timeout_ms` is not yet deprecated
but will be once 1.5 is released.
* `str::{R,}MatchIndices`
* `str::{r,}match_indices`
* `char::from_u32_unchecked`
* `VecDeque::insert`
* `VecDeque::shrink_to_fit`
* `VecDeque::as_slices`
* `VecDeque::as_mut_slices`
* `VecDeque::swap_remove_front` - (renamed from `swap_front_remove`)
* `VecDeque::swap_remove_back` - (renamed from `swap_back_remove`)
* `Vec::resize`
* `str::slice_mut_unchecked`
* `FileTypeExt`
* `FileTypeExt::{is_block_device, is_char_device, is_fifo, is_socket}`
* `BinaryHeap::from` - `from_vec` deprecated in favor of this
* `BinaryHeap::into_vec` - plus a `Into` impl
* `BinaryHeap::into_sorted_vec`
Deprecated APIs
* `slice::ref_slice`
* `slice::mut_ref_slice`
* `iter::{range_inclusive, RangeInclusive}`
* `std::dynamic_lib`
Closes #27706
Closes #27725
cc #27726 (align not stabilized yet)
Closes #27734
Closes #27737
Closes #27742
Closes #27743
Closes #27772
Closes #27774
Closes #27777
Closes #27781
cc #27788 (a few remaining methods though)
Closes #27790
Closes #27793
Closes #27796
Closes #27810
cc #28147 (not all parts stabilized)
2015-10-22 16:28:45 -07:00
|
|
|
/// assert_eq!(buf.swap_remove_front(0), None);
|
2015-07-17 23:49:13 +02:00
|
|
|
/// buf.push_back(1);
|
|
|
|
/// buf.push_back(2);
|
|
|
|
/// buf.push_back(3);
|
|
|
|
///
|
std: Stabilize library APIs for 1.5
This commit stabilizes and deprecates library APIs whose FCP has closed in the
last cycle, specifically:
Stabilized APIs:
* `fs::canonicalize`
* `Path::{metadata, symlink_metadata, canonicalize, read_link, read_dir, exists,
is_file, is_dir}` - all moved to inherent methods from the `PathExt` trait.
* `Formatter::fill`
* `Formatter::width`
* `Formatter::precision`
* `Formatter::sign_plus`
* `Formatter::sign_minus`
* `Formatter::alternate`
* `Formatter::sign_aware_zero_pad`
* `string::ParseError`
* `Utf8Error::valid_up_to`
* `Iterator::{cmp, partial_cmp, eq, ne, lt, le, gt, ge}`
* `<[T]>::split_{first,last}{,_mut}`
* `Condvar::wait_timeout` - note that `wait_timeout_ms` is not yet deprecated
but will be once 1.5 is released.
* `str::{R,}MatchIndices`
* `str::{r,}match_indices`
* `char::from_u32_unchecked`
* `VecDeque::insert`
* `VecDeque::shrink_to_fit`
* `VecDeque::as_slices`
* `VecDeque::as_mut_slices`
* `VecDeque::swap_remove_front` - (renamed from `swap_front_remove`)
* `VecDeque::swap_remove_back` - (renamed from `swap_back_remove`)
* `Vec::resize`
* `str::slice_mut_unchecked`
* `FileTypeExt`
* `FileTypeExt::{is_block_device, is_char_device, is_fifo, is_socket}`
* `BinaryHeap::from` - `from_vec` deprecated in favor of this
* `BinaryHeap::into_vec` - plus a `Into` impl
* `BinaryHeap::into_sorted_vec`
Deprecated APIs
* `slice::ref_slice`
* `slice::mut_ref_slice`
* `iter::{range_inclusive, RangeInclusive}`
* `std::dynamic_lib`
Closes #27706
Closes #27725
cc #27726 (align not stabilized yet)
Closes #27734
Closes #27737
Closes #27742
Closes #27743
Closes #27772
Closes #27774
Closes #27777
Closes #27781
cc #27788 (a few remaining methods though)
Closes #27790
Closes #27793
Closes #27796
Closes #27810
cc #28147 (not all parts stabilized)
2015-10-22 16:28:45 -07:00
|
|
|
/// assert_eq!(buf.swap_remove_front(2), Some(3));
|
2015-07-17 23:49:13 +02:00
|
|
|
/// assert_eq!(buf.len(), 2);
|
|
|
|
/// assert_eq!(buf[0], 2);
|
|
|
|
/// assert_eq!(buf[1], 1);
|
2015-01-05 15:48:58 +01:00
|
|
|
/// ```
|
std: Stabilize library APIs for 1.5
This commit stabilizes and deprecates library APIs whose FCP has closed in the
last cycle, specifically:
Stabilized APIs:
* `fs::canonicalize`
* `Path::{metadata, symlink_metadata, canonicalize, read_link, read_dir, exists,
is_file, is_dir}` - all moved to inherent methods from the `PathExt` trait.
* `Formatter::fill`
* `Formatter::width`
* `Formatter::precision`
* `Formatter::sign_plus`
* `Formatter::sign_minus`
* `Formatter::alternate`
* `Formatter::sign_aware_zero_pad`
* `string::ParseError`
* `Utf8Error::valid_up_to`
* `Iterator::{cmp, partial_cmp, eq, ne, lt, le, gt, ge}`
* `<[T]>::split_{first,last}{,_mut}`
* `Condvar::wait_timeout` - note that `wait_timeout_ms` is not yet deprecated
but will be once 1.5 is released.
* `str::{R,}MatchIndices`
* `str::{r,}match_indices`
* `char::from_u32_unchecked`
* `VecDeque::insert`
* `VecDeque::shrink_to_fit`
* `VecDeque::as_slices`
* `VecDeque::as_mut_slices`
* `VecDeque::swap_remove_front` - (renamed from `swap_front_remove`)
* `VecDeque::swap_remove_back` - (renamed from `swap_back_remove`)
* `Vec::resize`
* `str::slice_mut_unchecked`
* `FileTypeExt`
* `FileTypeExt::{is_block_device, is_char_device, is_fifo, is_socket}`
* `BinaryHeap::from` - `from_vec` deprecated in favor of this
* `BinaryHeap::into_vec` - plus a `Into` impl
* `BinaryHeap::into_sorted_vec`
Deprecated APIs
* `slice::ref_slice`
* `slice::mut_ref_slice`
* `iter::{range_inclusive, RangeInclusive}`
* `std::dynamic_lib`
Closes #27706
Closes #27725
cc #27726 (align not stabilized yet)
Closes #27734
Closes #27737
Closes #27742
Closes #27743
Closes #27772
Closes #27774
Closes #27777
Closes #27781
cc #27788 (a few remaining methods though)
Closes #27790
Closes #27793
Closes #27796
Closes #27810
cc #28147 (not all parts stabilized)
2015-10-22 16:28:45 -07:00
|
|
|
#[stable(feature = "deque_extras_15", since = "1.5.0")]
|
|
|
|
pub fn swap_remove_front(&mut self, index: usize) -> Option<T> {
|
2015-01-05 15:48:58 +01:00
|
|
|
let length = self.len();
|
|
|
|
if length > 0 && index < length && index != 0 {
|
|
|
|
self.swap(index, 0);
|
|
|
|
} else if index >= length {
|
|
|
|
return None;
|
|
|
|
}
|
|
|
|
self.pop_front()
|
|
|
|
}
|
|
|
|
|
2015-07-17 21:28:25 +02:00
|
|
|
/// Inserts an element at `index` within the `VecDeque`. Whichever
|
2014-12-03 11:12:30 -06:00
|
|
|
/// end is closer to the insertion point will be moved to make room,
|
|
|
|
/// and all the affected elements will be moved to new positions.
|
|
|
|
///
|
|
|
|
/// # Panics
|
|
|
|
///
|
2015-07-17 21:28:25 +02:00
|
|
|
/// Panics if `index` is greater than `VecDeque`'s length
|
2014-12-03 11:12:30 -06:00
|
|
|
///
|
2015-01-05 15:48:58 +01:00
|
|
|
/// # Examples
|
2015-03-12 22:42:38 -04:00
|
|
|
/// ```
|
2015-02-17 23:44:55 -08:00
|
|
|
/// use std::collections::VecDeque;
|
2014-12-03 11:12:30 -06:00
|
|
|
///
|
2015-02-17 23:44:55 -08:00
|
|
|
/// let mut buf = VecDeque::new();
|
2015-01-25 22:05:03 +01:00
|
|
|
/// buf.push_back(10);
|
2014-12-03 11:12:30 -06:00
|
|
|
/// buf.push_back(12);
|
2015-07-17 20:55:11 +02:00
|
|
|
/// buf.insert(1, 11);
|
2014-12-03 11:12:30 -06:00
|
|
|
/// assert_eq!(Some(&11), buf.get(1));
|
|
|
|
/// ```
|
std: Stabilize library APIs for 1.5
This commit stabilizes and deprecates library APIs whose FCP has closed in the
last cycle, specifically:
Stabilized APIs:
* `fs::canonicalize`
* `Path::{metadata, symlink_metadata, canonicalize, read_link, read_dir, exists,
is_file, is_dir}` - all moved to inherent methods from the `PathExt` trait.
* `Formatter::fill`
* `Formatter::width`
* `Formatter::precision`
* `Formatter::sign_plus`
* `Formatter::sign_minus`
* `Formatter::alternate`
* `Formatter::sign_aware_zero_pad`
* `string::ParseError`
* `Utf8Error::valid_up_to`
* `Iterator::{cmp, partial_cmp, eq, ne, lt, le, gt, ge}`
* `<[T]>::split_{first,last}{,_mut}`
* `Condvar::wait_timeout` - note that `wait_timeout_ms` is not yet deprecated
but will be once 1.5 is released.
* `str::{R,}MatchIndices`
* `str::{r,}match_indices`
* `char::from_u32_unchecked`
* `VecDeque::insert`
* `VecDeque::shrink_to_fit`
* `VecDeque::as_slices`
* `VecDeque::as_mut_slices`
* `VecDeque::swap_remove_front` - (renamed from `swap_front_remove`)
* `VecDeque::swap_remove_back` - (renamed from `swap_back_remove`)
* `Vec::resize`
* `str::slice_mut_unchecked`
* `FileTypeExt`
* `FileTypeExt::{is_block_device, is_char_device, is_fifo, is_socket}`
* `BinaryHeap::from` - `from_vec` deprecated in favor of this
* `BinaryHeap::into_vec` - plus a `Into` impl
* `BinaryHeap::into_sorted_vec`
Deprecated APIs
* `slice::ref_slice`
* `slice::mut_ref_slice`
* `iter::{range_inclusive, RangeInclusive}`
* `std::dynamic_lib`
Closes #27706
Closes #27725
cc #27726 (align not stabilized yet)
Closes #27734
Closes #27737
Closes #27742
Closes #27743
Closes #27772
Closes #27774
Closes #27777
Closes #27781
cc #27788 (a few remaining methods though)
Closes #27790
Closes #27793
Closes #27796
Closes #27810
cc #28147 (not all parts stabilized)
2015-10-22 16:28:45 -07:00
|
|
|
#[stable(feature = "deque_extras_15", since = "1.5.0")]
|
2015-07-17 21:28:25 +02:00
|
|
|
pub fn insert(&mut self, index: usize, value: T) {
|
|
|
|
assert!(index <= self.len(), "index out of bounds");
|
2014-12-03 11:12:30 -06:00
|
|
|
if self.is_full() {
|
2015-07-09 21:57:21 -07:00
|
|
|
let old_cap = self.cap();
|
|
|
|
self.buf.double();
|
2015-11-24 11:23:48 +13:00
|
|
|
unsafe {
|
|
|
|
self.handle_cap_increase(old_cap);
|
|
|
|
}
|
2014-12-03 11:12:30 -06:00
|
|
|
debug_assert!(!self.is_full());
|
|
|
|
}
|
|
|
|
|
|
|
|
// Move the least number of elements in the ring buffer and insert
|
|
|
|
// the given object
|
|
|
|
//
|
|
|
|
// At most len/2 - 1 elements will be moved. O(min(n, n-i))
|
|
|
|
//
|
|
|
|
// There are three main cases:
|
|
|
|
// Elements are contiguous
|
|
|
|
// - special case when tail is 0
|
|
|
|
// Elements are discontiguous and the insert is in the tail section
|
|
|
|
// Elements are discontiguous and the insert is in the head section
|
|
|
|
//
|
|
|
|
// For each of those there are two more cases:
|
|
|
|
// Insert is closer to tail
|
|
|
|
// Insert is closer to head
|
|
|
|
//
|
|
|
|
// Key: H - self.head
|
|
|
|
// T - self.tail
|
|
|
|
// o - Valid element
|
|
|
|
// I - Insertion element
|
|
|
|
// A - The element that should be after the insertion point
|
|
|
|
// M - Indicates element was moved
|
|
|
|
|
2015-07-17 21:28:25 +02:00
|
|
|
let idx = self.wrap_add(self.tail, index);
|
2014-12-03 11:12:30 -06:00
|
|
|
|
2015-07-17 21:28:25 +02:00
|
|
|
let distance_to_tail = index;
|
|
|
|
let distance_to_head = self.len() - index;
|
2014-12-03 11:12:30 -06:00
|
|
|
|
2014-12-15 23:01:58 -05:00
|
|
|
let contiguous = self.is_contiguous();
|
2014-12-03 11:12:30 -06:00
|
|
|
|
2015-11-24 11:23:48 +13:00
|
|
|
match (contiguous,
|
|
|
|
distance_to_tail <= distance_to_head,
|
|
|
|
idx >= self.tail) {
|
2015-07-17 21:28:25 +02:00
|
|
|
(true, true, _) if index == 0 => {
|
2014-12-03 11:12:30 -06:00
|
|
|
// push_front
|
|
|
|
//
|
|
|
|
// T
|
|
|
|
// I H
|
|
|
|
// [A o o o o o o . . . . . . . . .]
|
|
|
|
//
|
|
|
|
// H T
|
|
|
|
// [A o o o o o o o . . . . . I]
|
|
|
|
//
|
|
|
|
|
2015-02-19 08:33:32 +01:00
|
|
|
self.tail = self.wrap_sub(self.tail, 1);
|
2015-11-24 11:23:48 +13:00
|
|
|
}
|
|
|
|
(true, true, _) => {
|
|
|
|
unsafe {
|
|
|
|
// contiguous, insert closer to tail:
|
|
|
|
//
|
|
|
|
// T I H
|
|
|
|
// [. . . o o A o o o o . . . . . .]
|
|
|
|
//
|
|
|
|
// T H
|
|
|
|
// [. . o o I A o o o o . . . . . .]
|
|
|
|
// M M
|
|
|
|
//
|
|
|
|
// contiguous, insert closer to tail and tail is 0:
|
|
|
|
//
|
|
|
|
//
|
|
|
|
// T I H
|
|
|
|
// [o o A o o o o . . . . . . . . .]
|
|
|
|
//
|
|
|
|
// H T
|
|
|
|
// [o I A o o o o o . . . . . . . o]
|
|
|
|
// M M
|
|
|
|
|
|
|
|
let new_tail = self.wrap_sub(self.tail, 1);
|
|
|
|
|
|
|
|
self.copy(new_tail, self.tail, 1);
|
|
|
|
// Already moved the tail, so we only copy `index - 1` elements.
|
|
|
|
self.copy(self.tail, self.tail + 1, index - 1);
|
|
|
|
|
|
|
|
self.tail = new_tail;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
(true, false, _) => {
|
|
|
|
unsafe {
|
|
|
|
// contiguous, insert closer to head:
|
|
|
|
//
|
|
|
|
// T I H
|
|
|
|
// [. . . o o o o A o o . . . . . .]
|
|
|
|
//
|
|
|
|
// T H
|
|
|
|
// [. . . o o o o I A o o . . . . .]
|
|
|
|
// M M M
|
|
|
|
|
|
|
|
self.copy(idx + 1, idx, self.head - idx);
|
|
|
|
self.head = self.wrap_add(self.head, 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
(false, true, true) => {
|
|
|
|
unsafe {
|
|
|
|
// discontiguous, insert closer to tail, tail section:
|
|
|
|
//
|
|
|
|
// H T I
|
|
|
|
// [o o o o o o . . . . . o o A o o]
|
|
|
|
//
|
|
|
|
// H T
|
|
|
|
// [o o o o o o . . . . o o I A o o]
|
|
|
|
// M M
|
|
|
|
|
|
|
|
self.copy(self.tail - 1, self.tail, index);
|
|
|
|
self.tail -= 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
(false, false, true) => {
|
|
|
|
unsafe {
|
|
|
|
// discontiguous, insert closer to head, tail section:
|
|
|
|
//
|
|
|
|
// H T I
|
|
|
|
// [o o . . . . . . . o o o o o A o]
|
|
|
|
//
|
|
|
|
// H T
|
|
|
|
// [o o o . . . . . . o o o o o I A]
|
|
|
|
// M M M M
|
2014-12-17 00:37:55 +01:00
|
|
|
|
2015-11-24 11:23:48 +13:00
|
|
|
// copy elements up to new head
|
|
|
|
self.copy(1, 0, self.head);
|
2014-12-03 11:12:30 -06:00
|
|
|
|
2015-11-24 11:23:48 +13:00
|
|
|
// copy last element into empty spot at bottom of buffer
|
|
|
|
self.copy(0, self.cap() - 1, 1);
|
2014-12-03 11:12:30 -06:00
|
|
|
|
2015-11-24 11:23:48 +13:00
|
|
|
// move elements from idx to end forward not including ^ element
|
|
|
|
self.copy(idx + 1, idx, self.cap() - 1 - idx);
|
2014-12-17 00:37:55 +01:00
|
|
|
|
2015-11-24 11:23:48 +13:00
|
|
|
self.head += 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
(false, true, false) if idx == 0 => {
|
|
|
|
unsafe {
|
|
|
|
// discontiguous, insert is closer to tail, head section,
|
|
|
|
// and is at index zero in the internal buffer:
|
|
|
|
//
|
|
|
|
// I H T
|
|
|
|
// [A o o o o o o o o o . . . o o o]
|
|
|
|
//
|
|
|
|
// H T
|
|
|
|
// [A o o o o o o o o o . . o o o I]
|
|
|
|
// M M M
|
|
|
|
|
|
|
|
// copy elements up to new tail
|
|
|
|
self.copy(self.tail - 1, self.tail, self.cap() - self.tail);
|
|
|
|
|
|
|
|
// copy last element into empty spot at bottom of buffer
|
|
|
|
self.copy(self.cap() - 1, 0, 1);
|
2014-12-03 11:12:30 -06:00
|
|
|
|
2015-11-24 11:23:48 +13:00
|
|
|
self.tail -= 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
(false, true, false) => {
|
|
|
|
unsafe {
|
|
|
|
// discontiguous, insert closer to tail, head section:
|
|
|
|
//
|
|
|
|
// I H T
|
|
|
|
// [o o o A o o o o o o . . . o o o]
|
|
|
|
//
|
|
|
|
// H T
|
|
|
|
// [o o I A o o o o o o . . o o o o]
|
|
|
|
// M M M M M M
|
|
|
|
|
|
|
|
// copy elements up to new tail
|
|
|
|
self.copy(self.tail - 1, self.tail, self.cap() - self.tail);
|
|
|
|
|
|
|
|
// copy last element into empty spot at bottom of buffer
|
|
|
|
self.copy(self.cap() - 1, 0, 1);
|
2014-12-17 00:37:55 +01:00
|
|
|
|
2015-11-24 11:23:48 +13:00
|
|
|
// move elements from idx-1 to end forward not including ^ element
|
|
|
|
self.copy(0, 1, idx - 1);
|
2014-12-03 11:12:30 -06:00
|
|
|
|
2015-11-24 11:23:48 +13:00
|
|
|
self.tail -= 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
(false, false, false) => {
|
|
|
|
unsafe {
|
|
|
|
// discontiguous, insert closer to head, head section:
|
|
|
|
//
|
|
|
|
// I H T
|
|
|
|
// [o o o o A o o . . . . . . o o o]
|
|
|
|
//
|
|
|
|
// H T
|
|
|
|
// [o o o o I A o o . . . . . o o o]
|
|
|
|
// M M M
|
|
|
|
|
|
|
|
self.copy(idx + 1, idx, self.head - idx);
|
|
|
|
self.head += 1;
|
|
|
|
}
|
2014-12-03 11:12:30 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// tail might've been changed so we need to recalculate
|
2015-07-17 21:28:25 +02:00
|
|
|
let new_idx = self.wrap_add(self.tail, index);
|
2014-12-03 11:12:30 -06:00
|
|
|
unsafe {
|
2015-07-17 21:28:25 +02:00
|
|
|
self.buffer_write(new_idx, value);
|
2014-12-03 11:12:30 -06:00
|
|
|
}
|
|
|
|
}
|
2014-12-17 00:37:55 +01:00
|
|
|
|
2015-07-17 21:28:25 +02:00
|
|
|
/// Removes and returns the element at `index` from the `VecDeque`.
|
2014-12-17 00:37:55 +01:00
|
|
|
/// Whichever end is closer to the removal point will be moved to make
|
|
|
|
/// room, and all the affected elements will be moved to new positions.
|
2015-07-17 21:28:25 +02:00
|
|
|
/// Returns `None` if `index` is out of bounds.
|
2014-12-17 00:37:55 +01:00
|
|
|
///
|
2015-01-05 15:48:58 +01:00
|
|
|
/// # Examples
|
2015-03-12 22:42:38 -04:00
|
|
|
/// ```
|
2015-02-17 23:44:55 -08:00
|
|
|
/// use std::collections::VecDeque;
|
2014-12-17 00:37:55 +01:00
|
|
|
///
|
2015-02-17 23:44:55 -08:00
|
|
|
/// let mut buf = VecDeque::new();
|
2015-07-17 23:49:13 +02:00
|
|
|
/// buf.push_back(1);
|
|
|
|
/// buf.push_back(2);
|
|
|
|
/// buf.push_back(3);
|
|
|
|
///
|
|
|
|
/// assert_eq!(buf.remove(1), Some(2));
|
|
|
|
/// assert_eq!(buf.get(1), Some(&3));
|
2014-12-17 00:37:55 +01:00
|
|
|
/// ```
|
2015-01-23 21:48:20 -08:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2015-07-17 21:28:25 +02:00
|
|
|
pub fn remove(&mut self, index: usize) -> Option<T> {
|
|
|
|
if self.is_empty() || self.len() <= index {
|
2014-12-17 00:37:55 +01:00
|
|
|
return None;
|
|
|
|
}
|
|
|
|
|
|
|
|
// There are three main cases:
|
|
|
|
// Elements are contiguous
|
|
|
|
// Elements are discontiguous and the removal is in the tail section
|
|
|
|
// Elements are discontiguous and the removal is in the head section
|
|
|
|
// - special case when elements are technically contiguous,
|
|
|
|
// but self.head = 0
|
|
|
|
//
|
|
|
|
// For each of those there are two more cases:
|
|
|
|
// Insert is closer to tail
|
|
|
|
// Insert is closer to head
|
|
|
|
//
|
|
|
|
// Key: H - self.head
|
|
|
|
// T - self.tail
|
|
|
|
// o - Valid element
|
|
|
|
// x - Element marked for removal
|
|
|
|
// R - Indicates element that is being removed
|
|
|
|
// M - Indicates element was moved
|
|
|
|
|
2015-07-17 21:28:25 +02:00
|
|
|
let idx = self.wrap_add(self.tail, index);
|
2014-12-17 00:37:55 +01:00
|
|
|
|
2015-11-24 11:23:48 +13:00
|
|
|
let elem = unsafe { Some(self.buffer_read(idx)) };
|
2014-12-17 00:37:55 +01:00
|
|
|
|
2015-07-17 21:28:25 +02:00
|
|
|
let distance_to_tail = index;
|
|
|
|
let distance_to_head = self.len() - index;
|
2014-12-17 00:37:55 +01:00
|
|
|
|
2015-01-05 15:48:58 +01:00
|
|
|
let contiguous = self.is_contiguous();
|
2014-12-17 00:37:55 +01:00
|
|
|
|
2015-11-24 11:23:48 +13:00
|
|
|
match (contiguous,
|
|
|
|
distance_to_tail <= distance_to_head,
|
|
|
|
idx >= self.tail) {
|
|
|
|
(true, true, _) => {
|
|
|
|
unsafe {
|
|
|
|
// contiguous, remove closer to tail:
|
|
|
|
//
|
|
|
|
// T R H
|
|
|
|
// [. . . o o x o o o o . . . . . .]
|
|
|
|
//
|
|
|
|
// T H
|
|
|
|
// [. . . . o o o o o o . . . . . .]
|
|
|
|
// M M
|
|
|
|
|
|
|
|
self.copy(self.tail + 1, self.tail, index);
|
|
|
|
self.tail += 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
(true, false, _) => {
|
|
|
|
unsafe {
|
|
|
|
// contiguous, remove closer to head:
|
|
|
|
//
|
|
|
|
// T R H
|
|
|
|
// [. . . o o o o x o o . . . . . .]
|
|
|
|
//
|
|
|
|
// T H
|
|
|
|
// [. . . o o o o o o . . . . . . .]
|
|
|
|
// M M
|
|
|
|
|
|
|
|
self.copy(idx, idx + 1, self.head - idx - 1);
|
|
|
|
self.head -= 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
(false, true, true) => {
|
|
|
|
unsafe {
|
|
|
|
// discontiguous, remove closer to tail, tail section:
|
|
|
|
//
|
|
|
|
// H T R
|
|
|
|
// [o o o o o o . . . . . o o x o o]
|
|
|
|
//
|
|
|
|
// H T
|
|
|
|
// [o o o o o o . . . . . . o o o o]
|
|
|
|
// M M
|
|
|
|
|
|
|
|
self.copy(self.tail + 1, self.tail, index);
|
|
|
|
self.tail = self.wrap_add(self.tail, 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
(false, false, false) => {
|
|
|
|
unsafe {
|
|
|
|
// discontiguous, remove closer to head, head section:
|
|
|
|
//
|
|
|
|
// R H T
|
|
|
|
// [o o o o x o o . . . . . . o o o]
|
|
|
|
//
|
|
|
|
// H T
|
|
|
|
// [o o o o o o . . . . . . . o o o]
|
|
|
|
// M M
|
|
|
|
|
|
|
|
self.copy(idx, idx + 1, self.head - idx - 1);
|
|
|
|
self.head -= 1;
|
2014-12-17 00:37:55 +01:00
|
|
|
}
|
2015-11-24 11:23:48 +13:00
|
|
|
}
|
|
|
|
(false, false, true) => {
|
|
|
|
unsafe {
|
|
|
|
// discontiguous, remove closer to head, tail section:
|
|
|
|
//
|
|
|
|
// H T R
|
|
|
|
// [o o o . . . . . . o o o o o x o]
|
|
|
|
//
|
|
|
|
// H T
|
|
|
|
// [o o . . . . . . . o o o o o o o]
|
|
|
|
// M M M M
|
|
|
|
//
|
|
|
|
// or quasi-discontiguous, remove next to head, tail section:
|
|
|
|
//
|
|
|
|
// H T R
|
|
|
|
// [. . . . . . . . . o o o o o x o]
|
|
|
|
//
|
|
|
|
// T H
|
|
|
|
// [. . . . . . . . . o o o o o o .]
|
|
|
|
// M
|
|
|
|
|
|
|
|
// draw in elements in the tail section
|
|
|
|
self.copy(idx, idx + 1, self.cap() - idx - 1);
|
|
|
|
|
|
|
|
// Prevents underflow.
|
|
|
|
if self.head != 0 {
|
|
|
|
// copy first element into empty spot
|
|
|
|
self.copy(self.cap() - 1, 0, 1);
|
|
|
|
|
|
|
|
// move elements in the head section backwards
|
|
|
|
self.copy(0, 1, self.head - 1);
|
|
|
|
}
|
2014-12-17 00:37:55 +01:00
|
|
|
|
2015-11-24 11:23:48 +13:00
|
|
|
self.head = self.wrap_sub(self.head, 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
(false, true, false) => {
|
|
|
|
unsafe {
|
|
|
|
// discontiguous, remove closer to tail, head section:
|
|
|
|
//
|
|
|
|
// R H T
|
|
|
|
// [o o x o o o o o o o . . . o o o]
|
|
|
|
//
|
|
|
|
// H T
|
|
|
|
// [o o o o o o o o o o . . . . o o]
|
|
|
|
// M M M M M
|
2014-12-17 00:37:55 +01:00
|
|
|
|
2015-11-24 11:23:48 +13:00
|
|
|
// draw in elements up to idx
|
|
|
|
self.copy(1, 0, idx);
|
2014-12-17 00:37:55 +01:00
|
|
|
|
2015-11-24 11:23:48 +13:00
|
|
|
// copy last element into empty spot
|
|
|
|
self.copy(0, self.cap() - 1, 1);
|
2014-12-17 00:37:55 +01:00
|
|
|
|
2015-11-24 11:23:48 +13:00
|
|
|
// move elements from tail to end forward, excluding the last one
|
|
|
|
self.copy(self.tail + 1, self.tail, self.cap() - self.tail - 1);
|
2014-12-17 00:37:55 +01:00
|
|
|
|
2015-11-24 11:23:48 +13:00
|
|
|
self.tail = self.wrap_add(self.tail, 1);
|
|
|
|
}
|
2014-12-17 00:37:55 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return elem;
|
|
|
|
}
|
2015-02-07 11:46:16 -05:00
|
|
|
|
|
|
|
/// Splits the collection into two at the given index.
|
|
|
|
///
|
|
|
|
/// Returns a newly allocated `Self`. `self` contains elements `[0, at)`,
|
|
|
|
/// and the returned `Self` contains elements `[at, len)`.
|
|
|
|
///
|
|
|
|
/// Note that the capacity of `self` does not change.
|
|
|
|
///
|
|
|
|
/// # Panics
|
|
|
|
///
|
|
|
|
/// Panics if `at > len`
|
|
|
|
///
|
|
|
|
/// # Examples
|
2015-02-07 12:13:32 -05:00
|
|
|
///
|
|
|
|
/// ```
|
2015-02-17 23:44:55 -08:00
|
|
|
/// use std::collections::VecDeque;
|
2015-02-07 12:13:32 -05:00
|
|
|
///
|
2015-02-17 23:44:55 -08:00
|
|
|
/// let mut buf: VecDeque<_> = vec![1,2,3].into_iter().collect();
|
2015-02-07 11:46:16 -05:00
|
|
|
/// let buf2 = buf.split_off(1);
|
|
|
|
/// // buf = [1], buf2 = [2, 3]
|
|
|
|
/// assert_eq!(buf.len(), 1);
|
|
|
|
/// assert_eq!(buf2.len(), 2);
|
|
|
|
/// ```
|
|
|
|
#[inline]
|
2015-09-10 13:26:44 -07:00
|
|
|
#[stable(feature = "split_off", since = "1.4.0")]
|
2015-02-07 11:46:16 -05:00
|
|
|
pub fn split_off(&mut self, at: usize) -> Self {
|
|
|
|
let len = self.len();
|
|
|
|
assert!(at <= len, "`at` out of bounds");
|
|
|
|
|
|
|
|
let other_len = len - at;
|
2015-02-17 23:44:55 -08:00
|
|
|
let mut other = VecDeque::with_capacity(other_len);
|
2015-02-07 11:46:16 -05:00
|
|
|
|
|
|
|
unsafe {
|
|
|
|
let (first_half, second_half) = self.as_slices();
|
|
|
|
|
|
|
|
let first_len = first_half.len();
|
|
|
|
let second_len = second_half.len();
|
|
|
|
if at < first_len {
|
|
|
|
// `at` lies in the first half.
|
|
|
|
let amount_in_first = first_len - at;
|
|
|
|
|
2015-03-27 11:12:28 -07:00
|
|
|
ptr::copy_nonoverlapping(first_half.as_ptr().offset(at as isize),
|
2015-07-09 21:57:21 -07:00
|
|
|
other.ptr(),
|
2015-02-23 11:39:16 -08:00
|
|
|
amount_in_first);
|
2015-02-07 11:46:16 -05:00
|
|
|
|
|
|
|
// just take all of the second half.
|
2015-03-27 11:12:28 -07:00
|
|
|
ptr::copy_nonoverlapping(second_half.as_ptr(),
|
2015-07-09 21:57:21 -07:00
|
|
|
other.ptr().offset(amount_in_first as isize),
|
2015-02-23 11:39:16 -08:00
|
|
|
second_len);
|
2015-02-07 11:46:16 -05:00
|
|
|
} else {
|
|
|
|
// `at` lies in the second half, need to factor in the elements we skipped
|
|
|
|
// in the first half.
|
|
|
|
let offset = at - first_len;
|
|
|
|
let amount_in_second = second_len - offset;
|
2015-03-27 11:12:28 -07:00
|
|
|
ptr::copy_nonoverlapping(second_half.as_ptr().offset(offset as isize),
|
2015-07-09 21:57:21 -07:00
|
|
|
other.ptr(),
|
2015-02-23 11:39:16 -08:00
|
|
|
amount_in_second);
|
2015-02-07 11:46:16 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Cleanup where the ends of the buffers are
|
2015-02-19 08:33:32 +01:00
|
|
|
self.head = self.wrap_sub(self.head, other_len);
|
2015-02-07 11:46:16 -05:00
|
|
|
other.head = other.wrap_index(other_len);
|
|
|
|
|
|
|
|
other
|
|
|
|
}
|
2015-02-07 12:13:32 -05:00
|
|
|
|
|
|
|
/// Moves all the elements of `other` into `Self`, leaving `other` empty.
|
|
|
|
///
|
|
|
|
/// # Panics
|
|
|
|
///
|
|
|
|
/// Panics if the new number of elements in self overflows a `usize`.
|
|
|
|
///
|
|
|
|
/// # Examples
|
|
|
|
///
|
|
|
|
/// ```
|
2015-02-17 23:44:55 -08:00
|
|
|
/// use std::collections::VecDeque;
|
2015-02-07 12:13:32 -05:00
|
|
|
///
|
2015-02-17 23:44:55 -08:00
|
|
|
/// let mut buf: VecDeque<_> = vec![1, 2, 3].into_iter().collect();
|
|
|
|
/// let mut buf2: VecDeque<_> = vec![4, 5, 6].into_iter().collect();
|
2015-02-07 12:13:32 -05:00
|
|
|
/// buf.append(&mut buf2);
|
|
|
|
/// assert_eq!(buf.len(), 6);
|
|
|
|
/// assert_eq!(buf2.len(), 0);
|
|
|
|
/// ```
|
|
|
|
#[inline]
|
2015-09-10 13:26:44 -07:00
|
|
|
#[stable(feature = "append", since = "1.4.0")]
|
2015-02-07 12:13:32 -05:00
|
|
|
pub fn append(&mut self, other: &mut Self) {
|
|
|
|
// naive impl
|
2015-08-06 19:03:14 -07:00
|
|
|
self.extend(other.drain(..));
|
2015-02-07 12:13:32 -05:00
|
|
|
}
|
2015-04-27 13:47:19 -04:00
|
|
|
|
|
|
|
/// Retains only the elements specified by the predicate.
|
|
|
|
///
|
|
|
|
/// In other words, remove all elements `e` such that `f(&e)` returns false.
|
|
|
|
/// This method operates in place and preserves the order of the retained
|
|
|
|
/// elements.
|
|
|
|
///
|
|
|
|
/// # Examples
|
|
|
|
///
|
|
|
|
/// ```
|
|
|
|
/// use std::collections::VecDeque;
|
|
|
|
///
|
|
|
|
/// let mut buf = VecDeque::new();
|
|
|
|
/// buf.extend(1..5);
|
|
|
|
/// buf.retain(|&x| x%2 == 0);
|
|
|
|
///
|
|
|
|
/// let v: Vec<_> = buf.into_iter().collect();
|
|
|
|
/// assert_eq!(&v[..], &[2, 4]);
|
|
|
|
/// ```
|
2015-09-10 13:26:44 -07:00
|
|
|
#[stable(feature = "vec_deque_retain", since = "1.4.0")]
|
2015-11-24 11:23:48 +13:00
|
|
|
pub fn retain<F>(&mut self, mut f: F)
|
|
|
|
where F: FnMut(&T) -> bool
|
|
|
|
{
|
2015-04-27 13:47:19 -04:00
|
|
|
let len = self.len();
|
|
|
|
let mut del = 0;
|
|
|
|
for i in 0..len {
|
|
|
|
if !f(&self[i]) {
|
|
|
|
del += 1;
|
|
|
|
} else if del > 0 {
|
2015-11-24 11:23:48 +13:00
|
|
|
self.swap(i - del, i);
|
2015-04-27 13:47:19 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if del > 0 {
|
|
|
|
self.truncate(len - del);
|
|
|
|
}
|
|
|
|
}
|
2013-06-25 15:08:47 -04:00
|
|
|
}
|
|
|
|
|
2015-02-17 23:44:55 -08:00
|
|
|
impl<T: Clone> VecDeque<T> {
|
2015-07-05 03:49:36 -04:00
|
|
|
/// Modifies the `VecDeque` in-place so that `len()` is equal to new_len,
|
2015-01-05 15:48:58 +01:00
|
|
|
/// either by removing excess elements or by appending copies of a value to the back.
|
|
|
|
///
|
|
|
|
/// # Examples
|
|
|
|
///
|
|
|
|
/// ```
|
2015-07-27 10:50:19 -04:00
|
|
|
/// #![feature(deque_extras)]
|
|
|
|
///
|
2015-02-17 23:44:55 -08:00
|
|
|
/// use std::collections::VecDeque;
|
2015-01-05 15:48:58 +01:00
|
|
|
///
|
2015-02-17 23:44:55 -08:00
|
|
|
/// let mut buf = VecDeque::new();
|
2015-01-25 22:05:03 +01:00
|
|
|
/// buf.push_back(5);
|
|
|
|
/// buf.push_back(10);
|
2015-01-05 15:48:58 +01:00
|
|
|
/// buf.push_back(15);
|
|
|
|
/// buf.resize(2, 0);
|
|
|
|
/// buf.resize(6, 20);
|
2015-06-10 17:22:20 +01:00
|
|
|
/// for (a, b) in [5, 10, 20, 20, 20, 20].iter().zip(&buf) {
|
2015-01-05 15:48:58 +01:00
|
|
|
/// assert_eq!(a, b);
|
|
|
|
/// }
|
|
|
|
/// ```
|
2015-06-09 14:39:23 -07:00
|
|
|
#[unstable(feature = "deque_extras",
|
2015-08-12 22:19:51 -07:00
|
|
|
reason = "matches collection reform specification; waiting on panic semantics",
|
|
|
|
issue = "27788")]
|
2015-02-04 21:17:19 -05:00
|
|
|
pub fn resize(&mut self, new_len: usize, value: T) {
|
2015-01-05 15:48:58 +01:00
|
|
|
let len = self.len();
|
|
|
|
|
|
|
|
if new_len > len {
|
|
|
|
self.extend(repeat(value).take(new_len - len))
|
|
|
|
} else {
|
|
|
|
self.truncate(new_len);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-10-19 16:19:07 -04:00
|
|
|
/// Returns the index in the underlying buffer for a given logical element index.
|
|
|
|
#[inline]
|
2015-02-04 21:17:19 -05:00
|
|
|
fn wrap_index(index: usize, size: usize) -> usize {
|
2014-10-19 16:19:07 -04:00
|
|
|
// size is always a power of 2
|
2015-09-18 16:32:52 +02:00
|
|
|
debug_assert!(size.is_power_of_two());
|
2014-11-11 20:22:07 -05:00
|
|
|
index & (size - 1)
|
2014-10-19 16:19:07 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Calculate the number of elements left to be read in the buffer
|
|
|
|
#[inline]
|
2015-02-04 21:17:19 -05:00
|
|
|
fn count(tail: usize, head: usize, size: usize) -> usize {
|
2014-10-19 16:19:07 -04:00
|
|
|
// size is always a power of 2
|
2015-02-19 08:33:32 +01:00
|
|
|
(head.wrapping_sub(tail)) & (size - 1)
|
2014-10-19 16:19:07 -04:00
|
|
|
}
|
|
|
|
|
2015-02-17 23:44:55 -08:00
|
|
|
/// `VecDeque` iterator.
|
2015-01-23 21:48:20 -08:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2015-11-24 11:23:48 +13:00
|
|
|
pub struct Iter<'a, T: 'a> {
|
2014-10-19 16:19:07 -04:00
|
|
|
ring: &'a [T],
|
2015-02-04 21:17:19 -05:00
|
|
|
tail: usize,
|
2015-11-24 11:23:48 +13:00
|
|
|
head: usize,
|
2014-08-27 21:46:52 -04:00
|
|
|
}
|
|
|
|
|
2015-01-03 22:54:18 -05:00
|
|
|
// FIXME(#19839) Remove in favor of `#[derive(Clone)]`
|
2015-11-16 19:54:28 +03:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2014-12-30 21:01:36 +11:00
|
|
|
impl<'a, T> Clone for Iter<'a, T> {
|
|
|
|
fn clone(&self) -> Iter<'a, T> {
|
|
|
|
Iter {
|
|
|
|
ring: self.ring,
|
|
|
|
tail: self.tail,
|
2015-11-24 11:23:48 +13:00
|
|
|
head: self.head,
|
2014-12-30 21:01:36 +11:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-23 21:48:20 -08:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2015-01-01 23:15:35 -05:00
|
|
|
impl<'a, T> Iterator for Iter<'a, T> {
|
|
|
|
type Item = &'a T;
|
|
|
|
|
2013-11-16 17:29:39 -05:00
|
|
|
#[inline]
|
2013-12-09 23:16:18 -08:00
|
|
|
fn next(&mut self) -> Option<&'a T> {
|
2014-10-19 16:19:07 -04:00
|
|
|
if self.tail == self.head {
|
2013-11-16 17:29:39 -05:00
|
|
|
return None;
|
2013-06-26 11:38:29 -04:00
|
|
|
}
|
2014-10-19 16:19:07 -04:00
|
|
|
let tail = self.tail;
|
2015-02-19 08:33:32 +01:00
|
|
|
self.tail = wrap_index(self.tail.wrapping_add(1), self.ring.len());
|
2014-12-30 10:51:18 -08:00
|
|
|
unsafe { Some(self.ring.get_unchecked(tail)) }
|
2013-06-26 11:38:29 -04:00
|
|
|
}
|
|
|
|
|
2013-11-16 17:29:39 -05:00
|
|
|
#[inline]
|
2015-02-04 21:17:19 -05:00
|
|
|
fn size_hint(&self) -> (usize, Option<usize>) {
|
2014-10-19 16:19:07 -04:00
|
|
|
let len = count(self.tail, self.head, self.ring.len());
|
2013-11-16 17:29:39 -05:00
|
|
|
(len, Some(len))
|
2013-07-16 01:13:26 +02:00
|
|
|
}
|
2013-06-25 15:08:47 -04:00
|
|
|
}
|
|
|
|
|
2015-01-23 21:48:20 -08:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2015-01-01 23:15:35 -05:00
|
|
|
impl<'a, T> DoubleEndedIterator for Iter<'a, T> {
|
2013-11-16 17:29:39 -05:00
|
|
|
#[inline]
|
2013-12-09 23:16:18 -08:00
|
|
|
fn next_back(&mut self) -> Option<&'a T> {
|
2014-10-19 16:19:07 -04:00
|
|
|
if self.tail == self.head {
|
2013-11-16 17:29:39 -05:00
|
|
|
return None;
|
|
|
|
}
|
2015-02-19 08:33:32 +01:00
|
|
|
self.head = wrap_index(self.head.wrapping_sub(1), self.ring.len());
|
2014-12-30 10:51:18 -08:00
|
|
|
unsafe { Some(self.ring.get_unchecked(self.head)) }
|
2013-11-16 17:29:39 -05:00
|
|
|
}
|
2013-06-25 15:08:47 -04:00
|
|
|
}
|
2013-06-26 11:38:29 -04:00
|
|
|
|
2015-01-23 21:48:20 -08:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2015-01-01 23:15:35 -05:00
|
|
|
impl<'a, T> ExactSizeIterator for Iter<'a, T> {}
|
2013-09-01 18:20:24 +02:00
|
|
|
|
2015-02-17 23:44:55 -08:00
|
|
|
/// `VecDeque` mutable iterator.
|
2015-01-23 21:48:20 -08:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2015-11-24 11:23:48 +13:00
|
|
|
pub struct IterMut<'a, T: 'a> {
|
2015-02-25 18:11:23 +08:00
|
|
|
ring: &'a mut [T],
|
2015-02-04 21:17:19 -05:00
|
|
|
tail: usize,
|
|
|
|
head: usize,
|
2014-08-27 21:46:52 -04:00
|
|
|
}
|
|
|
|
|
2015-01-23 21:48:20 -08:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2015-01-01 23:15:35 -05:00
|
|
|
impl<'a, T> Iterator for IterMut<'a, T> {
|
|
|
|
type Item = &'a mut T;
|
|
|
|
|
2013-11-16 17:29:39 -05:00
|
|
|
#[inline]
|
2013-12-09 23:16:18 -08:00
|
|
|
fn next(&mut self) -> Option<&'a mut T> {
|
2014-10-19 16:19:07 -04:00
|
|
|
if self.tail == self.head {
|
2013-11-16 17:29:39 -05:00
|
|
|
return None;
|
|
|
|
}
|
2014-10-19 16:19:07 -04:00
|
|
|
let tail = self.tail;
|
2015-02-19 08:33:32 +01:00
|
|
|
self.tail = wrap_index(self.tail.wrapping_add(1), self.ring.len());
|
2014-11-22 21:34:11 -05:00
|
|
|
|
|
|
|
unsafe {
|
2015-02-25 18:11:23 +08:00
|
|
|
let elem = self.ring.get_unchecked_mut(tail);
|
|
|
|
Some(&mut *(elem as *mut _))
|
2014-10-14 23:05:01 -07:00
|
|
|
}
|
2013-11-16 17:29:39 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
2015-02-04 21:17:19 -05:00
|
|
|
fn size_hint(&self) -> (usize, Option<usize>) {
|
2015-02-25 18:11:23 +08:00
|
|
|
let len = count(self.tail, self.head, self.ring.len());
|
2014-10-19 16:19:07 -04:00
|
|
|
(len, Some(len))
|
2013-11-16 17:29:39 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-23 21:48:20 -08:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2015-01-01 23:15:35 -05:00
|
|
|
impl<'a, T> DoubleEndedIterator for IterMut<'a, T> {
|
2013-11-16 17:29:39 -05:00
|
|
|
#[inline]
|
2013-12-09 23:16:18 -08:00
|
|
|
fn next_back(&mut self) -> Option<&'a mut T> {
|
2014-10-19 16:19:07 -04:00
|
|
|
if self.tail == self.head {
|
2013-11-16 17:29:39 -05:00
|
|
|
return None;
|
|
|
|
}
|
2015-02-19 08:33:32 +01:00
|
|
|
self.head = wrap_index(self.head.wrapping_sub(1), self.ring.len());
|
2014-11-22 21:34:11 -05:00
|
|
|
|
|
|
|
unsafe {
|
2015-02-25 18:11:23 +08:00
|
|
|
let elem = self.ring.get_unchecked_mut(self.head);
|
|
|
|
Some(&mut *(elem as *mut _))
|
2014-11-22 21:34:11 -05:00
|
|
|
}
|
2013-11-16 17:29:39 -05:00
|
|
|
}
|
2013-06-25 15:08:47 -04:00
|
|
|
}
|
2012-01-11 12:49:33 +01:00
|
|
|
|
2015-01-23 21:48:20 -08:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2015-01-01 23:15:35 -05:00
|
|
|
impl<'a, T> ExactSizeIterator for IterMut<'a, T> {}
|
2013-09-01 18:20:24 +02:00
|
|
|
|
2015-02-17 23:44:55 -08:00
|
|
|
/// A by-value VecDeque iterator
|
2015-03-23 08:50:47 -04:00
|
|
|
#[derive(Clone)]
|
2015-01-23 21:48:20 -08:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2014-12-19 21:52:10 +01:00
|
|
|
pub struct IntoIter<T> {
|
2015-02-17 23:44:55 -08:00
|
|
|
inner: VecDeque<T>,
|
2014-11-22 21:34:11 -05:00
|
|
|
}
|
|
|
|
|
2015-01-23 21:48:20 -08:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2015-01-01 23:15:35 -05:00
|
|
|
impl<T> Iterator for IntoIter<T> {
|
|
|
|
type Item = T;
|
|
|
|
|
2014-11-22 21:34:11 -05:00
|
|
|
#[inline]
|
|
|
|
fn next(&mut self) -> Option<T> {
|
|
|
|
self.inner.pop_front()
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
2015-02-04 21:17:19 -05:00
|
|
|
fn size_hint(&self) -> (usize, Option<usize>) {
|
2014-11-22 21:34:11 -05:00
|
|
|
let len = self.inner.len();
|
|
|
|
(len, Some(len))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-23 21:48:20 -08:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2015-01-01 23:15:35 -05:00
|
|
|
impl<T> DoubleEndedIterator for IntoIter<T> {
|
2014-11-22 21:34:11 -05:00
|
|
|
#[inline]
|
|
|
|
fn next_back(&mut self) -> Option<T> {
|
|
|
|
self.inner.pop_back()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-23 21:48:20 -08:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2015-01-01 23:15:35 -05:00
|
|
|
impl<T> ExactSizeIterator for IntoIter<T> {}
|
2014-11-22 21:34:11 -05:00
|
|
|
|
2015-02-17 23:44:55 -08:00
|
|
|
/// A draining VecDeque iterator
|
2015-12-02 17:31:49 -08:00
|
|
|
#[stable(feature = "drain", since = "1.6.0")]
|
2014-12-16 17:45:03 -05:00
|
|
|
pub struct Drain<'a, T: 'a> {
|
2015-08-06 19:03:14 -07:00
|
|
|
after_tail: usize,
|
|
|
|
after_head: usize,
|
|
|
|
iter: Iter<'a, T>,
|
|
|
|
deque: *mut VecDeque<T>,
|
2014-12-16 17:45:03 -05:00
|
|
|
}
|
|
|
|
|
2015-12-02 17:31:49 -08:00
|
|
|
#[stable(feature = "drain", since = "1.6.0")]
|
2015-09-24 18:19:23 -04:00
|
|
|
unsafe impl<'a, T: Sync> Sync for Drain<'a, T> {}
|
2015-12-02 17:31:49 -08:00
|
|
|
#[stable(feature = "drain", since = "1.6.0")]
|
2015-09-24 18:19:23 -04:00
|
|
|
unsafe impl<'a, T: Send> Send for Drain<'a, T> {}
|
|
|
|
|
2015-01-23 21:48:20 -08:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2014-12-16 17:45:03 -05:00
|
|
|
impl<'a, T: 'a> Drop for Drain<'a, T> {
|
|
|
|
fn drop(&mut self) {
|
2015-01-10 21:50:07 -05:00
|
|
|
for _ in self.by_ref() {}
|
2015-08-06 19:03:14 -07:00
|
|
|
|
|
|
|
let source_deque = unsafe { &mut *self.deque };
|
|
|
|
|
|
|
|
// T = source_deque_tail; H = source_deque_head; t = drain_tail; h = drain_head
|
|
|
|
//
|
|
|
|
// T t h H
|
|
|
|
// [. . . o o x x o o . . .]
|
|
|
|
//
|
|
|
|
let orig_tail = source_deque.tail;
|
|
|
|
let drain_tail = source_deque.head;
|
|
|
|
let drain_head = self.after_tail;
|
|
|
|
let orig_head = self.after_head;
|
|
|
|
|
|
|
|
let tail_len = count(orig_tail, drain_tail, source_deque.cap());
|
|
|
|
let head_len = count(drain_head, orig_head, source_deque.cap());
|
|
|
|
|
|
|
|
// Restore the original head value
|
|
|
|
source_deque.head = orig_head;
|
|
|
|
|
|
|
|
match (tail_len, head_len) {
|
|
|
|
(0, 0) => {
|
|
|
|
source_deque.head = 0;
|
|
|
|
source_deque.tail = 0;
|
|
|
|
}
|
|
|
|
(0, _) => {
|
|
|
|
source_deque.tail = drain_head;
|
|
|
|
}
|
|
|
|
(_, 0) => {
|
|
|
|
source_deque.head = drain_tail;
|
|
|
|
}
|
2015-11-24 11:23:48 +13:00
|
|
|
_ => {
|
|
|
|
unsafe {
|
|
|
|
if tail_len <= head_len {
|
|
|
|
source_deque.tail = source_deque.wrap_sub(drain_head, tail_len);
|
|
|
|
source_deque.wrap_copy(source_deque.tail, orig_tail, tail_len);
|
|
|
|
} else {
|
|
|
|
source_deque.head = source_deque.wrap_add(drain_tail, head_len);
|
|
|
|
source_deque.wrap_copy(drain_tail, drain_head, head_len);
|
|
|
|
}
|
2015-08-06 19:03:14 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2014-12-16 17:45:03 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-23 21:48:20 -08:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2015-01-01 23:15:35 -05:00
|
|
|
impl<'a, T: 'a> Iterator for Drain<'a, T> {
|
|
|
|
type Item = T;
|
|
|
|
|
2014-12-16 17:45:03 -05:00
|
|
|
#[inline]
|
|
|
|
fn next(&mut self) -> Option<T> {
|
2015-11-24 11:23:48 +13:00
|
|
|
self.iter.next().map(|elt| unsafe { ptr::read(elt) })
|
2014-12-16 17:45:03 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
2015-02-04 21:17:19 -05:00
|
|
|
fn size_hint(&self) -> (usize, Option<usize>) {
|
2015-08-06 19:03:14 -07:00
|
|
|
self.iter.size_hint()
|
2014-12-16 17:45:03 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-23 21:48:20 -08:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2015-01-01 23:15:35 -05:00
|
|
|
impl<'a, T: 'a> DoubleEndedIterator for Drain<'a, T> {
|
2014-12-16 17:45:03 -05:00
|
|
|
#[inline]
|
|
|
|
fn next_back(&mut self) -> Option<T> {
|
2015-11-24 11:23:48 +13:00
|
|
|
self.iter.next_back().map(|elt| unsafe { ptr::read(elt) })
|
2014-12-16 17:45:03 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-23 21:48:20 -08:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2015-01-01 23:15:35 -05:00
|
|
|
impl<'a, T: 'a> ExactSizeIterator for Drain<'a, T> {}
|
2014-12-16 17:45:03 -05:00
|
|
|
|
2015-01-23 21:48:20 -08:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2015-02-17 23:44:55 -08:00
|
|
|
impl<A: PartialEq> PartialEq for VecDeque<A> {
|
|
|
|
fn eq(&self, other: &VecDeque<A>) -> bool {
|
2016-01-27 00:35:03 +01:00
|
|
|
if self.len() != other.len() {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
let (sa, sb) = self.as_slices();
|
|
|
|
let (oa, ob) = other.as_slices();
|
|
|
|
if sa.len() == oa.len() {
|
|
|
|
sa == oa && sb == ob
|
|
|
|
} else if sa.len() < oa.len() {
|
|
|
|
// Always divisible in three sections, for example:
|
|
|
|
// self: [a b c|d e f]
|
|
|
|
// other: [0 1 2 3|4 5]
|
|
|
|
// front = 3, mid = 1,
|
|
|
|
// [a b c] == [0 1 2] && [d] == [3] && [e f] == [4 5]
|
|
|
|
let front = sa.len();
|
|
|
|
let mid = oa.len() - front;
|
|
|
|
|
|
|
|
let (oa_front, oa_mid) = oa.split_at(front);
|
|
|
|
let (sb_mid, sb_back) = sb.split_at(mid);
|
|
|
|
debug_assert_eq!(sa.len(), oa_front.len());
|
|
|
|
debug_assert_eq!(sb_mid.len(), oa_mid.len());
|
|
|
|
debug_assert_eq!(sb_back.len(), ob.len());
|
|
|
|
sa == oa_front && sb_mid == oa_mid && sb_back == ob
|
|
|
|
} else {
|
|
|
|
let front = oa.len();
|
|
|
|
let mid = sa.len() - front;
|
|
|
|
|
|
|
|
let (sa_front, sa_mid) = sa.split_at(front);
|
|
|
|
let (ob_mid, ob_back) = ob.split_at(mid);
|
|
|
|
debug_assert_eq!(sa_front.len(), oa.len());
|
|
|
|
debug_assert_eq!(sa_mid.len(), ob_mid.len());
|
|
|
|
debug_assert_eq!(sb.len(), ob_back.len());
|
|
|
|
sa_front == oa && sa_mid == ob_mid && sb == ob_back
|
|
|
|
}
|
2013-07-06 15:27:32 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-23 21:48:20 -08:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2015-02-17 23:44:55 -08:00
|
|
|
impl<A: Eq> Eq for VecDeque<A> {}
|
2014-08-01 16:05:03 -04:00
|
|
|
|
2015-01-23 21:48:20 -08:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2015-02-17 23:44:55 -08:00
|
|
|
impl<A: PartialOrd> PartialOrd for VecDeque<A> {
|
|
|
|
fn partial_cmp(&self, other: &VecDeque<A>) -> Option<Ordering> {
|
2015-08-23 21:59:07 -07:00
|
|
|
self.iter().partial_cmp(other.iter())
|
2014-07-26 23:18:56 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-23 21:48:20 -08:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2015-02-17 23:44:55 -08:00
|
|
|
impl<A: Ord> Ord for VecDeque<A> {
|
2014-08-01 16:22:48 -04:00
|
|
|
#[inline]
|
2015-02-17 23:44:55 -08:00
|
|
|
fn cmp(&self, other: &VecDeque<A>) -> Ordering {
|
2015-08-23 21:59:07 -07:00
|
|
|
self.iter().cmp(other.iter())
|
2014-08-01 16:22:48 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-23 21:48:20 -08:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2015-02-18 14:34:08 -08:00
|
|
|
impl<A: Hash> Hash for VecDeque<A> {
|
2015-02-17 20:48:07 -08:00
|
|
|
fn hash<H: Hasher>(&self, state: &mut H) {
|
|
|
|
self.len().hash(state);
|
2016-01-26 21:31:33 +01:00
|
|
|
let (a, b) = self.as_slices();
|
|
|
|
Hash::hash_slice(a, state);
|
|
|
|
Hash::hash_slice(b, state);
|
2015-02-17 20:48:07 -08:00
|
|
|
}
|
|
|
|
}
|
2014-07-26 22:33:47 -04:00
|
|
|
|
2015-01-23 21:48:20 -08:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2015-02-17 23:44:55 -08:00
|
|
|
impl<A> Index<usize> for VecDeque<A> {
|
2015-01-03 10:40:10 -05:00
|
|
|
type Output = A;
|
|
|
|
|
2015-03-21 19:33:27 -04:00
|
|
|
#[inline]
|
2015-07-17 21:28:25 +02:00
|
|
|
fn index(&self, index: usize) -> &A {
|
|
|
|
self.get(index).expect("Out of bounds access")
|
2015-03-21 19:33:27 -04:00
|
|
|
}
|
2015-01-03 10:40:10 -05:00
|
|
|
}
|
|
|
|
|
2015-01-23 21:48:20 -08:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2015-02-17 23:44:55 -08:00
|
|
|
impl<A> IndexMut<usize> for VecDeque<A> {
|
2015-03-21 19:33:27 -04:00
|
|
|
#[inline]
|
2015-07-17 21:28:25 +02:00
|
|
|
fn index_mut(&mut self, index: usize) -> &mut A {
|
|
|
|
self.get_mut(index).expect("Out of bounds access")
|
2015-03-21 19:33:27 -04:00
|
|
|
}
|
2015-01-03 10:40:10 -05:00
|
|
|
}
|
|
|
|
|
2015-01-23 21:48:20 -08:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2015-02-17 23:44:55 -08:00
|
|
|
impl<A> FromIterator<A> for VecDeque<A> {
|
2015-11-24 11:23:48 +13:00
|
|
|
fn from_iter<T: IntoIterator<Item = A>>(iterable: T) -> VecDeque<A> {
|
2015-02-18 13:06:21 -05:00
|
|
|
let iterator = iterable.into_iter();
|
2013-07-30 02:06:49 +02:00
|
|
|
let (lower, _) = iterator.size_hint();
|
2015-02-17 23:44:55 -08:00
|
|
|
let mut deq = VecDeque::with_capacity(lower);
|
2013-07-30 02:06:49 +02:00
|
|
|
deq.extend(iterator);
|
|
|
|
deq
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-02-17 10:06:24 -08:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2015-02-17 23:44:55 -08:00
|
|
|
impl<T> IntoIterator for VecDeque<T> {
|
2015-02-13 17:55:10 -05:00
|
|
|
type Item = T;
|
|
|
|
type IntoIter = IntoIter<T>;
|
|
|
|
|
2015-04-17 14:31:30 -07:00
|
|
|
/// Consumes the list into a front-to-back iterator yielding elements by
|
|
|
|
/// value.
|
2015-02-13 17:55:10 -05:00
|
|
|
fn into_iter(self) -> IntoIter<T> {
|
2015-11-24 11:23:48 +13:00
|
|
|
IntoIter { inner: self }
|
2015-02-13 17:55:10 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-02-17 10:06:24 -08:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2015-02-17 23:44:55 -08:00
|
|
|
impl<'a, T> IntoIterator for &'a VecDeque<T> {
|
2015-02-13 17:55:10 -05:00
|
|
|
type Item = &'a T;
|
|
|
|
type IntoIter = Iter<'a, T>;
|
|
|
|
|
|
|
|
fn into_iter(self) -> Iter<'a, T> {
|
|
|
|
self.iter()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-02-17 10:06:24 -08:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2015-02-17 23:44:55 -08:00
|
|
|
impl<'a, T> IntoIterator for &'a mut VecDeque<T> {
|
2015-02-13 17:55:10 -05:00
|
|
|
type Item = &'a mut T;
|
|
|
|
type IntoIter = IterMut<'a, T>;
|
|
|
|
|
2015-01-07 22:01:05 -05:00
|
|
|
fn into_iter(mut self) -> IterMut<'a, T> {
|
|
|
|
self.iter_mut()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-23 21:48:20 -08:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2015-02-17 23:44:55 -08:00
|
|
|
impl<A> Extend<A> for VecDeque<A> {
|
2015-11-24 11:23:48 +13:00
|
|
|
fn extend<T: IntoIterator<Item = A>>(&mut self, iter: T) {
|
2015-02-18 10:04:30 -05:00
|
|
|
for elt in iter {
|
2014-11-06 12:24:47 -05:00
|
|
|
self.push_back(elt);
|
2013-07-06 05:42:45 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-06-03 12:38:42 +02:00
|
|
|
#[stable(feature = "extend_ref", since = "1.2.0")]
|
|
|
|
impl<'a, T: 'a + Copy> Extend<&'a T> for VecDeque<T> {
|
2015-11-24 11:23:48 +13:00
|
|
|
fn extend<I: IntoIterator<Item = &'a T>>(&mut self, iter: I) {
|
2015-06-03 12:38:42 +02:00
|
|
|
self.extend(iter.into_iter().cloned());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-23 21:48:20 -08:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2015-02-17 23:44:55 -08:00
|
|
|
impl<T: fmt::Debug> fmt::Debug for VecDeque<T> {
|
2014-06-04 16:15:04 +02:00
|
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
2015-09-25 12:03:03 -04:00
|
|
|
f.debug_list().entries(self).finish()
|
2014-06-04 16:15:04 +02:00
|
|
|
}
|
|
|
|
}
|
2015-03-11 19:44:02 -05:00
|
|
|
|
|
|
|
#[cfg(test)]
|
2015-04-24 17:30:41 +02:00
|
|
|
mod tests {
|
2015-05-30 19:49:56 -04:00
|
|
|
use core::iter::Iterator;
|
2015-03-11 19:44:02 -05:00
|
|
|
use core::option::Option::Some;
|
|
|
|
|
|
|
|
use test;
|
|
|
|
|
|
|
|
use super::VecDeque;
|
|
|
|
|
|
|
|
#[bench]
|
|
|
|
fn bench_push_back_100(b: &mut test::Bencher) {
|
|
|
|
let mut deq = VecDeque::with_capacity(101);
|
|
|
|
b.iter(|| {
|
|
|
|
for i in 0..100 {
|
|
|
|
deq.push_back(i);
|
|
|
|
}
|
|
|
|
deq.head = 0;
|
|
|
|
deq.tail = 0;
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
#[bench]
|
|
|
|
fn bench_push_front_100(b: &mut test::Bencher) {
|
|
|
|
let mut deq = VecDeque::with_capacity(101);
|
|
|
|
b.iter(|| {
|
|
|
|
for i in 0..100 {
|
|
|
|
deq.push_front(i);
|
|
|
|
}
|
|
|
|
deq.head = 0;
|
|
|
|
deq.tail = 0;
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
#[bench]
|
|
|
|
fn bench_pop_back_100(b: &mut test::Bencher) {
|
2015-11-24 11:23:48 +13:00
|
|
|
let mut deq = VecDeque::<i32>::with_capacity(101);
|
2015-03-11 19:44:02 -05:00
|
|
|
|
|
|
|
b.iter(|| {
|
|
|
|
deq.head = 100;
|
|
|
|
deq.tail = 0;
|
|
|
|
while !deq.is_empty() {
|
|
|
|
test::black_box(deq.pop_back());
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
#[bench]
|
|
|
|
fn bench_pop_front_100(b: &mut test::Bencher) {
|
|
|
|
let mut deq = VecDeque::<i32>::with_capacity(101);
|
|
|
|
|
|
|
|
b.iter(|| {
|
|
|
|
deq.head = 100;
|
|
|
|
deq.tail = 0;
|
|
|
|
while !deq.is_empty() {
|
|
|
|
test::black_box(deq.pop_front());
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_swap_front_back_remove() {
|
|
|
|
fn test(back: bool) {
|
|
|
|
// This test checks that every single combination of tail position and length is tested.
|
|
|
|
// Capacity 15 should be large enough to cover every case.
|
|
|
|
let mut tester = VecDeque::with_capacity(15);
|
|
|
|
let usable_cap = tester.capacity();
|
|
|
|
let final_len = usable_cap / 2;
|
|
|
|
|
|
|
|
for len in 0..final_len {
|
|
|
|
let expected = if back {
|
|
|
|
(0..len).collect()
|
|
|
|
} else {
|
|
|
|
(0..len).rev().collect()
|
|
|
|
};
|
|
|
|
for tail_pos in 0..usable_cap {
|
|
|
|
tester.tail = tail_pos;
|
|
|
|
tester.head = tail_pos;
|
|
|
|
if back {
|
|
|
|
for i in 0..len * 2 {
|
|
|
|
tester.push_front(i);
|
|
|
|
}
|
|
|
|
for i in 0..len {
|
2015-12-02 17:07:29 -08:00
|
|
|
assert_eq!(tester.swap_remove_back(i), Some(len * 2 - 1 - i));
|
2015-03-11 19:44:02 -05:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
for i in 0..len * 2 {
|
|
|
|
tester.push_back(i);
|
|
|
|
}
|
|
|
|
for i in 0..len {
|
|
|
|
let idx = tester.len() - 1 - i;
|
2015-12-02 17:07:29 -08:00
|
|
|
assert_eq!(tester.swap_remove_front(idx), Some(len * 2 - 1 - i));
|
2015-03-11 19:44:02 -05:00
|
|
|
}
|
|
|
|
}
|
2015-07-09 21:57:21 -07:00
|
|
|
assert!(tester.tail < tester.cap());
|
|
|
|
assert!(tester.head < tester.cap());
|
2015-03-11 19:44:02 -05:00
|
|
|
assert_eq!(tester, expected);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
test(true);
|
|
|
|
test(false);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_insert() {
|
|
|
|
// This test checks that every single combination of tail position, length, and
|
|
|
|
// insertion position is tested. Capacity 15 should be large enough to cover every case.
|
|
|
|
|
|
|
|
let mut tester = VecDeque::with_capacity(15);
|
|
|
|
// can't guarantee we got 15, so have to get what we got.
|
|
|
|
// 15 would be great, but we will definitely get 2^k - 1, for k >= 4, or else
|
|
|
|
// this test isn't covering what it wants to
|
|
|
|
let cap = tester.capacity();
|
|
|
|
|
|
|
|
|
|
|
|
// len is the length *after* insertion
|
|
|
|
for len in 1..cap {
|
|
|
|
// 0, 1, 2, .., len - 1
|
2015-03-30 11:00:05 -07:00
|
|
|
let expected = (0..).take(len).collect();
|
2015-03-11 19:44:02 -05:00
|
|
|
for tail_pos in 0..cap {
|
|
|
|
for to_insert in 0..len {
|
|
|
|
tester.tail = tail_pos;
|
|
|
|
tester.head = tail_pos;
|
|
|
|
for i in 0..len {
|
|
|
|
if i != to_insert {
|
|
|
|
tester.push_back(i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
tester.insert(to_insert, to_insert);
|
2015-07-09 21:57:21 -07:00
|
|
|
assert!(tester.tail < tester.cap());
|
|
|
|
assert!(tester.head < tester.cap());
|
2015-03-11 19:44:02 -05:00
|
|
|
assert_eq!(tester, expected);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_remove() {
|
|
|
|
// This test checks that every single combination of tail position, length, and
|
|
|
|
// removal position is tested. Capacity 15 should be large enough to cover every case.
|
|
|
|
|
|
|
|
let mut tester = VecDeque::with_capacity(15);
|
|
|
|
// can't guarantee we got 15, so have to get what we got.
|
|
|
|
// 15 would be great, but we will definitely get 2^k - 1, for k >= 4, or else
|
|
|
|
// this test isn't covering what it wants to
|
|
|
|
let cap = tester.capacity();
|
|
|
|
|
|
|
|
// len is the length *after* removal
|
|
|
|
for len in 0..cap - 1 {
|
|
|
|
// 0, 1, 2, .., len - 1
|
2015-03-30 11:00:05 -07:00
|
|
|
let expected = (0..).take(len).collect();
|
2015-03-11 19:44:02 -05:00
|
|
|
for tail_pos in 0..cap {
|
|
|
|
for to_remove in 0..len + 1 {
|
|
|
|
tester.tail = tail_pos;
|
|
|
|
tester.head = tail_pos;
|
|
|
|
for i in 0..len {
|
|
|
|
if i == to_remove {
|
|
|
|
tester.push_back(1234);
|
|
|
|
}
|
|
|
|
tester.push_back(i);
|
|
|
|
}
|
|
|
|
if to_remove == len {
|
|
|
|
tester.push_back(1234);
|
|
|
|
}
|
|
|
|
tester.remove(to_remove);
|
2015-07-09 21:57:21 -07:00
|
|
|
assert!(tester.tail < tester.cap());
|
|
|
|
assert!(tester.head < tester.cap());
|
2015-03-11 19:44:02 -05:00
|
|
|
assert_eq!(tester, expected);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-08-06 19:03:14 -07:00
|
|
|
#[test]
|
|
|
|
fn test_drain() {
|
|
|
|
let mut tester: VecDeque<usize> = VecDeque::with_capacity(7);
|
|
|
|
|
|
|
|
let cap = tester.capacity();
|
|
|
|
for len in 0..cap + 1 {
|
|
|
|
for tail in 0..cap + 1 {
|
|
|
|
for drain_start in 0..len + 1 {
|
|
|
|
for drain_end in drain_start..len + 1 {
|
|
|
|
tester.tail = tail;
|
|
|
|
tester.head = tail;
|
|
|
|
for i in 0..len {
|
|
|
|
tester.push_back(i);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check that we drain the correct values
|
2015-11-24 11:23:48 +13:00
|
|
|
let drained: VecDeque<_> = tester.drain(drain_start..drain_end).collect();
|
|
|
|
let drained_expected: VecDeque<_> = (drain_start..drain_end).collect();
|
2015-08-06 19:03:14 -07:00
|
|
|
assert_eq!(drained, drained_expected);
|
|
|
|
|
|
|
|
// We shouldn't have changed the capacity or made the
|
|
|
|
// head or tail out of bounds
|
|
|
|
assert_eq!(tester.capacity(), cap);
|
|
|
|
assert!(tester.tail < tester.cap());
|
|
|
|
assert!(tester.head < tester.cap());
|
|
|
|
|
|
|
|
// We should see the correct values in the VecDeque
|
2015-11-24 11:23:48 +13:00
|
|
|
let expected: VecDeque<_> = (0..drain_start)
|
|
|
|
.chain(drain_end..len)
|
|
|
|
.collect();
|
2015-08-06 19:03:14 -07:00
|
|
|
assert_eq!(expected, tester);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-03-11 19:44:02 -05:00
|
|
|
#[test]
|
|
|
|
fn test_shrink_to_fit() {
|
|
|
|
// This test checks that every single combination of head and tail position,
|
|
|
|
// is tested. Capacity 15 should be large enough to cover every case.
|
|
|
|
|
|
|
|
let mut tester = VecDeque::with_capacity(15);
|
|
|
|
// can't guarantee we got 15, so have to get what we got.
|
|
|
|
// 15 would be great, but we will definitely get 2^k - 1, for k >= 4, or else
|
|
|
|
// this test isn't covering what it wants to
|
|
|
|
let cap = tester.capacity();
|
|
|
|
tester.reserve(63);
|
|
|
|
let max_cap = tester.capacity();
|
|
|
|
|
|
|
|
for len in 0..cap + 1 {
|
|
|
|
// 0, 1, 2, .., len - 1
|
2015-03-30 11:00:05 -07:00
|
|
|
let expected = (0..).take(len).collect();
|
2015-03-11 19:44:02 -05:00
|
|
|
for tail_pos in 0..max_cap + 1 {
|
|
|
|
tester.tail = tail_pos;
|
|
|
|
tester.head = tail_pos;
|
|
|
|
tester.reserve(63);
|
|
|
|
for i in 0..len {
|
|
|
|
tester.push_back(i);
|
|
|
|
}
|
|
|
|
tester.shrink_to_fit();
|
|
|
|
assert!(tester.capacity() <= cap);
|
2015-07-09 21:57:21 -07:00
|
|
|
assert!(tester.tail < tester.cap());
|
|
|
|
assert!(tester.head < tester.cap());
|
2015-03-11 19:44:02 -05:00
|
|
|
assert_eq!(tester, expected);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_split_off() {
|
|
|
|
// This test checks that every single combination of tail position, length, and
|
|
|
|
// split position is tested. Capacity 15 should be large enough to cover every case.
|
|
|
|
|
|
|
|
let mut tester = VecDeque::with_capacity(15);
|
|
|
|
// can't guarantee we got 15, so have to get what we got.
|
|
|
|
// 15 would be great, but we will definitely get 2^k - 1, for k >= 4, or else
|
|
|
|
// this test isn't covering what it wants to
|
|
|
|
let cap = tester.capacity();
|
|
|
|
|
|
|
|
// len is the length *before* splitting
|
|
|
|
for len in 0..cap {
|
|
|
|
// index to split at
|
|
|
|
for at in 0..len + 1 {
|
|
|
|
// 0, 1, 2, .., at - 1 (may be empty)
|
2015-03-30 11:00:05 -07:00
|
|
|
let expected_self = (0..).take(at).collect();
|
2015-03-11 19:44:02 -05:00
|
|
|
// at, at + 1, .., len - 1 (may be empty)
|
2015-03-30 11:00:05 -07:00
|
|
|
let expected_other = (at..).take(len - at).collect();
|
2015-03-11 19:44:02 -05:00
|
|
|
|
|
|
|
for tail_pos in 0..cap {
|
|
|
|
tester.tail = tail_pos;
|
|
|
|
tester.head = tail_pos;
|
|
|
|
for i in 0..len {
|
|
|
|
tester.push_back(i);
|
|
|
|
}
|
|
|
|
let result = tester.split_off(at);
|
2015-07-09 21:57:21 -07:00
|
|
|
assert!(tester.tail < tester.cap());
|
|
|
|
assert!(tester.head < tester.cap());
|
|
|
|
assert!(result.tail < result.cap());
|
|
|
|
assert!(result.head < result.cap());
|
2015-03-11 19:44:02 -05:00
|
|
|
assert_eq!(tester, expected_self);
|
|
|
|
assert_eq!(result, expected_other);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|