summaryrefslogtreecommitdiffstats
path: root/third_party/rust/enumset/src/repr/primitive.rs
diff options
context:
space:
mode:
Diffstat (limited to 'third_party/rust/enumset/src/repr/primitive.rs')
-rw-r--r--third_party/rust/enumset/src/repr/primitive.rs300
1 files changed, 300 insertions, 0 deletions
diff --git a/third_party/rust/enumset/src/repr/primitive.rs b/third_party/rust/enumset/src/repr/primitive.rs
new file mode 100644
index 0000000000..ef44adaa79
--- /dev/null
+++ b/third_party/rust/enumset/src/repr/primitive.rs
@@ -0,0 +1,300 @@
+use crate::repr::EnumSetTypeRepr;
+
+macro_rules! prim {
+ ($name:ty, $width:expr, $preferred_array_len:expr) => {
+ const _: () = {
+ fn lo(v: $name) -> u64 {
+ v as u64
+ }
+ fn hi(v: $name) -> u64 {
+ ((v as u128) >> 64) as u64
+ }
+
+ impl EnumSetTypeRepr for $name {
+ const PREFERRED_ARRAY_LEN: usize = $preferred_array_len;
+ const WIDTH: u32 = $width;
+ const EMPTY: Self = 0;
+
+ #[inline(always)]
+ fn is_empty(&self) -> bool {
+ *self == 0
+ }
+
+ #[inline(always)]
+ fn add_bit(&mut self, bit: u32) {
+ *self |= 1 << bit as $name;
+ }
+ #[inline(always)]
+ fn remove_bit(&mut self, bit: u32) {
+ *self &= !(1 << bit as $name);
+ }
+ #[inline(always)]
+ fn has_bit(&self, bit: u32) -> bool {
+ (self & (1 << bit as $name)) != 0
+ }
+
+ #[inline(always)]
+ fn count_ones(&self) -> u32 {
+ (*self).count_ones()
+ }
+ #[inline(always)]
+ fn leading_zeros(&self) -> u32 {
+ (*self).leading_zeros()
+ }
+ #[inline(always)]
+ fn trailing_zeros(&self) -> u32 {
+ (*self).trailing_zeros()
+ }
+
+ #[inline(always)]
+ fn and_not(&self, other: Self) -> Self {
+ (*self) & !other
+ }
+
+ type Iter = PrimitiveIter<Self>;
+ fn iter(self) -> Self::Iter {
+ PrimitiveIter(self)
+ }
+
+ #[inline(always)]
+ fn from_u8(v: u8) -> Self {
+ v as $name
+ }
+ #[inline(always)]
+ fn from_u16(v: u16) -> Self {
+ v as $name
+ }
+ #[inline(always)]
+ fn from_u32(v: u32) -> Self {
+ v as $name
+ }
+ #[inline(always)]
+ fn from_u64(v: u64) -> Self {
+ v as $name
+ }
+ #[inline(always)]
+ fn from_u128(v: u128) -> Self {
+ v as $name
+ }
+ #[inline(always)]
+ fn from_usize(v: usize) -> Self {
+ v as $name
+ }
+
+ #[inline(always)]
+ fn to_u8(&self) -> u8 {
+ (*self) as u8
+ }
+ #[inline(always)]
+ fn to_u16(&self) -> u16 {
+ (*self) as u16
+ }
+ #[inline(always)]
+ fn to_u32(&self) -> u32 {
+ (*self) as u32
+ }
+ #[inline(always)]
+ fn to_u64(&self) -> u64 {
+ (*self) as u64
+ }
+ #[inline(always)]
+ fn to_u128(&self) -> u128 {
+ (*self) as u128
+ }
+ #[inline(always)]
+ fn to_usize(&self) -> usize {
+ (*self) as usize
+ }
+
+ #[inline(always)]
+ fn from_u8_opt(v: u8) -> Option<Self> {
+ v.try_into().ok()
+ }
+ #[inline(always)]
+ fn from_u16_opt(v: u16) -> Option<Self> {
+ v.try_into().ok()
+ }
+ #[inline(always)]
+ fn from_u32_opt(v: u32) -> Option<Self> {
+ v.try_into().ok()
+ }
+ #[inline(always)]
+ fn from_u64_opt(v: u64) -> Option<Self> {
+ v.try_into().ok()
+ }
+ #[inline(always)]
+ fn from_u128_opt(v: u128) -> Option<Self> {
+ v.try_into().ok()
+ }
+ #[inline(always)]
+ fn from_usize_opt(v: usize) -> Option<Self> {
+ v.try_into().ok()
+ }
+
+ #[inline(always)]
+ fn to_u8_opt(&self) -> Option<u8> {
+ (*self).try_into().ok()
+ }
+ #[inline(always)]
+ fn to_u16_opt(&self) -> Option<u16> {
+ (*self).try_into().ok()
+ }
+ #[inline(always)]
+ fn to_u32_opt(&self) -> Option<u32> {
+ (*self).try_into().ok()
+ }
+ #[inline(always)]
+ fn to_u64_opt(&self) -> Option<u64> {
+ (*self).try_into().ok()
+ }
+ #[inline(always)]
+ fn to_u128_opt(&self) -> Option<u128> {
+ (*self).try_into().ok()
+ }
+ #[inline(always)]
+ fn to_usize_opt(&self) -> Option<usize> {
+ (*self).try_into().ok()
+ }
+
+ #[inline(always)]
+ fn to_u64_array<const O: usize>(&self) -> [u64; O] {
+ let mut array = [0; O];
+ if O > 0 {
+ array[0] = lo(*self);
+ }
+ if O > 1 && $preferred_array_len == 2 {
+ array[1] = hi(*self);
+ }
+ array
+ }
+ #[inline(always)]
+ fn to_u64_array_opt<const O: usize>(&self) -> Option<[u64; O]> {
+ if O == 0 && *self != 0 {
+ None
+ } else if O == 1 && hi(*self) != 0 {
+ None
+ } else {
+ Some(self.to_u64_array())
+ }
+ }
+
+ #[inline(always)]
+ fn from_u64_array<const O: usize>(v: [u64; O]) -> Self {
+ if O == 0 {
+ 0
+ } else if O > 1 && $preferred_array_len == 2 {
+ Self::from_u128(v[0] as u128 | ((v[1] as u128) << 64))
+ } else {
+ Self::from_u64(v[0])
+ }
+ }
+ #[inline(always)]
+ fn from_u64_array_opt<const O: usize>(v: [u64; O]) -> Option<Self> {
+ if O == 0 {
+ Some(0)
+ } else if O == 1 {
+ Self::from_u64_opt(v[0])
+ } else {
+ for i in 2..O {
+ if v[i] != 0 {
+ return None;
+ }
+ }
+ Self::from_u128_opt(v[0] as u128 | ((v[1] as u128) << 64))
+ }
+ }
+
+ #[inline(always)]
+ fn to_u64_slice(&self, out: &mut [u64]) {
+ if out.len() > 0 {
+ out[0] = lo(*self);
+ }
+ if out.len() > 1 && $preferred_array_len == 2 {
+ out[1] = hi(*self);
+ }
+ for i in $preferred_array_len..out.len() {
+ out[i] = 0;
+ }
+ }
+ #[inline(always)]
+ #[must_use]
+ fn to_u64_slice_opt(&self, out: &mut [u64]) -> Option<()> {
+ if out.len() == 0 && *self != 0 {
+ None
+ } else if out.len() == 1 && hi(*self) != 0 {
+ None
+ } else {
+ self.to_u64_slice(out);
+ Some(())
+ }
+ }
+
+ #[inline(always)]
+ fn from_u64_slice(v: &[u64]) -> Self {
+ if v.len() == 0 {
+ 0
+ } else if v.len() > 1 && $preferred_array_len == 2 {
+ Self::from_u128(v[0] as u128 | ((v[1] as u128) << 64))
+ } else {
+ Self::from_u64(v[0])
+ }
+ }
+ #[inline(always)]
+ fn from_u64_slice_opt(v: &[u64]) -> Option<Self> {
+ if v.len() == 0 {
+ Some(0)
+ } else if v.len() == 1 {
+ Self::from_u64_opt(v[0])
+ } else {
+ for i in 2..v.len() {
+ if v[i] != 0 {
+ return None;
+ }
+ }
+ Self::from_u128_opt(v[0] as u128 | ((v[1] as u128) << 64))
+ }
+ }
+ }
+ };
+ };
+}
+prim!(u8, 8, 1);
+prim!(u16, 16, 1);
+prim!(u32, 32, 1);
+prim!(u64, 64, 1);
+prim!(u128, 128, 2);
+
+#[derive(Copy, Clone, Debug)]
+#[repr(transparent)]
+pub struct PrimitiveIter<T: EnumSetTypeRepr>(pub T);
+
+impl<T: EnumSetTypeRepr> Iterator for PrimitiveIter<T> {
+ type Item = u32;
+
+ fn next(&mut self) -> Option<Self::Item> {
+ if self.0.is_empty() {
+ None
+ } else {
+ let bit = self.0.trailing_zeros();
+ self.0.remove_bit(bit);
+ Some(bit)
+ }
+ }
+
+ fn size_hint(&self) -> (usize, Option<usize>) {
+ let left = self.0.count_ones() as usize;
+ (left, Some(left))
+ }
+}
+
+impl<T: EnumSetTypeRepr> DoubleEndedIterator for PrimitiveIter<T> {
+ fn next_back(&mut self) -> Option<Self::Item> {
+ if self.0.is_empty() {
+ None
+ } else {
+ let bit = T::WIDTH - 1 - self.0.leading_zeros();
+ self.0.remove_bit(bit);
+ Some(bit)
+ }
+ }
+}