aboutsummaryrefslogtreecommitdiff
path: root/libcollections/enum_set.rs
diff options
context:
space:
mode:
authorpravic <[email protected]>2016-04-12 17:44:24 +0300
committerpravic <[email protected]>2016-04-12 17:44:24 +0300
commitbcb1fb5ba7ecf8b208bd6053e689ad8e87b0654d (patch)
tree8de2327e8f25394e7c30324fddb4b7bcbf9a9f56 /libcollections/enum_set.rs
parentliballoc (diff)
downloadkmd-env-rs-bcb1fb5ba7ecf8b208bd6053e689ad8e87b0654d.tar.xz
kmd-env-rs-bcb1fb5ba7ecf8b208bd6053e689ad8e87b0654d.zip
libcollections
Diffstat (limited to 'libcollections/enum_set.rs')
-rw-r--r--libcollections/enum_set.rs301
1 files changed, 301 insertions, 0 deletions
diff --git a/libcollections/enum_set.rs b/libcollections/enum_set.rs
new file mode 100644
index 0000000..0c66c05
--- /dev/null
+++ b/libcollections/enum_set.rs
@@ -0,0 +1,301 @@
+// Copyright 2012 The Rust Project Developers. See the COPYRIGHT
+// file at the top-level directory of this distribution and at
+// http://rust-lang.org/COPYRIGHT.
+//
+// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
+// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
+// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
+// option. This file may not be copied, modified, or distributed
+// except according to those terms.
+
+//! A structure for holding a set of enum variants.
+//!
+//! This module defines a container which uses an efficient bit mask
+//! representation to hold C-like enum variants.
+
+#![unstable(feature = "enumset",
+ reason = "matches collection reform specification, \
+ waiting for dust to settle",
+ issue = "0")]
+
+use core::marker;
+use core::fmt;
+use core::iter::FromIterator;
+use core::ops::{Sub, BitOr, BitAnd, BitXor};
+
+// FIXME(contentions): implement union family of methods? (general design may be
+// wrong here)
+
+/// A specialized set implementation to use enum types.
+///
+/// It is a logic error for an item to be modified in such a way that the
+/// transformation of the item to or from a `usize`, as determined by the
+/// `CLike` trait, changes while the item is in the set. This is normally only
+/// possible through `Cell`, `RefCell`, global state, I/O, or unsafe code.
+#[derive(PartialEq, Eq, PartialOrd, Ord, Hash)]
+pub struct EnumSet<E> {
+ // We must maintain the invariant that no bits are set
+ // for which no variant exists
+ bits: usize,
+ marker: marker::PhantomData<E>,
+}
+
+impl<E> Copy for EnumSet<E> {}
+
+impl<E> Clone for EnumSet<E> {
+ fn clone(&self) -> EnumSet<E> {
+ *self
+ }
+}
+
+#[stable(feature = "rust1", since = "1.0.0")]
+impl<E: CLike + fmt::Debug> fmt::Debug for EnumSet<E> {
+ fn fmt(&self, fmt: &mut fmt::Formatter) -> fmt::Result {
+ fmt.debug_set().entries(self).finish()
+ }
+}
+
+/// An interface for casting C-like enum to usize and back.
+/// A typically implementation is as below.
+///
+/// ```{rust,ignore}
+/// #[repr(usize)]
+/// enum Foo {
+/// A, B, C
+/// }
+///
+/// impl CLike for Foo {
+/// fn to_usize(&self) -> usize {
+/// *self as usize
+/// }
+///
+/// fn from_usize(v: usize) -> Foo {
+/// unsafe { mem::transmute(v) }
+/// }
+/// }
+/// ```
+pub trait CLike {
+ /// Converts a C-like enum to a `usize`.
+ fn to_usize(&self) -> usize;
+ /// Converts a `usize` to a C-like enum.
+ fn from_usize(usize) -> Self;
+}
+
+fn bit<E: CLike>(e: &E) -> usize {
+ use core::mem;
+ let value = e.to_usize();
+ let bits = mem::size_of::<usize>() * 8;
+ assert!(value < bits,
+ "EnumSet only supports up to {} variants.",
+ bits - 1);
+ 1 << value
+}
+
+impl<E: CLike> EnumSet<E> {
+ /// Returns an empty `EnumSet`.
+ pub fn new() -> EnumSet<E> {
+ EnumSet {
+ bits: 0,
+ marker: marker::PhantomData,
+ }
+ }
+
+ /// Returns the number of elements in the given `EnumSet`.
+ pub fn len(&self) -> usize {
+ self.bits.count_ones() as usize
+ }
+
+ /// Returns true if the `EnumSet` is empty.
+ pub fn is_empty(&self) -> bool {
+ self.bits == 0
+ }
+
+ pub fn clear(&mut self) {
+ self.bits = 0;
+ }
+
+ /// Returns `false` if the `EnumSet` contains any enum of the given `EnumSet`.
+ pub fn is_disjoint(&self, other: &EnumSet<E>) -> bool {
+ (self.bits & other.bits) == 0
+ }
+
+ /// Returns `true` if a given `EnumSet` is included in this `EnumSet`.
+ pub fn is_superset(&self, other: &EnumSet<E>) -> bool {
+ (self.bits & other.bits) == other.bits
+ }
+
+ /// Returns `true` if this `EnumSet` is included in the given `EnumSet`.
+ pub fn is_subset(&self, other: &EnumSet<E>) -> bool {
+ other.is_superset(self)
+ }
+
+ /// Returns the union of both `EnumSets`.
+ pub fn union(&self, e: EnumSet<E>) -> EnumSet<E> {
+ EnumSet {
+ bits: self.bits | e.bits,
+ marker: marker::PhantomData,
+ }
+ }
+
+ /// Returns the intersection of both `EnumSets`.
+ pub fn intersection(&self, e: EnumSet<E>) -> EnumSet<E> {
+ EnumSet {
+ bits: self.bits & e.bits,
+ marker: marker::PhantomData,
+ }
+ }
+
+ /// Adds an enum to the `EnumSet`, and returns `true` if it wasn't there before
+ pub fn insert(&mut self, e: E) -> bool {
+ let result = !self.contains(&e);
+ self.bits |= bit(&e);
+ result
+ }
+
+ /// Removes an enum from the EnumSet
+ pub fn remove(&mut self, e: &E) -> bool {
+ let result = self.contains(e);
+ self.bits &= !bit(e);
+ result
+ }
+
+ /// Returns `true` if an `EnumSet` contains a given enum.
+ pub fn contains(&self, e: &E) -> bool {
+ (self.bits & bit(e)) != 0
+ }
+
+ /// Returns an iterator over an `EnumSet`.
+ pub fn iter(&self) -> Iter<E> {
+ Iter::new(self.bits)
+ }
+}
+
+impl<E: CLike> Sub for EnumSet<E> {
+ type Output = EnumSet<E>;
+
+ fn sub(self, e: EnumSet<E>) -> EnumSet<E> {
+ EnumSet {
+ bits: self.bits & !e.bits,
+ marker: marker::PhantomData,
+ }
+ }
+}
+
+impl<E: CLike> BitOr for EnumSet<E> {
+ type Output = EnumSet<E>;
+
+ fn bitor(self, e: EnumSet<E>) -> EnumSet<E> {
+ EnumSet {
+ bits: self.bits | e.bits,
+ marker: marker::PhantomData,
+ }
+ }
+}
+
+impl<E: CLike> BitAnd for EnumSet<E> {
+ type Output = EnumSet<E>;
+
+ fn bitand(self, e: EnumSet<E>) -> EnumSet<E> {
+ EnumSet {
+ bits: self.bits & e.bits,
+ marker: marker::PhantomData,
+ }
+ }
+}
+
+impl<E: CLike> BitXor for EnumSet<E> {
+ type Output = EnumSet<E>;
+
+ fn bitxor(self, e: EnumSet<E>) -> EnumSet<E> {
+ EnumSet {
+ bits: self.bits ^ e.bits,
+ marker: marker::PhantomData,
+ }
+ }
+}
+
+/// An iterator over an EnumSet
+pub struct Iter<E> {
+ index: usize,
+ bits: usize,
+ marker: marker::PhantomData<E>,
+}
+
+// FIXME(#19839) Remove in favor of `#[derive(Clone)]`
+impl<E> Clone for Iter<E> {
+ fn clone(&self) -> Iter<E> {
+ Iter {
+ index: self.index,
+ bits: self.bits,
+ marker: marker::PhantomData,
+ }
+ }
+}
+
+impl<E: CLike> Iter<E> {
+ fn new(bits: usize) -> Iter<E> {
+ Iter {
+ index: 0,
+ bits: bits,
+ marker: marker::PhantomData,
+ }
+ }
+}
+
+impl<E: CLike> Iterator for Iter<E> {
+ type Item = E;
+
+ fn next(&mut self) -> Option<E> {
+ if self.bits == 0 {
+ return None;
+ }
+
+ while (self.bits & 1) == 0 {
+ self.index += 1;
+ self.bits >>= 1;
+ }
+ let elem = CLike::from_usize(self.index);
+ self.index += 1;
+ self.bits >>= 1;
+ Some(elem)
+ }
+
+ fn size_hint(&self) -> (usize, Option<usize>) {
+ let exact = self.bits.count_ones() as usize;
+ (exact, Some(exact))
+ }
+}
+
+impl<E: CLike> FromIterator<E> for EnumSet<E> {
+ fn from_iter<I: IntoIterator<Item = E>>(iter: I) -> EnumSet<E> {
+ let mut ret = EnumSet::new();
+ ret.extend(iter);
+ ret
+ }
+}
+
+#[stable(feature = "rust1", since = "1.0.0")]
+impl<'a, E> IntoIterator for &'a EnumSet<E> where E: CLike
+{
+ type Item = E;
+ type IntoIter = Iter<E>;
+
+ fn into_iter(self) -> Iter<E> {
+ self.iter()
+ }
+}
+
+impl<E: CLike> Extend<E> for EnumSet<E> {
+ fn extend<I: IntoIterator<Item = E>>(&mut self, iter: I) {
+ for element in iter {
+ self.insert(element);
+ }
+ }
+}
+
+#[stable(feature = "extend_ref", since = "1.2.0")]
+impl<'a, E: 'a + CLike + Copy> Extend<&'a E> for EnumSet<E> {
+ fn extend<I: IntoIterator<Item = &'a E>>(&mut self, iter: I) {
+ self.extend(iter.into_iter().cloned());
+ }
+}