From e012b177ead16b88afdbe680605ba954aea213d9 Mon Sep 17 00:00:00 2001 From: Mac L Date: Sat, 27 Jul 2024 01:18:25 +1000 Subject: [PATCH 1/6] Add bitvector --- ssz/Cargo.toml | 8 +- ssz/src/bitfield.rs | 1394 +++++++++++++++++++++++++++++++++++++++ ssz/src/encode/impls.rs | 4 +- ssz/src/lib.rs | 2 + 4 files changed, 1405 insertions(+), 3 deletions(-) create mode 100644 ssz/src/bitfield.rs diff --git a/ssz/Cargo.toml b/ssz/Cargo.toml index e9881b6..cda946e 100644 --- a/ssz/Cargo.toml +++ b/ssz/Cargo.toml @@ -18,8 +18,14 @@ ethereum_ssz_derive = { version = "0.5.4", path = "../ssz_derive" } [dependencies] ethereum-types = "0.14.1" +ethereum_serde_utils = "0.5.0" smallvec = { version = "1.6.1", features = ["const_generics"] } itertools = "0.13.0" +serde = "1.0.0" +serde_derive = "1.0.0" +typenum = "1.12.0" +derivative = "2.1.1" +arbitrary = { version = "1.0", features = ["derive"], optional = true } [features] -arbitrary = ["ethereum-types/arbitrary"] +arbitrary = ["dep:arbitrary", "ethereum-types/arbitrary"] diff --git a/ssz/src/bitfield.rs b/ssz/src/bitfield.rs new file mode 100644 index 0000000..1003956 --- /dev/null +++ b/ssz/src/bitfield.rs @@ -0,0 +1,1394 @@ +use crate::{Decode, DecodeError, Encode}; +use core::marker::PhantomData; +use derivative::Derivative; +use serde::de::{Deserialize, Deserializer}; +use serde::ser::{Serialize, Serializer}; +use serde_utils::hex::{encode as hex_encode, PrefixedHexVisitor}; +use smallvec::{smallvec, SmallVec, ToSmallVec}; +use typenum::Unsigned; + +/// Returned when an item encounters an error. +#[derive(PartialEq, Debug, Clone)] +pub enum Error { + OutOfBounds { + i: usize, + len: usize, + }, + /// A `BitList` does not have a set bit, therefore it's length is unknowable. + MissingLengthInformation, + /// A `BitList` has excess bits set to true. + ExcessBits, + /// A `BitList` has an invalid number of bytes for a given bit length. + InvalidByteCount { + given: usize, + expected: usize, + }, +} + +/// Maximum number of bytes to store on the stack in a bitfield's `SmallVec`. +/// +/// The default of 32 bytes is enough to take us through to ~500K validators, as the byte length of +/// attestation bitfields is roughly `N // 32 slots // 64 committes // 8 bits`. +pub const SMALLVEC_LEN: usize = 32; + +/// A marker trait applied to `Variable` and `Fixed` that defines the behaviour of a `Bitfield`. +pub trait BitfieldBehaviour: Clone {} + +/// A marker struct used to declare SSZ `Variable` behaviour on a `Bitfield`. +/// +/// See the [`Bitfield`](struct.Bitfield.html) docs for usage. +#[derive(Clone, PartialEq, Eq, Debug)] +pub struct Variable { + _phantom: PhantomData, +} + +/// A marker struct used to declare SSZ `Fixed` behaviour on a `Bitfield`. +/// +/// See the [`Bitfield`](struct.Bitfield.html) docs for usage. +#[derive(Clone, PartialEq, Eq, Debug)] +pub struct Fixed { + _phantom: PhantomData, +} + +impl BitfieldBehaviour for Variable {} +impl BitfieldBehaviour for Fixed {} + +/// A heap-allocated, ordered, variable-length collection of `bool` values, limited to `N` bits. +pub type BitList = Bitfield>; + +/// A heap-allocated, ordered, fixed-length collection of `bool` values, with `N` bits. +/// +/// See [Bitfield](struct.Bitfield.html) documentation. +pub type BitVector = Bitfield>; + +/// A heap-allocated, ordered, fixed-length, collection of `bool` values. Use of +/// [`BitList`](type.BitList.html) or [`BitVector`](type.BitVector.html) type aliases is preferred +/// over direct use of this struct. +/// +/// The `T` type parameter is used to define length behaviour with the `Variable` or `Fixed` marker +/// structs. +/// +/// The length of the Bitfield is set at instantiation (i.e., runtime, not compile time). However, +/// use with a `Variable` sets a type-level (i.e., compile-time) maximum length and `Fixed` +/// provides a type-level fixed length. +/// +/// ## Example +/// +/// The example uses the following crate-level type aliases: +/// +/// - `BitList` is an alias for `Bitfield>` +/// - `BitVector` is an alias for `Bitfield>` +/// +/// ``` +/// use ssz_types::{BitVector, BitList, typenum}; +/// +/// // `BitList` has a type-level maximum length. The length of the list is specified at runtime +/// // and it must be less than or equal to `N`. After instantiation, `BitList` cannot grow or +/// // shrink. +/// type BitList8 = BitList; +/// +/// // Creating a `BitList` with a larger-than-`N` capacity returns `None`. +/// assert!(BitList8::with_capacity(9).is_err()); +/// +/// let mut bitlist = BitList8::with_capacity(4).unwrap(); // `BitList` permits a capacity of less than the maximum. +/// assert!(bitlist.set(3, true).is_ok()); // Setting inside the instantiation capacity is permitted. +/// assert!(bitlist.set(5, true).is_err()); // Setting outside that capacity is not. +/// +/// // `BitVector` has a type-level fixed length. Unlike `BitList`, it cannot be instantiated with a custom length +/// // or grow/shrink. +/// type BitVector8 = BitVector; +/// +/// let mut bitvector = BitVector8::new(); +/// assert_eq!(bitvector.len(), 8); // `BitVector` length is fixed at the type-level. +/// assert!(bitvector.set(7, true).is_ok()); // Setting inside the capacity is permitted. +/// assert!(bitvector.set(9, true).is_err()); // Setting outside the capacity is not. +/// +/// ``` +/// +/// ## Note +/// +/// The internal representation of the bitfield is the same as that required by SSZ. The lowest +/// byte (by `Vec` index) stores the lowest bit-indices and the right-most bit stores the lowest +/// bit-index. E.g., `smallvec![0b0000_0001, 0b0000_0010]` has bits `0, 9` set. +#[derive(Clone, Debug, Derivative)] +#[derivative(PartialEq, Eq, Hash(bound = ""))] +pub struct Bitfield { + bytes: SmallVec<[u8; SMALLVEC_LEN]>, + len: usize, + _phantom: PhantomData, +} + +impl Bitfield> { + /// Instantiate with capacity for `num_bits` boolean values. The length cannot be grown or + /// shrunk after instantiation. + /// + /// All bits are initialized to `false`. + /// + /// Returns `None` if `num_bits > N`. + pub fn with_capacity(num_bits: usize) -> Result { + if num_bits <= N::to_usize() { + Ok(Self { + bytes: smallvec![0; bytes_for_bit_len(num_bits)], + len: num_bits, + _phantom: PhantomData, + }) + } else { + Err(Error::OutOfBounds { + i: Self::max_len(), + len: Self::max_len(), + }) + } + } + + /// Equal to `N` regardless of the value supplied to `with_capacity`. + pub fn max_len() -> usize { + N::to_usize() + } + + /// Consumes `self`, returning a serialized representation. + /// + /// The output is faithful to the SSZ encoding of `self`, such that a leading `true` bit is + /// used to indicate the length of the bitfield. + /// + /// ## Example + /// ``` + /// use ssz_types::{BitList, typenum}; + /// use smallvec::SmallVec; + /// + /// type BitList8 = BitList; + /// + /// let b = BitList8::with_capacity(4).unwrap(); + /// + /// assert_eq!(b.into_bytes(), SmallVec::from_buf([0b0001_0000])); + /// ``` + pub fn into_bytes(self) -> SmallVec<[u8; SMALLVEC_LEN]> { + let len = self.len(); + let mut bytes = self.bytes; + + bytes.resize(bytes_for_bit_len(len + 1), 0); + + let mut bitfield: Bitfield> = Bitfield::from_raw_bytes(bytes, len + 1) + .unwrap_or_else(|_| { + unreachable!( + "Bitfield with {} bytes must have enough capacity for {} bits.", + bytes_for_bit_len(len + 1), + len + 1 + ) + }); + bitfield + .set(len, true) + .expect("len must be in bounds for bitfield."); + + bitfield.bytes + } + + /// Instantiates a new instance from `bytes`. Consumes the same format that `self.into_bytes()` + /// produces (SSZ). + /// + /// Returns `None` if `bytes` are not a valid encoding. + pub fn from_bytes(bytes: SmallVec<[u8; SMALLVEC_LEN]>) -> Result { + let bytes_len = bytes.len(); + let mut initial_bitfield: Bitfield> = { + let num_bits = bytes.len() * 8; + Bitfield::from_raw_bytes(bytes, num_bits)? + }; + + let len = initial_bitfield + .highest_set_bit() + .ok_or(Error::MissingLengthInformation)?; + + // The length bit should be in the last byte, or else it means we have too many bytes. + if len / 8 + 1 != bytes_len { + return Err(Error::InvalidByteCount { + given: bytes_len, + expected: len / 8 + 1, + }); + } + + if len <= Self::max_len() { + initial_bitfield + .set(len, false) + .expect("Bit has been confirmed to exist"); + + let mut bytes = initial_bitfield.into_raw_bytes(); + + bytes.truncate(bytes_for_bit_len(len)); + + Self::from_raw_bytes(bytes, len) + } else { + Err(Error::OutOfBounds { + i: Self::max_len(), + len: Self::max_len(), + }) + } + } + + /// Compute the intersection of two BitLists of potentially different lengths. + /// + /// Return a new BitList with length equal to the shorter of the two inputs. + pub fn intersection(&self, other: &Self) -> Self { + let min_len = std::cmp::min(self.len(), other.len()); + let mut result = Self::with_capacity(min_len).expect("min len always less than N"); + // Bitwise-and the bytes together, starting from the left of each vector. This takes care + // of masking out any entries beyond `min_len` as well, assuming the bitfield doesn't + // contain any set bits beyond its length. + for i in 0..result.bytes.len() { + result.bytes[i] = self.bytes[i] & other.bytes[i]; + } + result + } + + /// Compute the union of two BitLists of potentially different lengths. + /// + /// Return a new BitList with length equal to the longer of the two inputs. + pub fn union(&self, other: &Self) -> Self { + let max_len = std::cmp::max(self.len(), other.len()); + let mut result = Self::with_capacity(max_len).expect("max len always less than N"); + for i in 0..result.bytes.len() { + result.bytes[i] = + self.bytes.get(i).copied().unwrap_or(0) | other.bytes.get(i).copied().unwrap_or(0); + } + result + } + + /// Returns `true` if `self` is a subset of `other` and `false` otherwise. + pub fn is_subset(&self, other: &Self) -> bool { + self.difference(other).is_zero() + } +} + +impl Bitfield> { + /// Instantiate a new `Bitfield` with a fixed-length of `N` bits. + /// + /// All bits are initialized to `false`. + pub fn new() -> Self { + Self { + bytes: smallvec![0; bytes_for_bit_len(Self::capacity())], + len: Self::capacity(), + _phantom: PhantomData, + } + } + + /// Returns `N`, the number of bits in `Self`. + pub fn capacity() -> usize { + N::to_usize() + } + + /// Consumes `self`, returning a serialized representation. + /// + /// The output is faithful to the SSZ encoding of `self`. + /// + /// ## Example + /// ``` + /// use ssz_types::{BitVector, typenum}; + /// use smallvec::SmallVec; + /// + /// type BitVector4 = BitVector; + /// + /// assert_eq!(BitVector4::new().into_bytes(), SmallVec::from_buf([0b0000_0000])); + /// ``` + pub fn into_bytes(self) -> SmallVec<[u8; SMALLVEC_LEN]> { + self.into_raw_bytes() + } + + /// Instantiates a new instance from `bytes`. Consumes the same format that `self.into_bytes()` + /// produces (SSZ). + /// + /// Returns `None` if `bytes` are not a valid encoding. + pub fn from_bytes(bytes: SmallVec<[u8; SMALLVEC_LEN]>) -> Result { + Self::from_raw_bytes(bytes, Self::capacity()) + } + + /// Compute the intersection of two fixed-length `Bitfield`s. + /// + /// Return a new fixed-length `Bitfield`. + pub fn intersection(&self, other: &Self) -> Self { + let mut result = Self::new(); + // Bitwise-and the bytes together, starting from the left of each vector. This takes care + // of masking out any entries beyond `min_len` as well, assuming the bitfield doesn't + // contain any set bits beyond its length. + for i in 0..result.bytes.len() { + result.bytes[i] = self.bytes[i] & other.bytes[i]; + } + result + } + + /// Compute the union of two fixed-length `Bitfield`s. + /// + /// Return a new fixed-length `Bitfield`. + pub fn union(&self, other: &Self) -> Self { + let mut result = Self::new(); + for i in 0..result.bytes.len() { + result.bytes[i] = + self.bytes.get(i).copied().unwrap_or(0) | other.bytes.get(i).copied().unwrap_or(0); + } + result + } + + /// Returns `true` if `self` is a subset of `other` and `false` otherwise. + pub fn is_subset(&self, other: &Self) -> bool { + self.difference(other).is_zero() + } +} + +impl std::fmt::Display for Bitfield> { + fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { + let mut field: String = "".to_string(); + for i in self.iter() { + if i { + field.push('1') + } else { + field.push('0') + } + } + write!(f, "{field}") + } +} + +impl Default for Bitfield> { + fn default() -> Self { + Self::new() + } +} + +impl Bitfield { + /// Sets the `i`'th bit to `value`. + /// + /// Returns `None` if `i` is out-of-bounds of `self`. + pub fn set(&mut self, i: usize, value: bool) -> Result<(), Error> { + let len = self.len; + + if i < len { + let byte = self + .bytes + .get_mut(i / 8) + .ok_or(Error::OutOfBounds { i, len })?; + + if value { + *byte |= 1 << (i % 8) + } else { + *byte &= !(1 << (i % 8)) + } + + Ok(()) + } else { + Err(Error::OutOfBounds { i, len: self.len }) + } + } + + /// Returns the value of the `i`'th bit. + /// + /// Returns `Error` if `i` is out-of-bounds of `self`. + pub fn get(&self, i: usize) -> Result { + if i < self.len { + let byte = self + .bytes + .get(i / 8) + .ok_or(Error::OutOfBounds { i, len: self.len })?; + + Ok(*byte & 1 << (i % 8) > 0) + } else { + Err(Error::OutOfBounds { i, len: self.len }) + } + } + + /// Returns the number of bits stored in `self`. + pub fn len(&self) -> usize { + self.len + } + + /// Returns `true` if `self.len() == 0`. + pub fn is_empty(&self) -> bool { + self.len == 0 + } + + /// Returns the underlying bytes representation of the bitfield. + pub fn into_raw_bytes(self) -> SmallVec<[u8; SMALLVEC_LEN]> { + self.bytes + } + + /// Returns a view into the underlying bytes representation of the bitfield. + pub fn as_slice(&self) -> &[u8] { + &self.bytes + } + + /// Instantiates from the given `bytes`, which are the same format as output from + /// `self.into_raw_bytes()`. + /// + /// Returns `None` if: + /// + /// - `bytes` is not the minimal required bytes to represent a bitfield of `bit_len` bits. + /// - `bit_len` is not a multiple of 8 and `bytes` contains set bits that are higher than, or + /// equal to `bit_len`. + fn from_raw_bytes(bytes: SmallVec<[u8; SMALLVEC_LEN]>, bit_len: usize) -> Result { + if bit_len == 0 { + if bytes.len() == 1 && bytes[0] == 0 { + // A bitfield with `bit_len` 0 can only be represented by a single zero byte. + Ok(Self { + bytes, + len: 0, + _phantom: PhantomData, + }) + } else { + Err(Error::ExcessBits) + } + } else if bytes.len() != bytes_for_bit_len(bit_len) { + // The number of bytes must be the minimum required to represent `bit_len`. + Err(Error::InvalidByteCount { + given: bytes.len(), + expected: bytes_for_bit_len(bit_len), + }) + } else { + // Ensure there are no bits higher than `bit_len` that are set to true. + let (mask, _) = u8::max_value().overflowing_shr(8 - (bit_len as u32 % 8)); + + if (bytes.last().expect("Guarded against empty bytes") & !mask) == 0 { + Ok(Self { + bytes, + len: bit_len, + _phantom: PhantomData, + }) + } else { + Err(Error::ExcessBits) + } + } + } + + /// Returns the `Some(i)` where `i` is the highest index with a set bit. Returns `None` if + /// there are no set bits. + pub fn highest_set_bit(&self) -> Option { + self.bytes + .iter() + .enumerate() + .rev() + .find(|(_, byte)| **byte > 0) + .map(|(i, byte)| i * 8 + 7 - byte.leading_zeros() as usize) + } + + /// Returns an iterator across bitfield `bool` values, starting at the lowest index. + pub fn iter(&self) -> BitIter<'_, T> { + BitIter { + bitfield: self, + i: 0, + } + } + + /// Returns true if no bits are set. + pub fn is_zero(&self) -> bool { + self.bytes.iter().all(|byte| *byte == 0) + } + + /// Returns the number of bits that are set to `true`. + pub fn num_set_bits(&self) -> usize { + self.bytes + .iter() + .map(|byte| byte.count_ones() as usize) + .sum() + } + + /// Compute the difference of this Bitfield and another of potentially different length. + pub fn difference(&self, other: &Self) -> Self { + let mut result = self.clone(); + result.difference_inplace(other); + result + } + + /// Compute the difference of this Bitfield and another of potentially different length. + pub fn difference_inplace(&mut self, other: &Self) { + let min_byte_len = std::cmp::min(self.bytes.len(), other.bytes.len()); + + for i in 0..min_byte_len { + self.bytes[i] &= !other.bytes[i]; + } + } + + /// Shift the bits to higher indices, filling the lower indices with zeroes. + /// + /// The amount to shift by, `n`, must be less than or equal to `self.len()`. + pub fn shift_up(&mut self, n: usize) -> Result<(), Error> { + if n <= self.len() { + // Shift the bits up (starting from the high indices to avoid overwriting) + for i in (n..self.len()).rev() { + self.set(i, self.get(i - n)?)?; + } + // Zero the low bits + for i in 0..n { + self.set(i, false).unwrap(); + } + Ok(()) + } else { + Err(Error::OutOfBounds { + i: n, + len: self.len(), + }) + } + } +} + +/// Returns the minimum required bytes to represent a given number of bits. +/// +/// `bit_len == 0` requires a single byte. +fn bytes_for_bit_len(bit_len: usize) -> usize { + std::cmp::max(1, (bit_len + 7) / 8) +} + +/// An iterator over the bits in a `Bitfield`. +pub struct BitIter<'a, T> { + bitfield: &'a Bitfield, + i: usize, +} + +impl<'a, T: BitfieldBehaviour> Iterator for BitIter<'a, T> { + type Item = bool; + + fn next(&mut self) -> Option { + let res = self.bitfield.get(self.i).ok()?; + self.i += 1; + Some(res) + } +} + +impl Encode for Bitfield> { + fn is_ssz_fixed_len() -> bool { + false + } + + fn ssz_bytes_len(&self) -> usize { + // We could likely do better than turning this into bytes and reading the length, however + // it is kept this way for simplicity. + self.clone().into_bytes().len() + } + + fn ssz_append(&self, buf: &mut Vec) { + buf.extend_from_slice(&self.clone().into_bytes()) + } +} + +impl Decode for Bitfield> { + fn is_ssz_fixed_len() -> bool { + false + } + + fn from_ssz_bytes(bytes: &[u8]) -> Result { + Self::from_bytes(bytes.to_smallvec()) + .map_err(|e| DecodeError::BytesInvalid(format!("BitList failed to decode: {:?}", e))) + } +} + +impl Encode for Bitfield> { + fn is_ssz_fixed_len() -> bool { + true + } + + fn ssz_bytes_len(&self) -> usize { + self.as_slice().len() + } + + fn ssz_fixed_len() -> usize { + bytes_for_bit_len(N::to_usize()) + } + + fn ssz_append(&self, buf: &mut Vec) { + buf.extend_from_slice(&self.clone().into_bytes()) + } +} + +impl Decode for Bitfield> { + fn is_ssz_fixed_len() -> bool { + true + } + + fn ssz_fixed_len() -> usize { + bytes_for_bit_len(N::to_usize()) + } + + fn from_ssz_bytes(bytes: &[u8]) -> Result { + Self::from_bytes(bytes.to_smallvec()) + .map_err(|e| DecodeError::BytesInvalid(format!("BitVector failed to decode: {:?}", e))) + } +} + +impl Serialize for Bitfield> { + /// Serde serialization is compliant with the Ethereum YAML test format. + fn serialize(&self, serializer: S) -> Result + where + S: Serializer, + { + serializer.serialize_str(&hex_encode(self.as_ssz_bytes())) + } +} + +impl<'de, N: Unsigned + Clone> Deserialize<'de> for Bitfield> { + /// Serde serialization is compliant with the Ethereum YAML test format. + fn deserialize(deserializer: D) -> Result + where + D: Deserializer<'de>, + { + let bytes = deserializer.deserialize_str(PrefixedHexVisitor)?; + Self::from_ssz_bytes(&bytes) + .map_err(|e| serde::de::Error::custom(format!("Bitfield {:?}", e))) + } +} + +impl Serialize for Bitfield> { + /// Serde serialization is compliant with the Ethereum YAML test format. + fn serialize(&self, serializer: S) -> Result + where + S: Serializer, + { + serializer.serialize_str(&hex_encode(self.as_ssz_bytes())) + } +} + +impl<'de, N: Unsigned + Clone> Deserialize<'de> for Bitfield> { + /// Serde serialization is compliant with the Ethereum YAML test format. + fn deserialize(deserializer: D) -> Result + where + D: Deserializer<'de>, + { + let bytes = deserializer.deserialize_str(PrefixedHexVisitor)?; + Self::from_ssz_bytes(&bytes) + .map_err(|e| serde::de::Error::custom(format!("Bitfield {:?}", e))) + } +} + +#[cfg(feature = "arbitrary")] +impl arbitrary::Arbitrary<'_> for Bitfield> { + fn arbitrary(u: &mut arbitrary::Unstructured<'_>) -> arbitrary::Result { + let size = N::to_usize(); + let mut vec = smallvec![0u8; size]; + u.fill_buffer(&mut vec)?; + Ok(Self::from_bytes(vec).map_err(|_| arbitrary::Error::IncorrectFormat)?) + } +} + +#[cfg(feature = "arbitrary")] +impl arbitrary::Arbitrary<'_> for Bitfield> { + fn arbitrary(u: &mut arbitrary::Unstructured<'_>) -> arbitrary::Result { + let max_size = N::to_usize(); + let rand = usize::arbitrary(u)?; + let size = std::cmp::min(rand, max_size); + let mut vec = smallvec![0u8; size]; + u.fill_buffer(&mut vec)?; + Ok(Self::from_bytes(vec).map_err(|_| arbitrary::Error::IncorrectFormat)?) + } +} + +#[cfg(test)] +mod bitvector { + use super::*; + use crate::BitVector; + + pub type BitVector0 = BitVector; + pub type BitVector1 = BitVector; + pub type BitVector4 = BitVector; + pub type BitVector8 = BitVector; + pub type BitVector16 = BitVector; + pub type BitVector64 = BitVector; + + #[test] + fn ssz_encode() { + assert_eq!(BitVector0::new().as_ssz_bytes(), vec![0b0000_0000]); + assert_eq!(BitVector1::new().as_ssz_bytes(), vec![0b0000_0000]); + assert_eq!(BitVector4::new().as_ssz_bytes(), vec![0b0000_0000]); + assert_eq!(BitVector8::new().as_ssz_bytes(), vec![0b0000_0000]); + assert_eq!( + BitVector16::new().as_ssz_bytes(), + vec![0b0000_0000, 0b0000_0000] + ); + + let mut b = BitVector8::new(); + for i in 0..8 { + b.set(i, true).unwrap(); + } + assert_eq!(b.as_ssz_bytes(), vec![255]); + + let mut b = BitVector4::new(); + for i in 0..4 { + b.set(i, true).unwrap(); + } + assert_eq!(b.as_ssz_bytes(), vec![0b0000_1111]); + } + + #[test] + fn ssz_decode() { + assert!(BitVector0::from_ssz_bytes(&[0b0000_0000]).is_ok()); + assert!(BitVector0::from_ssz_bytes(&[0b0000_0001]).is_err()); + assert!(BitVector0::from_ssz_bytes(&[0b0000_0010]).is_err()); + + assert!(BitVector1::from_ssz_bytes(&[0b0000_0001]).is_ok()); + assert!(BitVector1::from_ssz_bytes(&[0b0000_0010]).is_err()); + assert!(BitVector1::from_ssz_bytes(&[0b0000_0100]).is_err()); + assert!(BitVector1::from_ssz_bytes(&[0b0000_0000, 0b0000_0000]).is_err()); + + assert!(BitVector8::from_ssz_bytes(&[0b0000_0000]).is_ok()); + assert!(BitVector8::from_ssz_bytes(&[1, 0b0000_0000]).is_err()); + assert!(BitVector8::from_ssz_bytes(&[0b0000_0000, 1]).is_err()); + assert!(BitVector8::from_ssz_bytes(&[0b0000_0001]).is_ok()); + assert!(BitVector8::from_ssz_bytes(&[0b0000_0010]).is_ok()); + assert!(BitVector8::from_ssz_bytes(&[0b0000_0100, 0b0000_0001]).is_err()); + assert!(BitVector8::from_ssz_bytes(&[0b0000_0100, 0b0000_0010]).is_err()); + assert!(BitVector8::from_ssz_bytes(&[0b0000_0100, 0b0000_0100]).is_err()); + + assert!(BitVector16::from_ssz_bytes(&[0b0000_0000]).is_err()); + assert!(BitVector16::from_ssz_bytes(&[0b0000_0000, 0b0000_0000]).is_ok()); + assert!(BitVector16::from_ssz_bytes(&[1, 0b0000_0000, 0b0000_0000]).is_err()); + } + + #[test] + fn intersection() { + let a = BitVector16::from_raw_bytes(smallvec![0b1100, 0b0001], 16).unwrap(); + let b = BitVector16::from_raw_bytes(smallvec![0b1011, 0b1001], 16).unwrap(); + let c = BitVector16::from_raw_bytes(smallvec![0b1000, 0b0001], 16).unwrap(); + + assert_eq!(a.intersection(&b), c); + assert_eq!(b.intersection(&a), c); + assert_eq!(a.intersection(&c), c); + assert_eq!(b.intersection(&c), c); + assert_eq!(a.intersection(&a), a); + assert_eq!(b.intersection(&b), b); + assert_eq!(c.intersection(&c), c); + } + + #[test] + fn intersection_diff_length() { + let a = BitVector16::from_bytes(smallvec![0b0010_1110, 0b0010_1011]).unwrap(); + let b = BitVector16::from_bytes(smallvec![0b0010_1101, 0b0000_0001]).unwrap(); + let c = BitVector16::from_bytes(smallvec![0b0010_1100, 0b0000_0001]).unwrap(); + + assert_eq!(a.len(), 16); + assert_eq!(b.len(), 16); + assert_eq!(c.len(), 16); + assert_eq!(a.intersection(&b), c); + assert_eq!(b.intersection(&a), c); + } + + #[test] + fn subset() { + let a = BitVector16::from_raw_bytes(smallvec![0b1000, 0b0001], 16).unwrap(); + let b = BitVector16::from_raw_bytes(smallvec![0b1100, 0b0001], 16).unwrap(); + let c = BitVector16::from_raw_bytes(smallvec![0b1100, 0b1001], 16).unwrap(); + + assert_eq!(a.len(), 16); + assert_eq!(b.len(), 16); + assert_eq!(c.len(), 16); + + // a vector is always a subset of itself + assert!(a.is_subset(&a)); + assert!(b.is_subset(&b)); + assert!(c.is_subset(&c)); + + assert!(a.is_subset(&b)); + assert!(a.is_subset(&c)); + assert!(b.is_subset(&c)); + + assert!(!b.is_subset(&a)); + assert!(!c.is_subset(&a)); + assert!(!c.is_subset(&b)); + } + + #[test] + fn union() { + let a = BitVector16::from_raw_bytes(smallvec![0b1100, 0b0001], 16).unwrap(); + let b = BitVector16::from_raw_bytes(smallvec![0b1011, 0b1001], 16).unwrap(); + let c = BitVector16::from_raw_bytes(smallvec![0b1111, 0b1001], 16).unwrap(); + + assert_eq!(a.union(&b), c); + assert_eq!(b.union(&a), c); + assert_eq!(a.union(&a), a); + assert_eq!(b.union(&b), b); + assert_eq!(c.union(&c), c); + } + + #[test] + fn union_diff_length() { + let a = BitVector16::from_bytes(smallvec![0b0010_1011, 0b0010_1110]).unwrap(); + let b = BitVector16::from_bytes(smallvec![0b0000_0001, 0b0010_1101]).unwrap(); + let c = BitVector16::from_bytes(smallvec![0b0010_1011, 0b0010_1111]).unwrap(); + + assert_eq!(a.len(), c.len()); + assert_eq!(a.union(&b), c); + assert_eq!(b.union(&a), c); + } + + #[test] + fn ssz_round_trip() { + assert_round_trip(BitVector0::new()); + + let mut b = BitVector1::new(); + b.set(0, true).unwrap(); + assert_round_trip(b); + + let mut b = BitVector8::new(); + for j in 0..8 { + if j % 2 == 0 { + b.set(j, true).unwrap(); + } + } + assert_round_trip(b); + + let mut b = BitVector8::new(); + for j in 0..8 { + b.set(j, true).unwrap(); + } + assert_round_trip(b); + + let mut b = BitVector16::new(); + for j in 0..16 { + if j % 2 == 0 { + b.set(j, true).unwrap(); + } + } + assert_round_trip(b); + + let mut b = BitVector16::new(); + for j in 0..16 { + b.set(j, true).unwrap(); + } + assert_round_trip(b); + } + + fn assert_round_trip(t: T) { + assert_eq!(T::from_ssz_bytes(&t.as_ssz_bytes()).unwrap(), t); + } + + #[test] + fn ssz_bytes_len() { + for i in 0..64 { + let mut bitfield = BitVector64::new(); + for j in 0..i { + bitfield.set(j, true).expect("should set bit in bounds"); + } + let bytes = bitfield.as_ssz_bytes(); + assert_eq!(bitfield.ssz_bytes_len(), bytes.len(), "i = {}", i); + } + } + + #[test] + fn excess_bits_nimbus() { + let bad = vec![0b0001_1111]; + + assert!(BitVector4::from_ssz_bytes(&bad).is_err()); + } + + // Ensure that stack size of a BitVector is manageable. + #[test] + fn size_of() { + assert_eq!(std::mem::size_of::(), SMALLVEC_LEN + 24); + } +} + +#[cfg(test)] +#[allow(clippy::cognitive_complexity)] +mod bitlist { + use super::*; + use crate::BitList; + + pub type BitList0 = BitList; + pub type BitList1 = BitList; + pub type BitList8 = BitList; + pub type BitList16 = BitList; + pub type BitList1024 = BitList; + + #[test] + fn ssz_encode() { + assert_eq!( + BitList0::with_capacity(0).unwrap().as_ssz_bytes(), + vec![0b0000_0001], + ); + + assert_eq!( + BitList1::with_capacity(0).unwrap().as_ssz_bytes(), + vec![0b0000_0001], + ); + + assert_eq!( + BitList1::with_capacity(1).unwrap().as_ssz_bytes(), + vec![0b0000_0010], + ); + + assert_eq!( + BitList8::with_capacity(8).unwrap().as_ssz_bytes(), + vec![0b0000_0000, 0b0000_0001], + ); + + assert_eq!( + BitList8::with_capacity(7).unwrap().as_ssz_bytes(), + vec![0b1000_0000] + ); + + let mut b = BitList8::with_capacity(8).unwrap(); + for i in 0..8 { + b.set(i, true).unwrap(); + } + assert_eq!(b.as_ssz_bytes(), vec![255, 0b0000_0001]); + + let mut b = BitList8::with_capacity(8).unwrap(); + for i in 0..4 { + b.set(i, true).unwrap(); + } + assert_eq!(b.as_ssz_bytes(), vec![0b0000_1111, 0b0000_0001]); + + assert_eq!( + BitList16::with_capacity(16).unwrap().as_ssz_bytes(), + vec![0b0000_0000, 0b0000_0000, 0b0000_0001] + ); + } + + #[test] + fn ssz_decode() { + assert!(BitList0::from_ssz_bytes(&[]).is_err()); + assert!(BitList1::from_ssz_bytes(&[]).is_err()); + assert!(BitList8::from_ssz_bytes(&[]).is_err()); + assert!(BitList16::from_ssz_bytes(&[]).is_err()); + + assert!(BitList0::from_ssz_bytes(&[0b0000_0000]).is_err()); + assert!(BitList1::from_ssz_bytes(&[0b0000_0000, 0b0000_0000]).is_err()); + assert!(BitList8::from_ssz_bytes(&[0b0000_0000]).is_err()); + assert!(BitList16::from_ssz_bytes(&[0b0000_0000]).is_err()); + + assert!(BitList0::from_ssz_bytes(&[0b0000_0001]).is_ok()); + assert!(BitList0::from_ssz_bytes(&[0b0000_0010]).is_err()); + + assert!(BitList1::from_ssz_bytes(&[0b0000_0001]).is_ok()); + assert!(BitList1::from_ssz_bytes(&[0b0000_0010]).is_ok()); + assert!(BitList1::from_ssz_bytes(&[0b0000_0100]).is_err()); + + assert!(BitList8::from_ssz_bytes(&[0b0000_0001]).is_ok()); + assert!(BitList8::from_ssz_bytes(&[0b0000_0010]).is_ok()); + assert!(BitList8::from_ssz_bytes(&[0b0000_0001, 0b0000_0001]).is_ok()); + assert!(BitList8::from_ssz_bytes(&[0b0000_0001, 0b0000_0010]).is_err()); + assert!(BitList8::from_ssz_bytes(&[0b0000_0001, 0b0000_0100]).is_err()); + } + + #[test] + fn ssz_decode_extra_bytes() { + assert!(BitList0::from_ssz_bytes(&[0b0000_0001, 0b0000_0000]).is_err()); + assert!(BitList1::from_ssz_bytes(&[0b0000_0001, 0b0000_0000]).is_err()); + assert!(BitList8::from_ssz_bytes(&[0b0000_0001, 0b0000_0000]).is_err()); + assert!(BitList16::from_ssz_bytes(&[0b0000_0001, 0b0000_0000]).is_err()); + assert!(BitList1024::from_ssz_bytes(&[0b1000_0000, 0]).is_err()); + assert!(BitList1024::from_ssz_bytes(&[0b1000_0000, 0, 0]).is_err()); + assert!(BitList1024::from_ssz_bytes(&[0b1000_0000, 0, 0, 0, 0]).is_err()); + } + + #[test] + fn ssz_round_trip() { + assert_round_trip(BitList0::with_capacity(0).unwrap()); + + for i in 0..2 { + assert_round_trip(BitList1::with_capacity(i).unwrap()); + } + for i in 0..9 { + assert_round_trip(BitList8::with_capacity(i).unwrap()); + } + for i in 0..17 { + assert_round_trip(BitList16::with_capacity(i).unwrap()); + } + + let mut b = BitList1::with_capacity(1).unwrap(); + b.set(0, true).unwrap(); + assert_round_trip(b); + + for i in 0..8 { + let mut b = BitList8::with_capacity(i).unwrap(); + for j in 0..i { + if j % 2 == 0 { + b.set(j, true).unwrap(); + } + } + assert_round_trip(b); + + let mut b = BitList8::with_capacity(i).unwrap(); + for j in 0..i { + b.set(j, true).unwrap(); + } + assert_round_trip(b); + } + + for i in 0..16 { + let mut b = BitList16::with_capacity(i).unwrap(); + for j in 0..i { + if j % 2 == 0 { + b.set(j, true).unwrap(); + } + } + assert_round_trip(b); + + let mut b = BitList16::with_capacity(i).unwrap(); + for j in 0..i { + b.set(j, true).unwrap(); + } + assert_round_trip(b); + } + } + + fn assert_round_trip(t: T) { + assert_eq!(T::from_ssz_bytes(&t.as_ssz_bytes()).unwrap(), t); + } + + #[test] + fn from_raw_bytes() { + assert!(BitList1024::from_raw_bytes(smallvec![0b0000_0000], 0).is_ok()); + assert!(BitList1024::from_raw_bytes(smallvec![0b0000_0001], 1).is_ok()); + assert!(BitList1024::from_raw_bytes(smallvec![0b0000_0011], 2).is_ok()); + assert!(BitList1024::from_raw_bytes(smallvec![0b0000_0111], 3).is_ok()); + assert!(BitList1024::from_raw_bytes(smallvec![0b0000_1111], 4).is_ok()); + assert!(BitList1024::from_raw_bytes(smallvec![0b0001_1111], 5).is_ok()); + assert!(BitList1024::from_raw_bytes(smallvec![0b0011_1111], 6).is_ok()); + assert!(BitList1024::from_raw_bytes(smallvec![0b0111_1111], 7).is_ok()); + assert!(BitList1024::from_raw_bytes(smallvec![0b1111_1111], 8).is_ok()); + + assert!(BitList1024::from_raw_bytes(smallvec![0b1111_1111, 0b0000_0001], 9).is_ok()); + assert!(BitList1024::from_raw_bytes(smallvec![0b1111_1111, 0b0000_0011], 10).is_ok()); + assert!(BitList1024::from_raw_bytes(smallvec![0b1111_1111, 0b0000_0111], 11).is_ok()); + assert!(BitList1024::from_raw_bytes(smallvec![0b1111_1111, 0b0000_1111], 12).is_ok()); + assert!(BitList1024::from_raw_bytes(smallvec![0b1111_1111, 0b0001_1111], 13).is_ok()); + assert!(BitList1024::from_raw_bytes(smallvec![0b1111_1111, 0b0011_1111], 14).is_ok()); + assert!(BitList1024::from_raw_bytes(smallvec![0b1111_1111, 0b0111_1111], 15).is_ok()); + assert!(BitList1024::from_raw_bytes(smallvec![0b1111_1111, 0b1111_1111], 16).is_ok()); + + for i in 0..8 { + assert!(BitList1024::from_raw_bytes(smallvec![], i).is_err()); + assert!(BitList1024::from_raw_bytes(smallvec![0b1111_1111], i).is_err()); + assert!(BitList1024::from_raw_bytes(smallvec![0b0000_0000, 0b1111_1110], i).is_err()); + } + + assert!(BitList1024::from_raw_bytes(smallvec![0b0000_0001], 0).is_err()); + + assert!(BitList1024::from_raw_bytes(smallvec![0b0000_0001], 0).is_err()); + assert!(BitList1024::from_raw_bytes(smallvec![0b0000_0011], 1).is_err()); + assert!(BitList1024::from_raw_bytes(smallvec![0b0000_0111], 2).is_err()); + assert!(BitList1024::from_raw_bytes(smallvec![0b0000_1111], 3).is_err()); + assert!(BitList1024::from_raw_bytes(smallvec![0b0001_1111], 4).is_err()); + assert!(BitList1024::from_raw_bytes(smallvec![0b0011_1111], 5).is_err()); + assert!(BitList1024::from_raw_bytes(smallvec![0b0111_1111], 6).is_err()); + assert!(BitList1024::from_raw_bytes(smallvec![0b1111_1111], 7).is_err()); + + assert!(BitList1024::from_raw_bytes(smallvec![0b1111_1111, 0b0000_0001], 8).is_err()); + assert!(BitList1024::from_raw_bytes(smallvec![0b1111_1111, 0b0000_0011], 9).is_err()); + assert!(BitList1024::from_raw_bytes(smallvec![0b1111_1111, 0b0000_0111], 10).is_err()); + assert!(BitList1024::from_raw_bytes(smallvec![0b1111_1111, 0b0000_1111], 11).is_err()); + assert!(BitList1024::from_raw_bytes(smallvec![0b1111_1111, 0b0001_1111], 12).is_err()); + assert!(BitList1024::from_raw_bytes(smallvec![0b1111_1111, 0b0011_1111], 13).is_err()); + assert!(BitList1024::from_raw_bytes(smallvec![0b1111_1111, 0b0111_1111], 14).is_err()); + assert!(BitList1024::from_raw_bytes(smallvec![0b1111_1111, 0b1111_1111], 15).is_err()); + } + + fn test_set_unset(num_bits: usize) { + let mut bitfield = BitList1024::with_capacity(num_bits).unwrap(); + + for i in 0..=num_bits { + if i < num_bits { + // Starts as false + assert_eq!(bitfield.get(i), Ok(false)); + // Can be set true. + assert!(bitfield.set(i, true).is_ok()); + assert_eq!(bitfield.get(i), Ok(true)); + // Can be set false + assert!(bitfield.set(i, false).is_ok()); + assert_eq!(bitfield.get(i), Ok(false)); + } else { + assert!(bitfield.get(i).is_err()); + assert!(bitfield.set(i, true).is_err()); + assert!(bitfield.get(i).is_err()); + } + } + } + + fn test_bytes_round_trip(num_bits: usize) { + for i in 0..num_bits { + let mut bitfield = BitList1024::with_capacity(num_bits).unwrap(); + bitfield.set(i, true).unwrap(); + + let bytes = bitfield.clone().into_raw_bytes(); + assert_eq!(bitfield, Bitfield::from_raw_bytes(bytes, num_bits).unwrap()); + } + } + + #[test] + fn set_unset() { + for i in 0..8 * 5 { + test_set_unset(i) + } + } + + #[test] + fn bytes_round_trip() { + for i in 0..8 * 5 { + test_bytes_round_trip(i) + } + } + + /// Type-specialised `smallvec` macro for testing. + macro_rules! bytevec { + ($($x : expr),* $(,)*) => { + { + let __smallvec: SmallVec<[u8; SMALLVEC_LEN]> = smallvec!($($x),*); + __smallvec + } + }; + } + + #[test] + fn into_raw_bytes() { + let mut bitfield = BitList1024::with_capacity(9).unwrap(); + bitfield.set(0, true).unwrap(); + assert_eq!( + bitfield.clone().into_raw_bytes(), + bytevec![0b0000_0001, 0b0000_0000] + ); + bitfield.set(1, true).unwrap(); + assert_eq!( + bitfield.clone().into_raw_bytes(), + bytevec![0b0000_0011, 0b0000_0000] + ); + bitfield.set(2, true).unwrap(); + assert_eq!( + bitfield.clone().into_raw_bytes(), + bytevec![0b0000_0111, 0b0000_0000] + ); + bitfield.set(3, true).unwrap(); + assert_eq!( + bitfield.clone().into_raw_bytes(), + bytevec![0b0000_1111, 0b0000_0000] + ); + bitfield.set(4, true).unwrap(); + assert_eq!( + bitfield.clone().into_raw_bytes(), + bytevec![0b0001_1111, 0b0000_0000] + ); + bitfield.set(5, true).unwrap(); + assert_eq!( + bitfield.clone().into_raw_bytes(), + bytevec![0b0011_1111, 0b0000_0000] + ); + bitfield.set(6, true).unwrap(); + assert_eq!( + bitfield.clone().into_raw_bytes(), + bytevec![0b0111_1111, 0b0000_0000] + ); + bitfield.set(7, true).unwrap(); + assert_eq!( + bitfield.clone().into_raw_bytes(), + bytevec![0b1111_1111, 0b0000_0000] + ); + bitfield.set(8, true).unwrap(); + assert_eq!( + bitfield.into_raw_bytes(), + bytevec![0b1111_1111, 0b0000_0001] + ); + } + + #[test] + fn highest_set_bit() { + assert_eq!( + BitList1024::with_capacity(16).unwrap().highest_set_bit(), + None + ); + + assert_eq!( + BitList1024::from_raw_bytes(smallvec![0b0000_0001, 0b0000_0000], 16) + .unwrap() + .highest_set_bit(), + Some(0) + ); + + assert_eq!( + BitList1024::from_raw_bytes(smallvec![0b0000_0010, 0b0000_0000], 16) + .unwrap() + .highest_set_bit(), + Some(1) + ); + + assert_eq!( + BitList1024::from_raw_bytes(smallvec![0b0000_1000], 8) + .unwrap() + .highest_set_bit(), + Some(3) + ); + + assert_eq!( + BitList1024::from_raw_bytes(smallvec![0b0000_0000, 0b1000_0000], 16) + .unwrap() + .highest_set_bit(), + Some(15) + ); + } + + #[test] + fn intersection() { + let a = BitList1024::from_raw_bytes(smallvec![0b1100, 0b0001], 16).unwrap(); + let b = BitList1024::from_raw_bytes(smallvec![0b1011, 0b1001], 16).unwrap(); + let c = BitList1024::from_raw_bytes(smallvec![0b1000, 0b0001], 16).unwrap(); + + assert_eq!(a.intersection(&b), c); + assert_eq!(b.intersection(&a), c); + assert_eq!(a.intersection(&c), c); + assert_eq!(b.intersection(&c), c); + assert_eq!(a.intersection(&a), a); + assert_eq!(b.intersection(&b), b); + assert_eq!(c.intersection(&c), c); + } + + #[test] + fn subset() { + let a = BitList1024::from_raw_bytes(smallvec![0b1000, 0b0001], 16).unwrap(); + let b = BitList1024::from_raw_bytes(smallvec![0b1100, 0b0001], 16).unwrap(); + let c = BitList1024::from_raw_bytes(smallvec![0b1100, 0b1001], 16).unwrap(); + + assert_eq!(a.len(), 16); + assert_eq!(b.len(), 16); + assert_eq!(c.len(), 16); + + // a vector is always a subset of itself + assert!(a.is_subset(&a)); + assert!(b.is_subset(&b)); + assert!(c.is_subset(&c)); + + assert!(a.is_subset(&b)); + assert!(a.is_subset(&c)); + assert!(b.is_subset(&c)); + + assert!(!b.is_subset(&a)); + assert!(!c.is_subset(&a)); + assert!(!c.is_subset(&b)); + + let d = BitList1024::from_raw_bytes(smallvec![0b1100, 0b1001, 0b1010], 24).unwrap(); + assert!(d.is_subset(&d)); + + assert!(a.is_subset(&d)); + assert!(b.is_subset(&d)); + assert!(c.is_subset(&d)); + + // A bigger length bitlist cannot be a subset of a smaller length bitlist + assert!(!d.is_subset(&a)); + assert!(!d.is_subset(&b)); + assert!(!d.is_subset(&c)); + + let e = BitList1024::from_raw_bytes(smallvec![0b1100, 0b1001, 0b0000], 24).unwrap(); + assert!(e.is_subset(&c)); + assert!(c.is_subset(&e)); + } + + #[test] + fn intersection_diff_length() { + let a = BitList1024::from_bytes(smallvec![0b0010_1110, 0b0010_1011]).unwrap(); + let b = BitList1024::from_bytes(smallvec![0b0010_1101, 0b0000_0001]).unwrap(); + let c = BitList1024::from_bytes(smallvec![0b0010_1100, 0b0000_0001]).unwrap(); + let d = BitList1024::from_bytes(smallvec![0b0010_1110, 0b1111_1111, 0b1111_1111]).unwrap(); + + assert_eq!(a.len(), 13); + assert_eq!(b.len(), 8); + assert_eq!(c.len(), 8); + assert_eq!(d.len(), 23); + assert_eq!(a.intersection(&b), c); + assert_eq!(b.intersection(&a), c); + assert_eq!(a.intersection(&d), a); + assert_eq!(d.intersection(&a), a); + } + + #[test] + fn union() { + let a = BitList1024::from_raw_bytes(smallvec![0b1100, 0b0001], 16).unwrap(); + let b = BitList1024::from_raw_bytes(smallvec![0b1011, 0b1001], 16).unwrap(); + let c = BitList1024::from_raw_bytes(smallvec![0b1111, 0b1001], 16).unwrap(); + + assert_eq!(a.union(&b), c); + assert_eq!(b.union(&a), c); + assert_eq!(a.union(&a), a); + assert_eq!(b.union(&b), b); + assert_eq!(c.union(&c), c); + } + + #[test] + fn union_diff_length() { + let a = BitList1024::from_bytes(smallvec![0b0010_1011, 0b0010_1110]).unwrap(); + let b = BitList1024::from_bytes(smallvec![0b0000_0001, 0b0010_1101]).unwrap(); + let c = BitList1024::from_bytes(smallvec![0b0010_1011, 0b0010_1111]).unwrap(); + let d = BitList1024::from_bytes(smallvec![0b0010_1011, 0b1011_1110, 0b1000_1101]).unwrap(); + + assert_eq!(a.len(), c.len()); + assert_eq!(a.union(&b), c); + assert_eq!(b.union(&a), c); + assert_eq!(a.union(&d), d); + assert_eq!(d.union(&a), d); + } + + #[test] + fn difference() { + let a = BitList1024::from_raw_bytes(smallvec![0b1100, 0b0001], 16).unwrap(); + let b = BitList1024::from_raw_bytes(smallvec![0b1011, 0b1001], 16).unwrap(); + let a_b = BitList1024::from_raw_bytes(smallvec![0b0100, 0b0000], 16).unwrap(); + let b_a = BitList1024::from_raw_bytes(smallvec![0b0011, 0b1000], 16).unwrap(); + + assert_eq!(a.difference(&b), a_b); + assert_eq!(b.difference(&a), b_a); + assert!(a.difference(&a).is_zero()); + } + + #[test] + fn difference_diff_length() { + let a = BitList1024::from_raw_bytes(smallvec![0b0110, 0b1100, 0b0011], 24).unwrap(); + let b = BitList1024::from_raw_bytes(smallvec![0b1011, 0b1001], 16).unwrap(); + let a_b = BitList1024::from_raw_bytes(smallvec![0b0100, 0b0100, 0b0011], 24).unwrap(); + let b_a = BitList1024::from_raw_bytes(smallvec![0b1001, 0b0001], 16).unwrap(); + + assert_eq!(a.difference(&b), a_b); + assert_eq!(b.difference(&a), b_a); + } + + #[test] + fn shift_up() { + let mut a = BitList1024::from_raw_bytes(smallvec![0b1100_1111, 0b1101_0110], 16).unwrap(); + let mut b = BitList1024::from_raw_bytes(smallvec![0b1001_1110, 0b1010_1101], 16).unwrap(); + + a.shift_up(1).unwrap(); + assert_eq!(a, b); + a.shift_up(15).unwrap(); + assert!(a.is_zero()); + + b.shift_up(16).unwrap(); + assert!(b.is_zero()); + assert!(b.shift_up(17).is_err()); + } + + #[test] + fn num_set_bits() { + let a = BitList1024::from_raw_bytes(smallvec![0b1100, 0b0001], 16).unwrap(); + let b = BitList1024::from_raw_bytes(smallvec![0b1011, 0b1001], 16).unwrap(); + + assert_eq!(a.num_set_bits(), 3); + assert_eq!(b.num_set_bits(), 5); + } + + #[test] + fn iter() { + let mut bitfield = BitList1024::with_capacity(9).unwrap(); + bitfield.set(2, true).unwrap(); + bitfield.set(8, true).unwrap(); + + assert_eq!( + bitfield.iter().collect::>(), + vec![false, false, true, false, false, false, false, false, true] + ); + } + + #[test] + fn ssz_bytes_len() { + for i in 1..64 { + let mut bitfield = BitList1024::with_capacity(i).unwrap(); + for j in 0..i { + bitfield.set(j, true).expect("should set bit in bounds"); + } + let bytes = bitfield.as_ssz_bytes(); + assert_eq!(bitfield.ssz_bytes_len(), bytes.len(), "i = {}", i); + } + } + + // Ensure that the stack size of a BitList is manageable. + #[test] + fn size_of() { + assert_eq!(std::mem::size_of::(), SMALLVEC_LEN + 24); + } +} diff --git a/ssz/src/encode/impls.rs b/ssz/src/encode/impls.rs index c4bf0c0..b884e5b 100644 --- a/ssz/src/encode/impls.rs +++ b/ssz/src/encode/impls.rs @@ -522,7 +522,7 @@ mod tests { #[test] fn vec_of_u8() { let vec: Vec = vec![]; - assert_eq!(vec.as_ssz_bytes(), vec![]); + assert_eq!(vec.as_ssz_bytes(), Vec::::new()); let vec: Vec = vec![1]; assert_eq!(vec.as_ssz_bytes(), vec![1]); @@ -534,7 +534,7 @@ mod tests { #[test] fn vec_of_vec_of_u8() { let vec: Vec> = vec![]; - assert_eq!(vec.as_ssz_bytes(), vec![]); + assert_eq!(vec.as_ssz_bytes(), Vec::::new()); let vec: Vec> = vec![vec![]]; assert_eq!(vec.as_ssz_bytes(), vec![4, 0, 0, 0]); diff --git a/ssz/src/lib.rs b/ssz/src/lib.rs index e71157a..245c7bd 100644 --- a/ssz/src/lib.rs +++ b/ssz/src/lib.rs @@ -34,11 +34,13 @@ //! //! See `examples/` for manual implementations of the `Encode` and `Decode` traits. +mod bitfield; mod decode; mod encode; pub mod legacy; mod union_selector; +pub use bitfield::{BitList, BitVector}; pub use decode::{ impls::decode_list_of_variable_length_items, read_offset, split_union_bytes, try_from_iter::TryFromIter, Decode, DecodeError, SszDecoder, SszDecoderBuilder, From f088a3aa00af46b7caaa110f989d2472b0222a71 Mon Sep 17 00:00:00 2001 From: Mac L Date: Sat, 27 Jul 2024 01:42:29 +1000 Subject: [PATCH 2/6] Also export Fixed and Variable --- ssz/src/lib.rs | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/ssz/src/lib.rs b/ssz/src/lib.rs index 245c7bd..8b38018 100644 --- a/ssz/src/lib.rs +++ b/ssz/src/lib.rs @@ -40,7 +40,8 @@ mod encode; pub mod legacy; mod union_selector; -pub use bitfield::{BitList, BitVector}; +#[doc(hidden)] +pub use bitfield::{BitList, BitVector, Fixed, Variable}; pub use decode::{ impls::decode_list_of_variable_length_items, read_offset, split_union_bytes, try_from_iter::TryFromIter, Decode, DecodeError, SszDecoder, SszDecoderBuilder, From 16b8b1ddb3d1d8b355160e43a970e71b3854724f Mon Sep 17 00:00:00 2001 From: Mac L Date: Sat, 27 Jul 2024 01:47:03 +1000 Subject: [PATCH 3/6] Fix tests --- ssz/src/bitfield.rs | 9 ++++++--- 1 file changed, 6 insertions(+), 3 deletions(-) diff --git a/ssz/src/bitfield.rs b/ssz/src/bitfield.rs index 1003956..551a11f 100644 --- a/ssz/src/bitfield.rs +++ b/ssz/src/bitfield.rs @@ -80,7 +80,8 @@ pub type BitVector = Bitfield>; /// - `BitVector` is an alias for `Bitfield>` /// /// ``` -/// use ssz_types::{BitVector, BitList, typenum}; +/// use ssz::{BitVector, BitList}; +/// use typenum; /// /// // `BitList` has a type-level maximum length. The length of the list is specified at runtime /// // and it must be less than or equal to `N`. After instantiation, `BitList` cannot grow or @@ -152,8 +153,9 @@ impl Bitfield> { /// /// ## Example /// ``` - /// use ssz_types::{BitList, typenum}; + /// use ssz::BitList; /// use smallvec::SmallVec; + /// use typenum; /// /// type BitList8 = BitList; /// @@ -280,8 +282,9 @@ impl Bitfield> { /// /// ## Example /// ``` - /// use ssz_types::{BitVector, typenum}; + /// use ssz::BitVector; /// use smallvec::SmallVec; + /// use typenum; /// /// type BitVector4 = BitVector; /// From 6fea0d1e393e9b80161b9e3db0557238b08253a7 Mon Sep 17 00:00:00 2001 From: Mac L Date: Sat, 27 Jul 2024 01:51:22 +1000 Subject: [PATCH 4/6] Also export Bitfield --- ssz/src/lib.rs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/ssz/src/lib.rs b/ssz/src/lib.rs index 8b38018..9b7d777 100644 --- a/ssz/src/lib.rs +++ b/ssz/src/lib.rs @@ -41,7 +41,7 @@ pub mod legacy; mod union_selector; #[doc(hidden)] -pub use bitfield::{BitList, BitVector, Fixed, Variable}; +pub use bitfield::{Bitfield, BitList, BitVector, Fixed, Variable}; pub use decode::{ impls::decode_list_of_variable_length_items, read_offset, split_union_bytes, try_from_iter::TryFromIter, Decode, DecodeError, SszDecoder, SszDecoderBuilder, From 90804d4d7e0ec9935fd62f8e1c7097667b3a1dce Mon Sep 17 00:00:00 2001 From: Mac L Date: Sat, 27 Jul 2024 01:57:13 +1000 Subject: [PATCH 5/6] Formatting --- ssz/src/lib.rs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/ssz/src/lib.rs b/ssz/src/lib.rs index 9b7d777..66b16de 100644 --- a/ssz/src/lib.rs +++ b/ssz/src/lib.rs @@ -41,7 +41,7 @@ pub mod legacy; mod union_selector; #[doc(hidden)] -pub use bitfield::{Bitfield, BitList, BitVector, Fixed, Variable}; +pub use bitfield::{BitList, BitVector, Bitfield, Fixed, Variable}; pub use decode::{ impls::decode_list_of_variable_length_items, read_offset, split_union_bytes, try_from_iter::TryFromIter, Decode, DecodeError, SszDecoder, SszDecoderBuilder, From 91b51302357fbb257a2c4638cef2696f48619d6d Mon Sep 17 00:00:00 2001 From: Michael Sproul Date: Tue, 3 Sep 2024 14:43:27 +1000 Subject: [PATCH 6/6] Add `BitList::resize` --- ssz/src/bitfield.rs | 35 +++++++++++++++++++++++++++++++++++ 1 file changed, 35 insertions(+) diff --git a/ssz/src/bitfield.rs b/ssz/src/bitfield.rs index 551a11f..7bac6bd 100644 --- a/ssz/src/bitfield.rs +++ b/ssz/src/bitfield.rs @@ -257,6 +257,24 @@ impl Bitfield> { pub fn is_subset(&self, other: &Self) -> bool { self.difference(other).is_zero() } + + /// Returns a new BitList of length M, with the same bits set as `self`. + pub fn resize(&self) -> Result>, Error> { + if N::to_usize() > M::to_usize() { + return Err(Error::InvalidByteCount { + given: M::to_usize(), + expected: N::to_usize() + 1, + }); + } + + let mut resized = Bitfield::>::with_capacity(M::to_usize())?; + + for (i, bit) in self.iter().enumerate() { + resized.set(i, bit)?; + } + + Ok(resized) + } } impl Bitfield> { @@ -1394,4 +1412,21 @@ mod bitlist { fn size_of() { assert_eq!(std::mem::size_of::(), SMALLVEC_LEN + 24); } + + #[test] + fn resize() { + let mut bit_list = BitList1::with_capacity(1).unwrap(); + bit_list.set(0, true).unwrap(); + assert_eq!(bit_list.len(), 1); + assert_eq!(bit_list.num_set_bits(), 1); + assert_eq!(bit_list.highest_set_bit().unwrap(), 0); + + let resized_bit_list = bit_list.resize::().unwrap(); + assert_eq!(resized_bit_list.len(), 1024); + assert_eq!(resized_bit_list.num_set_bits(), 1); + assert_eq!(resized_bit_list.highest_set_bit().unwrap(), 0); + + // Can't extend a BitList to a smaller BitList + resized_bit_list.resize::().unwrap_err(); + } }