2019-04-15 23:14:33 +00:00
|
|
|
use super::*;
|
|
|
|
use hashing::hash;
|
|
|
|
use int_to_bytes::int_to_bytes32;
|
|
|
|
use ssz::ssz_encode;
|
|
|
|
|
|
|
|
pub trait TreeHash {
|
2019-04-15 23:34:23 +00:00
|
|
|
fn tree_hash_type() -> TreeHashType;
|
2019-04-15 23:14:33 +00:00
|
|
|
|
|
|
|
fn tree_hash_packed_encoding(&self) -> Vec<u8>;
|
|
|
|
|
2019-04-15 23:34:23 +00:00
|
|
|
fn tree_hash_packing_factor() -> usize;
|
|
|
|
|
|
|
|
fn tree_hash_root(&self) -> Vec<u8>;
|
2019-04-15 23:14:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl TreeHash for u64 {
|
2019-04-15 23:34:23 +00:00
|
|
|
fn tree_hash_type() -> TreeHashType {
|
|
|
|
TreeHashType::Basic
|
2019-04-15 23:14:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
fn tree_hash_packed_encoding(&self) -> Vec<u8> {
|
|
|
|
ssz_encode(self)
|
|
|
|
}
|
|
|
|
|
2019-04-15 23:34:23 +00:00
|
|
|
fn tree_hash_packing_factor() -> usize {
|
|
|
|
HASHSIZE / 8
|
|
|
|
}
|
|
|
|
|
|
|
|
fn tree_hash_root(&self) -> Vec<u8> {
|
2019-04-15 23:14:33 +00:00
|
|
|
int_to_bytes32(*self)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<T> TreeHash for Vec<T>
|
|
|
|
where
|
|
|
|
T: TreeHash,
|
|
|
|
{
|
2019-04-15 23:34:23 +00:00
|
|
|
fn tree_hash_type() -> TreeHashType {
|
|
|
|
TreeHashType::List
|
2019-04-15 23:14:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
fn tree_hash_packed_encoding(&self) -> Vec<u8> {
|
|
|
|
unreachable!("List should never be packed.")
|
|
|
|
}
|
|
|
|
|
2019-04-15 23:34:23 +00:00
|
|
|
fn tree_hash_packing_factor() -> usize {
|
|
|
|
unreachable!("List should never be packed.")
|
|
|
|
}
|
|
|
|
|
|
|
|
fn tree_hash_root(&self) -> Vec<u8> {
|
|
|
|
let leaves = match T::tree_hash_type() {
|
|
|
|
TreeHashType::Basic => {
|
|
|
|
let mut leaves =
|
|
|
|
Vec::with_capacity((HASHSIZE / T::tree_hash_packing_factor()) * self.len());
|
2019-04-15 23:14:33 +00:00
|
|
|
|
|
|
|
for item in self {
|
|
|
|
leaves.append(&mut item.tree_hash_packed_encoding());
|
|
|
|
}
|
|
|
|
|
|
|
|
leaves
|
|
|
|
}
|
2019-04-15 23:34:23 +00:00
|
|
|
TreeHashType::Composite | TreeHashType::List => {
|
2019-04-15 23:14:33 +00:00
|
|
|
let mut leaves = Vec::with_capacity(self.len() * HASHSIZE);
|
|
|
|
|
|
|
|
for item in self {
|
2019-04-15 23:34:23 +00:00
|
|
|
leaves.append(&mut item.tree_hash_root())
|
2019-04-15 23:14:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
leaves
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
// Mix in the length
|
|
|
|
let mut root_and_len = Vec::with_capacity(HASHSIZE * 2);
|
|
|
|
root_and_len.append(&mut efficient_merkleize(&leaves)[0..32].to_vec());
|
|
|
|
root_and_len.append(&mut int_to_bytes32(self.len() as u64));
|
|
|
|
|
|
|
|
hash(&root_and_len)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn efficient_merkleize(bytes: &[u8]) -> Vec<u8> {
|
|
|
|
let leaves = num_sanitized_leaves(bytes.len());
|
|
|
|
let nodes = num_nodes(leaves);
|
|
|
|
let internal_nodes = nodes - leaves;
|
|
|
|
|
|
|
|
let num_bytes = internal_nodes * HASHSIZE + bytes.len();
|
|
|
|
|
|
|
|
let mut o: Vec<u8> = vec![0; internal_nodes * HASHSIZE];
|
|
|
|
o.append(&mut bytes.to_vec());
|
|
|
|
|
|
|
|
assert_eq!(o.len(), num_bytes);
|
|
|
|
|
|
|
|
let empty_chunk_hash = hash(&[0; MERKLE_HASH_CHUNCK]);
|
|
|
|
|
|
|
|
let mut i = nodes * HASHSIZE;
|
|
|
|
let mut j = internal_nodes * HASHSIZE;
|
|
|
|
|
|
|
|
while i >= MERKLE_HASH_CHUNCK {
|
|
|
|
i -= MERKLE_HASH_CHUNCK;
|
|
|
|
|
|
|
|
j -= HASHSIZE;
|
|
|
|
let hash = match o.get(i..i + MERKLE_HASH_CHUNCK) {
|
|
|
|
// All bytes are available, hash as ususal.
|
|
|
|
Some(slice) => hash(slice),
|
|
|
|
// Unable to get all the bytes.
|
|
|
|
None => {
|
|
|
|
match o.get(i..) {
|
|
|
|
// Able to get some of the bytes, pad them out.
|
|
|
|
Some(slice) => {
|
|
|
|
let mut bytes = slice.to_vec();
|
|
|
|
bytes.resize(MERKLE_HASH_CHUNCK, 0);
|
|
|
|
hash(&bytes)
|
|
|
|
}
|
|
|
|
// Unable to get any bytes, use the empty-chunk hash.
|
|
|
|
None => empty_chunk_hash.clone(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
o[j..j + HASHSIZE].copy_from_slice(&hash);
|
|
|
|
}
|
|
|
|
|
|
|
|
o
|
|
|
|
}
|