Use SmallVec in Bitfield (#3025)

## Issue Addressed

Alternative to #2935

## Proposed Changes

Replace the `Vec<u8>` inside `Bitfield` with a `SmallVec<[u8; 32>`. This eliminates heap allocations for attestation bitfields until we reach 500K validators, at which point we can consider increasing `SMALLVEC_LEN` to 40 or 48.

While running Lighthouse under `heaptrack` I found that SSZ encoding and decoding of bitfields corresponded to 22% of all allocations by count. I've confirmed that with this change applied those allocations disappear entirely.

## Additional Info

We can win another 8 bytes of space by using `smallvec`'s [`union` feature](https://docs.rs/smallvec/1.8.0/smallvec/#union), although I might leave that for a future PR because I don't know how experimental that feature is and whether it uses some spicy `unsafe` blocks.
This commit is contained in:
Michael Sproul 2022-02-17 23:55:04 +00:00
parent 0a6a8ea3b0
commit da4ca024f1
6 changed files with 174 additions and 110 deletions

2
Cargo.lock generated
View File

@ -1678,6 +1678,7 @@ dependencies = [
"serde", "serde",
"serde_derive", "serde_derive",
"serde_json", "serde_json",
"smallvec",
"tree_hash", "tree_hash",
"tree_hash_derive 0.4.0 (registry+https://github.com/rust-lang/crates.io-index)", "tree_hash_derive 0.4.0 (registry+https://github.com/rust-lang/crates.io-index)",
"typenum", "typenum",
@ -6496,6 +6497,7 @@ dependencies = [
"serde_json", "serde_json",
"serde_yaml", "serde_yaml",
"slog", "slog",
"smallvec",
"state_processing", "state_processing",
"superstruct", "superstruct",
"swap_or_not_shuffle", "swap_or_not_shuffle",

View File

@ -18,6 +18,7 @@ eth2_ssz = "0.4.1"
typenum = "1.12.0" typenum = "1.12.0"
arbitrary = { version = "1.0", features = ["derive"], optional = true } arbitrary = { version = "1.0", features = ["derive"], optional = true }
derivative = "2.1.1" derivative = "2.1.1"
smallvec = "1.8.0"
[dev-dependencies] [dev-dependencies]
serde_json = "1.0.58" serde_json = "1.0.58"

View File

@ -5,10 +5,17 @@ use derivative::Derivative;
use eth2_serde_utils::hex::{encode as hex_encode, PrefixedHexVisitor}; use eth2_serde_utils::hex::{encode as hex_encode, PrefixedHexVisitor};
use serde::de::{Deserialize, Deserializer}; use serde::de::{Deserialize, Deserializer};
use serde::ser::{Serialize, Serializer}; use serde::ser::{Serialize, Serializer};
use smallvec::{smallvec, SmallVec, ToSmallVec};
use ssz::{Decode, Encode}; use ssz::{Decode, Encode};
use tree_hash::Hash256; use tree_hash::Hash256;
use typenum::Unsigned; use typenum::Unsigned;
/// 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`. /// A marker trait applied to `Variable` and `Fixed` that defines the behaviour of a `Bitfield`.
pub trait BitfieldBehaviour: Clone {} pub trait BitfieldBehaviour: Clone {}
@ -87,11 +94,11 @@ pub type BitVector<N> = Bitfield<Fixed<N>>;
/// ///
/// The internal representation of the bitfield is the same as that required by SSZ. The lowest /// 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 /// byte (by `Vec` index) stores the lowest bit-indices and the right-most bit stores the lowest
/// bit-index. E.g., `vec![0b0000_0001, 0b0000_0010]` has bits `0, 9` set. /// bit-index. E.g., `smallvec![0b0000_0001, 0b0000_0010]` has bits `0, 9` set.
#[derive(Clone, Debug, Derivative)] #[derive(Clone, Debug, Derivative)]
#[derivative(PartialEq, Hash(bound = ""))] #[derivative(PartialEq, Hash(bound = ""))]
pub struct Bitfield<T> { pub struct Bitfield<T> {
bytes: Vec<u8>, bytes: SmallVec<[u8; SMALLVEC_LEN]>,
len: usize, len: usize,
_phantom: PhantomData<T>, _phantom: PhantomData<T>,
} }
@ -106,7 +113,7 @@ impl<N: Unsigned + Clone> Bitfield<Variable<N>> {
pub fn with_capacity(num_bits: usize) -> Result<Self, Error> { pub fn with_capacity(num_bits: usize) -> Result<Self, Error> {
if num_bits <= N::to_usize() { if num_bits <= N::to_usize() {
Ok(Self { Ok(Self {
bytes: vec![0; bytes_for_bit_len(num_bits)], bytes: smallvec![0; bytes_for_bit_len(num_bits)],
len: num_bits, len: num_bits,
_phantom: PhantomData, _phantom: PhantomData,
}) })
@ -131,14 +138,15 @@ impl<N: Unsigned + Clone> Bitfield<Variable<N>> {
/// ## Example /// ## Example
/// ``` /// ```
/// use ssz_types::{BitList, typenum}; /// use ssz_types::{BitList, typenum};
/// use smallvec::SmallVec;
/// ///
/// type BitList8 = BitList<typenum::U8>; /// type BitList8 = BitList<typenum::U8>;
/// ///
/// let b = BitList8::with_capacity(4).unwrap(); /// let b = BitList8::with_capacity(4).unwrap();
/// ///
/// assert_eq!(b.into_bytes(), vec![0b0001_0000]); /// assert_eq!(b.into_bytes(), SmallVec::from_buf([0b0001_0000]));
/// ``` /// ```
pub fn into_bytes(self) -> Vec<u8> { pub fn into_bytes(self) -> SmallVec<[u8; SMALLVEC_LEN]> {
let len = self.len(); let len = self.len();
let mut bytes = self.bytes; let mut bytes = self.bytes;
@ -163,7 +171,7 @@ impl<N: Unsigned + Clone> Bitfield<Variable<N>> {
/// produces (SSZ). /// produces (SSZ).
/// ///
/// Returns `None` if `bytes` are not a valid encoding. /// Returns `None` if `bytes` are not a valid encoding.
pub fn from_bytes(bytes: Vec<u8>) -> Result<Self, Error> { pub fn from_bytes(bytes: SmallVec<[u8; SMALLVEC_LEN]>) -> Result<Self, Error> {
let bytes_len = bytes.len(); let bytes_len = bytes.len();
let mut initial_bitfield: Bitfield<Variable<N>> = { let mut initial_bitfield: Bitfield<Variable<N>> = {
let num_bits = bytes.len() * 8; let num_bits = bytes.len() * 8;
@ -235,7 +243,7 @@ impl<N: Unsigned + Clone> Bitfield<Fixed<N>> {
/// All bits are initialized to `false`. /// All bits are initialized to `false`.
pub fn new() -> Self { pub fn new() -> Self {
Self { Self {
bytes: vec![0; bytes_for_bit_len(Self::capacity())], bytes: smallvec![0; bytes_for_bit_len(Self::capacity())],
len: Self::capacity(), len: Self::capacity(),
_phantom: PhantomData, _phantom: PhantomData,
} }
@ -253,12 +261,13 @@ impl<N: Unsigned + Clone> Bitfield<Fixed<N>> {
/// ## Example /// ## Example
/// ``` /// ```
/// use ssz_types::{BitVector, typenum}; /// use ssz_types::{BitVector, typenum};
/// use smallvec::SmallVec;
/// ///
/// type BitVector4 = BitVector<typenum::U4>; /// type BitVector4 = BitVector<typenum::U4>;
/// ///
/// assert_eq!(BitVector4::new().into_bytes(), vec![0b0000_0000]); /// assert_eq!(BitVector4::new().into_bytes(), SmallVec::from_buf([0b0000_0000]));
/// ``` /// ```
pub fn into_bytes(self) -> Vec<u8> { pub fn into_bytes(self) -> SmallVec<[u8; SMALLVEC_LEN]> {
self.into_raw_bytes() self.into_raw_bytes()
} }
@ -266,7 +275,7 @@ impl<N: Unsigned + Clone> Bitfield<Fixed<N>> {
/// produces (SSZ). /// produces (SSZ).
/// ///
/// Returns `None` if `bytes` are not a valid encoding. /// Returns `None` if `bytes` are not a valid encoding.
pub fn from_bytes(bytes: Vec<u8>) -> Result<Self, Error> { pub fn from_bytes(bytes: SmallVec<[u8; SMALLVEC_LEN]>) -> Result<Self, Error> {
Self::from_raw_bytes(bytes, Self::capacity()) Self::from_raw_bytes(bytes, Self::capacity())
} }
@ -355,7 +364,7 @@ impl<T: BitfieldBehaviour> Bitfield<T> {
} }
/// Returns the underlying bytes representation of the bitfield. /// Returns the underlying bytes representation of the bitfield.
pub fn into_raw_bytes(self) -> Vec<u8> { pub fn into_raw_bytes(self) -> SmallVec<[u8; SMALLVEC_LEN]> {
self.bytes self.bytes
} }
@ -372,9 +381,9 @@ impl<T: BitfieldBehaviour> Bitfield<T> {
/// - `bytes` is not the minimal required bytes to represent a bitfield of `bit_len` bits. /// - `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 /// - `bit_len` is not a multiple of 8 and `bytes` contains set bits that are higher than, or
/// equal to `bit_len`. /// equal to `bit_len`.
fn from_raw_bytes(bytes: Vec<u8>, bit_len: usize) -> Result<Self, Error> { fn from_raw_bytes(bytes: SmallVec<[u8; SMALLVEC_LEN]>, bit_len: usize) -> Result<Self, Error> {
if bit_len == 0 { if bit_len == 0 {
if bytes.len() == 1 && bytes == [0] { if bytes.len() == 1 && bytes[0] == 0 {
// A bitfield with `bit_len` 0 can only be represented by a single zero byte. // A bitfield with `bit_len` 0 can only be represented by a single zero byte.
Ok(Self { Ok(Self {
bytes, bytes,
@ -512,7 +521,7 @@ impl<N: Unsigned + Clone> Encode for Bitfield<Variable<N>> {
} }
fn ssz_append(&self, buf: &mut Vec<u8>) { fn ssz_append(&self, buf: &mut Vec<u8>) {
buf.append(&mut self.clone().into_bytes()) buf.extend_from_slice(&self.clone().into_bytes())
} }
} }
@ -522,7 +531,7 @@ impl<N: Unsigned + Clone> Decode for Bitfield<Variable<N>> {
} }
fn from_ssz_bytes(bytes: &[u8]) -> Result<Self, ssz::DecodeError> { fn from_ssz_bytes(bytes: &[u8]) -> Result<Self, ssz::DecodeError> {
Self::from_bytes(bytes.to_vec()).map_err(|e| { Self::from_bytes(bytes.to_smallvec()).map_err(|e| {
ssz::DecodeError::BytesInvalid(format!("BitList failed to decode: {:?}", e)) ssz::DecodeError::BytesInvalid(format!("BitList failed to decode: {:?}", e))
}) })
} }
@ -542,7 +551,7 @@ impl<N: Unsigned + Clone> Encode for Bitfield<Fixed<N>> {
} }
fn ssz_append(&self, buf: &mut Vec<u8>) { fn ssz_append(&self, buf: &mut Vec<u8>) {
buf.append(&mut self.clone().into_bytes()) buf.extend_from_slice(&self.clone().into_bytes())
} }
} }
@ -556,7 +565,7 @@ impl<N: Unsigned + Clone> Decode for Bitfield<Fixed<N>> {
} }
fn from_ssz_bytes(bytes: &[u8]) -> Result<Self, ssz::DecodeError> { fn from_ssz_bytes(bytes: &[u8]) -> Result<Self, ssz::DecodeError> {
Self::from_bytes(bytes.to_vec()).map_err(|e| { Self::from_bytes(bytes.to_smallvec()).map_err(|e| {
ssz::DecodeError::BytesInvalid(format!("BitVector failed to decode: {:?}", e)) ssz::DecodeError::BytesInvalid(format!("BitVector failed to decode: {:?}", e))
}) })
} }
@ -649,7 +658,7 @@ impl<N: Unsigned + Clone> tree_hash::TreeHash for Bitfield<Fixed<N>> {
impl<N: 'static + Unsigned> arbitrary::Arbitrary<'_> for Bitfield<Fixed<N>> { impl<N: 'static + Unsigned> arbitrary::Arbitrary<'_> for Bitfield<Fixed<N>> {
fn arbitrary(u: &mut arbitrary::Unstructured<'_>) -> arbitrary::Result<Self> { fn arbitrary(u: &mut arbitrary::Unstructured<'_>) -> arbitrary::Result<Self> {
let size = N::to_usize(); let size = N::to_usize();
let mut vec: Vec<u8> = vec![0u8; size]; let mut vec = smallvec![0u8; size];
u.fill_buffer(&mut vec)?; u.fill_buffer(&mut vec)?;
Ok(Self::from_bytes(vec).map_err(|_| arbitrary::Error::IncorrectFormat)?) Ok(Self::from_bytes(vec).map_err(|_| arbitrary::Error::IncorrectFormat)?)
} }
@ -661,7 +670,7 @@ impl<N: 'static + Unsigned> arbitrary::Arbitrary<'_> for Bitfield<Variable<N>> {
let max_size = N::to_usize(); let max_size = N::to_usize();
let rand = usize::arbitrary(u)?; let rand = usize::arbitrary(u)?;
let size = std::cmp::min(rand, max_size); let size = std::cmp::min(rand, max_size);
let mut vec: Vec<u8> = vec![0u8; size]; let mut vec = smallvec![0u8; size];
u.fill_buffer(&mut vec)?; u.fill_buffer(&mut vec)?;
Ok(Self::from_bytes(vec).map_err(|_| arbitrary::Error::IncorrectFormat)?) Ok(Self::from_bytes(vec).map_err(|_| arbitrary::Error::IncorrectFormat)?)
} }
@ -730,9 +739,9 @@ mod bitvector {
#[test] #[test]
fn intersection() { fn intersection() {
let a = BitVector16::from_raw_bytes(vec![0b1100, 0b0001], 16).unwrap(); let a = BitVector16::from_raw_bytes(smallvec![0b1100, 0b0001], 16).unwrap();
let b = BitVector16::from_raw_bytes(vec![0b1011, 0b1001], 16).unwrap(); let b = BitVector16::from_raw_bytes(smallvec![0b1011, 0b1001], 16).unwrap();
let c = BitVector16::from_raw_bytes(vec![0b1000, 0b0001], 16).unwrap(); let c = BitVector16::from_raw_bytes(smallvec![0b1000, 0b0001], 16).unwrap();
assert_eq!(a.intersection(&b), c); assert_eq!(a.intersection(&b), c);
assert_eq!(b.intersection(&a), c); assert_eq!(b.intersection(&a), c);
@ -745,9 +754,9 @@ mod bitvector {
#[test] #[test]
fn intersection_diff_length() { fn intersection_diff_length() {
let a = BitVector16::from_bytes(vec![0b0010_1110, 0b0010_1011]).unwrap(); let a = BitVector16::from_bytes(smallvec![0b0010_1110, 0b0010_1011]).unwrap();
let b = BitVector16::from_bytes(vec![0b0010_1101, 0b0000_0001]).unwrap(); let b = BitVector16::from_bytes(smallvec![0b0010_1101, 0b0000_0001]).unwrap();
let c = BitVector16::from_bytes(vec![0b0010_1100, 0b0000_0001]).unwrap(); let c = BitVector16::from_bytes(smallvec![0b0010_1100, 0b0000_0001]).unwrap();
assert_eq!(a.len(), 16); assert_eq!(a.len(), 16);
assert_eq!(b.len(), 16); assert_eq!(b.len(), 16);
@ -758,9 +767,9 @@ mod bitvector {
#[test] #[test]
fn union() { fn union() {
let a = BitVector16::from_raw_bytes(vec![0b1100, 0b0001], 16).unwrap(); let a = BitVector16::from_raw_bytes(smallvec![0b1100, 0b0001], 16).unwrap();
let b = BitVector16::from_raw_bytes(vec![0b1011, 0b1001], 16).unwrap(); let b = BitVector16::from_raw_bytes(smallvec![0b1011, 0b1001], 16).unwrap();
let c = BitVector16::from_raw_bytes(vec![0b1111, 0b1001], 16).unwrap(); let c = BitVector16::from_raw_bytes(smallvec![0b1111, 0b1001], 16).unwrap();
assert_eq!(a.union(&b), c); assert_eq!(a.union(&b), c);
assert_eq!(b.union(&a), c); assert_eq!(b.union(&a), c);
@ -771,9 +780,9 @@ mod bitvector {
#[test] #[test]
fn union_diff_length() { fn union_diff_length() {
let a = BitVector16::from_bytes(vec![0b0010_1011, 0b0010_1110]).unwrap(); let a = BitVector16::from_bytes(smallvec![0b0010_1011, 0b0010_1110]).unwrap();
let b = BitVector16::from_bytes(vec![0b0000_0001, 0b0010_1101]).unwrap(); let b = BitVector16::from_bytes(smallvec![0b0000_0001, 0b0010_1101]).unwrap();
let c = BitVector16::from_bytes(vec![0b0010_1011, 0b0010_1111]).unwrap(); let c = BitVector16::from_bytes(smallvec![0b0010_1011, 0b0010_1111]).unwrap();
assert_eq!(a.len(), c.len()); assert_eq!(a.len(), c.len());
assert_eq!(a.union(&b), c); assert_eq!(a.union(&b), c);
@ -839,6 +848,12 @@ mod bitvector {
assert!(BitVector4::from_ssz_bytes(&bad).is_err()); 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::<BitVector64>(), SMALLVEC_LEN + 24);
}
} }
#[cfg(test)] #[cfg(test)]
@ -992,50 +1007,50 @@ mod bitlist {
#[test] #[test]
fn from_raw_bytes() { fn from_raw_bytes() {
assert!(BitList1024::from_raw_bytes(vec![0b0000_0000], 0).is_ok()); assert!(BitList1024::from_raw_bytes(smallvec![0b0000_0000], 0).is_ok());
assert!(BitList1024::from_raw_bytes(vec![0b0000_0001], 1).is_ok()); assert!(BitList1024::from_raw_bytes(smallvec![0b0000_0001], 1).is_ok());
assert!(BitList1024::from_raw_bytes(vec![0b0000_0011], 2).is_ok()); assert!(BitList1024::from_raw_bytes(smallvec![0b0000_0011], 2).is_ok());
assert!(BitList1024::from_raw_bytes(vec![0b0000_0111], 3).is_ok()); assert!(BitList1024::from_raw_bytes(smallvec![0b0000_0111], 3).is_ok());
assert!(BitList1024::from_raw_bytes(vec![0b0000_1111], 4).is_ok()); assert!(BitList1024::from_raw_bytes(smallvec![0b0000_1111], 4).is_ok());
assert!(BitList1024::from_raw_bytes(vec![0b0001_1111], 5).is_ok()); assert!(BitList1024::from_raw_bytes(smallvec![0b0001_1111], 5).is_ok());
assert!(BitList1024::from_raw_bytes(vec![0b0011_1111], 6).is_ok()); assert!(BitList1024::from_raw_bytes(smallvec![0b0011_1111], 6).is_ok());
assert!(BitList1024::from_raw_bytes(vec![0b0111_1111], 7).is_ok()); assert!(BitList1024::from_raw_bytes(smallvec![0b0111_1111], 7).is_ok());
assert!(BitList1024::from_raw_bytes(vec![0b1111_1111], 8).is_ok()); assert!(BitList1024::from_raw_bytes(smallvec![0b1111_1111], 8).is_ok());
assert!(BitList1024::from_raw_bytes(vec![0b1111_1111, 0b0000_0001], 9).is_ok()); assert!(BitList1024::from_raw_bytes(smallvec![0b1111_1111, 0b0000_0001], 9).is_ok());
assert!(BitList1024::from_raw_bytes(vec![0b1111_1111, 0b0000_0011], 10).is_ok()); assert!(BitList1024::from_raw_bytes(smallvec![0b1111_1111, 0b0000_0011], 10).is_ok());
assert!(BitList1024::from_raw_bytes(vec![0b1111_1111, 0b0000_0111], 11).is_ok()); assert!(BitList1024::from_raw_bytes(smallvec![0b1111_1111, 0b0000_0111], 11).is_ok());
assert!(BitList1024::from_raw_bytes(vec![0b1111_1111, 0b0000_1111], 12).is_ok()); assert!(BitList1024::from_raw_bytes(smallvec![0b1111_1111, 0b0000_1111], 12).is_ok());
assert!(BitList1024::from_raw_bytes(vec![0b1111_1111, 0b0001_1111], 13).is_ok()); assert!(BitList1024::from_raw_bytes(smallvec![0b1111_1111, 0b0001_1111], 13).is_ok());
assert!(BitList1024::from_raw_bytes(vec![0b1111_1111, 0b0011_1111], 14).is_ok()); assert!(BitList1024::from_raw_bytes(smallvec![0b1111_1111, 0b0011_1111], 14).is_ok());
assert!(BitList1024::from_raw_bytes(vec![0b1111_1111, 0b0111_1111], 15).is_ok()); assert!(BitList1024::from_raw_bytes(smallvec![0b1111_1111, 0b0111_1111], 15).is_ok());
assert!(BitList1024::from_raw_bytes(vec![0b1111_1111, 0b1111_1111], 16).is_ok()); assert!(BitList1024::from_raw_bytes(smallvec![0b1111_1111, 0b1111_1111], 16).is_ok());
for i in 0..8 { for i in 0..8 {
assert!(BitList1024::from_raw_bytes(vec![], i).is_err()); assert!(BitList1024::from_raw_bytes(smallvec![], i).is_err());
assert!(BitList1024::from_raw_bytes(vec![0b1111_1111], i).is_err()); assert!(BitList1024::from_raw_bytes(smallvec![0b1111_1111], i).is_err());
assert!(BitList1024::from_raw_bytes(vec![0b0000_0000, 0b1111_1110], i).is_err()); assert!(BitList1024::from_raw_bytes(smallvec![0b0000_0000, 0b1111_1110], i).is_err());
} }
assert!(BitList1024::from_raw_bytes(vec![0b0000_0001], 0).is_err()); assert!(BitList1024::from_raw_bytes(smallvec![0b0000_0001], 0).is_err());
assert!(BitList1024::from_raw_bytes(vec![0b0000_0001], 0).is_err()); assert!(BitList1024::from_raw_bytes(smallvec![0b0000_0001], 0).is_err());
assert!(BitList1024::from_raw_bytes(vec![0b0000_0011], 1).is_err()); assert!(BitList1024::from_raw_bytes(smallvec![0b0000_0011], 1).is_err());
assert!(BitList1024::from_raw_bytes(vec![0b0000_0111], 2).is_err()); assert!(BitList1024::from_raw_bytes(smallvec![0b0000_0111], 2).is_err());
assert!(BitList1024::from_raw_bytes(vec![0b0000_1111], 3).is_err()); assert!(BitList1024::from_raw_bytes(smallvec![0b0000_1111], 3).is_err());
assert!(BitList1024::from_raw_bytes(vec![0b0001_1111], 4).is_err()); assert!(BitList1024::from_raw_bytes(smallvec![0b0001_1111], 4).is_err());
assert!(BitList1024::from_raw_bytes(vec![0b0011_1111], 5).is_err()); assert!(BitList1024::from_raw_bytes(smallvec![0b0011_1111], 5).is_err());
assert!(BitList1024::from_raw_bytes(vec![0b0111_1111], 6).is_err()); assert!(BitList1024::from_raw_bytes(smallvec![0b0111_1111], 6).is_err());
assert!(BitList1024::from_raw_bytes(vec![0b1111_1111], 7).is_err()); assert!(BitList1024::from_raw_bytes(smallvec![0b1111_1111], 7).is_err());
assert!(BitList1024::from_raw_bytes(vec![0b1111_1111, 0b0000_0001], 8).is_err()); assert!(BitList1024::from_raw_bytes(smallvec![0b1111_1111, 0b0000_0001], 8).is_err());
assert!(BitList1024::from_raw_bytes(vec![0b1111_1111, 0b0000_0011], 9).is_err()); assert!(BitList1024::from_raw_bytes(smallvec![0b1111_1111, 0b0000_0011], 9).is_err());
assert!(BitList1024::from_raw_bytes(vec![0b1111_1111, 0b0000_0111], 10).is_err()); assert!(BitList1024::from_raw_bytes(smallvec![0b1111_1111, 0b0000_0111], 10).is_err());
assert!(BitList1024::from_raw_bytes(vec![0b1111_1111, 0b0000_1111], 11).is_err()); assert!(BitList1024::from_raw_bytes(smallvec![0b1111_1111, 0b0000_1111], 11).is_err());
assert!(BitList1024::from_raw_bytes(vec![0b1111_1111, 0b0001_1111], 12).is_err()); assert!(BitList1024::from_raw_bytes(smallvec![0b1111_1111, 0b0001_1111], 12).is_err());
assert!(BitList1024::from_raw_bytes(vec![0b1111_1111, 0b0011_1111], 13).is_err()); assert!(BitList1024::from_raw_bytes(smallvec![0b1111_1111, 0b0011_1111], 13).is_err());
assert!(BitList1024::from_raw_bytes(vec![0b1111_1111, 0b0111_1111], 14).is_err()); assert!(BitList1024::from_raw_bytes(smallvec![0b1111_1111, 0b0111_1111], 14).is_err());
assert!(BitList1024::from_raw_bytes(vec![0b1111_1111, 0b1111_1111], 15).is_err()); assert!(BitList1024::from_raw_bytes(smallvec![0b1111_1111, 0b1111_1111], 15).is_err());
} }
fn test_set_unset(num_bits: usize) { fn test_set_unset(num_bits: usize) {
@ -1083,51 +1098,64 @@ mod bitlist {
} }
} }
/// Type-specialised `smallvec` macro for testing.
macro_rules! bytevec {
($($x : expr),* $(,)*) => {
{
let __smallvec: SmallVec<[u8; SMALLVEC_LEN]> = smallvec!($($x),*);
__smallvec
}
};
}
#[test] #[test]
fn into_raw_bytes() { fn into_raw_bytes() {
let mut bitfield = BitList1024::with_capacity(9).unwrap(); let mut bitfield = BitList1024::with_capacity(9).unwrap();
bitfield.set(0, true).unwrap(); bitfield.set(0, true).unwrap();
assert_eq!( assert_eq!(
bitfield.clone().into_raw_bytes(), bitfield.clone().into_raw_bytes(),
vec![0b0000_0001, 0b0000_0000] bytevec![0b0000_0001, 0b0000_0000]
); );
bitfield.set(1, true).unwrap(); bitfield.set(1, true).unwrap();
assert_eq!( assert_eq!(
bitfield.clone().into_raw_bytes(), bitfield.clone().into_raw_bytes(),
vec![0b0000_0011, 0b0000_0000] bytevec![0b0000_0011, 0b0000_0000]
); );
bitfield.set(2, true).unwrap(); bitfield.set(2, true).unwrap();
assert_eq!( assert_eq!(
bitfield.clone().into_raw_bytes(), bitfield.clone().into_raw_bytes(),
vec![0b0000_0111, 0b0000_0000] bytevec![0b0000_0111, 0b0000_0000]
); );
bitfield.set(3, true).unwrap(); bitfield.set(3, true).unwrap();
assert_eq!( assert_eq!(
bitfield.clone().into_raw_bytes(), bitfield.clone().into_raw_bytes(),
vec![0b0000_1111, 0b0000_0000] bytevec![0b0000_1111, 0b0000_0000]
); );
bitfield.set(4, true).unwrap(); bitfield.set(4, true).unwrap();
assert_eq!( assert_eq!(
bitfield.clone().into_raw_bytes(), bitfield.clone().into_raw_bytes(),
vec![0b0001_1111, 0b0000_0000] bytevec![0b0001_1111, 0b0000_0000]
); );
bitfield.set(5, true).unwrap(); bitfield.set(5, true).unwrap();
assert_eq!( assert_eq!(
bitfield.clone().into_raw_bytes(), bitfield.clone().into_raw_bytes(),
vec![0b0011_1111, 0b0000_0000] bytevec![0b0011_1111, 0b0000_0000]
); );
bitfield.set(6, true).unwrap(); bitfield.set(6, true).unwrap();
assert_eq!( assert_eq!(
bitfield.clone().into_raw_bytes(), bitfield.clone().into_raw_bytes(),
vec![0b0111_1111, 0b0000_0000] bytevec![0b0111_1111, 0b0000_0000]
); );
bitfield.set(7, true).unwrap(); bitfield.set(7, true).unwrap();
assert_eq!( assert_eq!(
bitfield.clone().into_raw_bytes(), bitfield.clone().into_raw_bytes(),
vec![0b1111_1111, 0b0000_0000] bytevec![0b1111_1111, 0b0000_0000]
); );
bitfield.set(8, true).unwrap(); bitfield.set(8, true).unwrap();
assert_eq!(bitfield.into_raw_bytes(), vec![0b1111_1111, 0b0000_0001]); assert_eq!(
bitfield.into_raw_bytes(),
bytevec![0b1111_1111, 0b0000_0001]
);
} }
#[test] #[test]
@ -1138,28 +1166,28 @@ mod bitlist {
); );
assert_eq!( assert_eq!(
BitList1024::from_raw_bytes(vec![0b0000_0001, 0b0000_0000], 16) BitList1024::from_raw_bytes(smallvec![0b0000_0001, 0b0000_0000], 16)
.unwrap() .unwrap()
.highest_set_bit(), .highest_set_bit(),
Some(0) Some(0)
); );
assert_eq!( assert_eq!(
BitList1024::from_raw_bytes(vec![0b0000_0010, 0b0000_0000], 16) BitList1024::from_raw_bytes(smallvec![0b0000_0010, 0b0000_0000], 16)
.unwrap() .unwrap()
.highest_set_bit(), .highest_set_bit(),
Some(1) Some(1)
); );
assert_eq!( assert_eq!(
BitList1024::from_raw_bytes(vec![0b0000_1000], 8) BitList1024::from_raw_bytes(smallvec![0b0000_1000], 8)
.unwrap() .unwrap()
.highest_set_bit(), .highest_set_bit(),
Some(3) Some(3)
); );
assert_eq!( assert_eq!(
BitList1024::from_raw_bytes(vec![0b0000_0000, 0b1000_0000], 16) BitList1024::from_raw_bytes(smallvec![0b0000_0000, 0b1000_0000], 16)
.unwrap() .unwrap()
.highest_set_bit(), .highest_set_bit(),
Some(15) Some(15)
@ -1168,9 +1196,9 @@ mod bitlist {
#[test] #[test]
fn intersection() { fn intersection() {
let a = BitList1024::from_raw_bytes(vec![0b1100, 0b0001], 16).unwrap(); let a = BitList1024::from_raw_bytes(smallvec![0b1100, 0b0001], 16).unwrap();
let b = BitList1024::from_raw_bytes(vec![0b1011, 0b1001], 16).unwrap(); let b = BitList1024::from_raw_bytes(smallvec![0b1011, 0b1001], 16).unwrap();
let c = BitList1024::from_raw_bytes(vec![0b1000, 0b0001], 16).unwrap(); let c = BitList1024::from_raw_bytes(smallvec![0b1000, 0b0001], 16).unwrap();
assert_eq!(a.intersection(&b), c); assert_eq!(a.intersection(&b), c);
assert_eq!(b.intersection(&a), c); assert_eq!(b.intersection(&a), c);
@ -1183,10 +1211,10 @@ mod bitlist {
#[test] #[test]
fn intersection_diff_length() { fn intersection_diff_length() {
let a = BitList1024::from_bytes(vec![0b0010_1110, 0b0010_1011]).unwrap(); let a = BitList1024::from_bytes(smallvec![0b0010_1110, 0b0010_1011]).unwrap();
let b = BitList1024::from_bytes(vec![0b0010_1101, 0b0000_0001]).unwrap(); let b = BitList1024::from_bytes(smallvec![0b0010_1101, 0b0000_0001]).unwrap();
let c = BitList1024::from_bytes(vec![0b0010_1100, 0b0000_0001]).unwrap(); let c = BitList1024::from_bytes(smallvec![0b0010_1100, 0b0000_0001]).unwrap();
let d = BitList1024::from_bytes(vec![0b0010_1110, 0b1111_1111, 0b1111_1111]).unwrap(); let d = BitList1024::from_bytes(smallvec![0b0010_1110, 0b1111_1111, 0b1111_1111]).unwrap();
assert_eq!(a.len(), 13); assert_eq!(a.len(), 13);
assert_eq!(b.len(), 8); assert_eq!(b.len(), 8);
@ -1200,9 +1228,9 @@ mod bitlist {
#[test] #[test]
fn union() { fn union() {
let a = BitList1024::from_raw_bytes(vec![0b1100, 0b0001], 16).unwrap(); let a = BitList1024::from_raw_bytes(smallvec![0b1100, 0b0001], 16).unwrap();
let b = BitList1024::from_raw_bytes(vec![0b1011, 0b1001], 16).unwrap(); let b = BitList1024::from_raw_bytes(smallvec![0b1011, 0b1001], 16).unwrap();
let c = BitList1024::from_raw_bytes(vec![0b1111, 0b1001], 16).unwrap(); let c = BitList1024::from_raw_bytes(smallvec![0b1111, 0b1001], 16).unwrap();
assert_eq!(a.union(&b), c); assert_eq!(a.union(&b), c);
assert_eq!(b.union(&a), c); assert_eq!(b.union(&a), c);
@ -1213,10 +1241,10 @@ mod bitlist {
#[test] #[test]
fn union_diff_length() { fn union_diff_length() {
let a = BitList1024::from_bytes(vec![0b0010_1011, 0b0010_1110]).unwrap(); let a = BitList1024::from_bytes(smallvec![0b0010_1011, 0b0010_1110]).unwrap();
let b = BitList1024::from_bytes(vec![0b0000_0001, 0b0010_1101]).unwrap(); let b = BitList1024::from_bytes(smallvec![0b0000_0001, 0b0010_1101]).unwrap();
let c = BitList1024::from_bytes(vec![0b0010_1011, 0b0010_1111]).unwrap(); let c = BitList1024::from_bytes(smallvec![0b0010_1011, 0b0010_1111]).unwrap();
let d = BitList1024::from_bytes(vec![0b0010_1011, 0b1011_1110, 0b1000_1101]).unwrap(); let d = BitList1024::from_bytes(smallvec![0b0010_1011, 0b1011_1110, 0b1000_1101]).unwrap();
assert_eq!(a.len(), c.len()); assert_eq!(a.len(), c.len());
assert_eq!(a.union(&b), c); assert_eq!(a.union(&b), c);
@ -1227,10 +1255,10 @@ mod bitlist {
#[test] #[test]
fn difference() { fn difference() {
let a = BitList1024::from_raw_bytes(vec![0b1100, 0b0001], 16).unwrap(); let a = BitList1024::from_raw_bytes(smallvec![0b1100, 0b0001], 16).unwrap();
let b = BitList1024::from_raw_bytes(vec![0b1011, 0b1001], 16).unwrap(); let b = BitList1024::from_raw_bytes(smallvec![0b1011, 0b1001], 16).unwrap();
let a_b = BitList1024::from_raw_bytes(vec![0b0100, 0b0000], 16).unwrap(); let a_b = BitList1024::from_raw_bytes(smallvec![0b0100, 0b0000], 16).unwrap();
let b_a = BitList1024::from_raw_bytes(vec![0b0011, 0b1000], 16).unwrap(); let b_a = BitList1024::from_raw_bytes(smallvec![0b0011, 0b1000], 16).unwrap();
assert_eq!(a.difference(&b), a_b); assert_eq!(a.difference(&b), a_b);
assert_eq!(b.difference(&a), b_a); assert_eq!(b.difference(&a), b_a);
@ -1239,10 +1267,10 @@ mod bitlist {
#[test] #[test]
fn difference_diff_length() { fn difference_diff_length() {
let a = BitList1024::from_raw_bytes(vec![0b0110, 0b1100, 0b0011], 24).unwrap(); let a = BitList1024::from_raw_bytes(smallvec![0b0110, 0b1100, 0b0011], 24).unwrap();
let b = BitList1024::from_raw_bytes(vec![0b1011, 0b1001], 16).unwrap(); let b = BitList1024::from_raw_bytes(smallvec![0b1011, 0b1001], 16).unwrap();
let a_b = BitList1024::from_raw_bytes(vec![0b0100, 0b0100, 0b0011], 24).unwrap(); let a_b = BitList1024::from_raw_bytes(smallvec![0b0100, 0b0100, 0b0011], 24).unwrap();
let b_a = BitList1024::from_raw_bytes(vec![0b1001, 0b0001], 16).unwrap(); let b_a = BitList1024::from_raw_bytes(smallvec![0b1001, 0b0001], 16).unwrap();
assert_eq!(a.difference(&b), a_b); assert_eq!(a.difference(&b), a_b);
assert_eq!(b.difference(&a), b_a); assert_eq!(b.difference(&a), b_a);
@ -1250,8 +1278,8 @@ mod bitlist {
#[test] #[test]
fn shift_up() { fn shift_up() {
let mut a = BitList1024::from_raw_bytes(vec![0b1100_1111, 0b1101_0110], 16).unwrap(); let mut a = BitList1024::from_raw_bytes(smallvec![0b1100_1111, 0b1101_0110], 16).unwrap();
let mut b = BitList1024::from_raw_bytes(vec![0b1001_1110, 0b1010_1101], 16).unwrap(); let mut b = BitList1024::from_raw_bytes(smallvec![0b1001_1110, 0b1010_1101], 16).unwrap();
a.shift_up(1).unwrap(); a.shift_up(1).unwrap();
assert_eq!(a, b); assert_eq!(a, b);
@ -1265,8 +1293,8 @@ mod bitlist {
#[test] #[test]
fn num_set_bits() { fn num_set_bits() {
let a = BitList1024::from_raw_bytes(vec![0b1100, 0b0001], 16).unwrap(); let a = BitList1024::from_raw_bytes(smallvec![0b1100, 0b0001], 16).unwrap();
let b = BitList1024::from_raw_bytes(vec![0b1011, 0b1001], 16).unwrap(); let b = BitList1024::from_raw_bytes(smallvec![0b1011, 0b1001], 16).unwrap();
assert_eq!(a.num_set_bits(), 3); assert_eq!(a.num_set_bits(), 3);
assert_eq!(b.num_set_bits(), 5); assert_eq!(b.num_set_bits(), 5);
@ -1295,4 +1323,10 @@ mod bitlist {
assert_eq!(bitfield.ssz_bytes_len(), bytes.len(), "i = {}", i); 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::<BitList1024>(), SMALLVEC_LEN + 24);
}
} }

View File

@ -45,6 +45,7 @@ parking_lot = "0.11.1"
itertools = "0.10.0" itertools = "0.10.0"
superstruct = "0.4.0" superstruct = "0.4.0"
serde_json = "1.0.74" serde_json = "1.0.74"
smallvec = "1.8.0"
[dev-dependencies] [dev-dependencies]
criterion = "0.3.3" criterion = "0.3.3"

View File

@ -110,9 +110,34 @@ impl<T: EthSpec> SlotData for Attestation<T> {
#[cfg(test)] #[cfg(test)]
mod tests { mod tests {
use super::*;
use crate::*; use crate::*;
use super::*; // Check the in-memory size of an `Attestation`, which is useful for reasoning about memory
// and preventing regressions.
//
// This test will only pass with `blst`, if we run these tests with Milagro or another
// BLS library in future we will have to make it generic.
#[test]
fn size_of() {
use std::mem::size_of;
let aggregation_bits =
size_of::<BitList<<MainnetEthSpec as EthSpec>::MaxValidatorsPerCommittee>>();
let attestation_data = size_of::<AttestationData>();
let signature = size_of::<AggregateSignature>();
assert_eq!(aggregation_bits, 56);
assert_eq!(attestation_data, 128);
assert_eq!(signature, 288 + 16);
let attestation_expected = aggregation_bits + attestation_data + signature;
assert_eq!(attestation_expected, 488);
assert_eq!(
size_of::<Attestation<MainnetEthSpec>>(),
attestation_expected
);
}
ssz_and_tree_hash_tests!(Attestation<MainnetEthSpec>); ssz_and_tree_hash_tests!(Attestation<MainnetEthSpec>);
} }

View File

@ -1,9 +1,10 @@
use super::*; use super::*;
use crate::{BitList, BitVector, Unsigned}; use crate::{BitList, BitVector, Unsigned};
use smallvec::smallvec;
impl<N: Unsigned + Clone> TestRandom for BitList<N> { impl<N: Unsigned + Clone> TestRandom for BitList<N> {
fn random_for_test(rng: &mut impl RngCore) -> Self { fn random_for_test(rng: &mut impl RngCore) -> Self {
let mut raw_bytes = vec![0; std::cmp::max(1, (N::to_usize() + 7) / 8)]; let mut raw_bytes = smallvec![0; std::cmp::max(1, (N::to_usize() + 7) / 8)];
rng.fill_bytes(&mut raw_bytes); rng.fill_bytes(&mut raw_bytes);
Self::from_bytes(raw_bytes).expect("we generate a valid BitList") Self::from_bytes(raw_bytes).expect("we generate a valid BitList")
} }
@ -11,7 +12,7 @@ impl<N: Unsigned + Clone> TestRandom for BitList<N> {
impl<N: Unsigned + Clone> TestRandom for BitVector<N> { impl<N: Unsigned + Clone> TestRandom for BitVector<N> {
fn random_for_test(rng: &mut impl RngCore) -> Self { fn random_for_test(rng: &mut impl RngCore) -> Self {
let mut raw_bytes = vec![0; std::cmp::max(1, (N::to_usize() + 7) / 8)]; let mut raw_bytes = smallvec![0; std::cmp::max(1, (N::to_usize() + 7) / 8)];
rng.fill_bytes(&mut raw_bytes); rng.fill_bytes(&mut raw_bytes);
Self::from_bytes(raw_bytes).expect("we generate a valid BitVector") Self::from_bytes(raw_bytes).expect("we generate a valid BitVector")
} }