2019-04-15 01:37:29 +00:00
|
|
|
use super::*;
|
|
|
|
|
2019-04-21 02:12:47 +00:00
|
|
|
#[derive(Debug, PartialEq, Clone)]
|
2019-04-15 01:37:29 +00:00
|
|
|
pub struct BTreeOverlay {
|
2019-04-21 02:12:47 +00:00
|
|
|
pub offset: usize,
|
2019-04-23 23:29:32 +00:00
|
|
|
pub depth: usize,
|
2019-04-21 23:20:13 +00:00
|
|
|
pub num_items: usize,
|
|
|
|
pub lengths: Vec<usize>,
|
2019-04-15 01:37:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl BTreeOverlay {
|
2019-04-23 23:29:32 +00:00
|
|
|
pub fn new<T>(item: &T, initial_offset: usize, depth: usize) -> Result<Self, Error>
|
2019-04-15 01:37:29 +00:00
|
|
|
where
|
2019-04-24 08:23:58 +00:00
|
|
|
T: CachedTreeHash<T>,
|
2019-04-15 01:37:29 +00:00
|
|
|
{
|
2019-04-23 23:29:32 +00:00
|
|
|
item.tree_hash_cache_overlay(initial_offset, depth)
|
2019-04-15 01:37:29 +00:00
|
|
|
}
|
|
|
|
|
2019-04-21 23:20:13 +00:00
|
|
|
pub fn from_lengths(
|
|
|
|
offset: usize,
|
|
|
|
num_items: usize,
|
2019-04-23 23:29:32 +00:00
|
|
|
depth: usize,
|
2019-04-21 23:20:13 +00:00
|
|
|
lengths: Vec<usize>,
|
|
|
|
) -> Result<Self, Error> {
|
2019-04-21 02:12:47 +00:00
|
|
|
if lengths.is_empty() {
|
|
|
|
Err(Error::TreeCannotHaveZeroNodes)
|
2019-04-15 01:37:29 +00:00
|
|
|
} else {
|
2019-04-21 23:20:13 +00:00
|
|
|
Ok(Self {
|
|
|
|
offset,
|
|
|
|
num_items,
|
2019-04-23 23:29:32 +00:00
|
|
|
depth,
|
2019-04-21 23:20:13 +00:00
|
|
|
lengths,
|
|
|
|
})
|
2019-04-21 02:12:47 +00:00
|
|
|
}
|
|
|
|
}
|
2019-04-15 01:37:29 +00:00
|
|
|
|
2019-04-21 02:12:47 +00:00
|
|
|
pub fn num_leaf_nodes(&self) -> usize {
|
|
|
|
self.lengths.len().next_power_of_two()
|
|
|
|
}
|
2019-04-15 01:37:29 +00:00
|
|
|
|
2019-04-22 06:09:29 +00:00
|
|
|
pub fn num_padding_leaves(&self) -> usize {
|
2019-04-21 02:12:47 +00:00
|
|
|
self.num_leaf_nodes() - self.lengths.len()
|
|
|
|
}
|
2019-04-15 01:37:29 +00:00
|
|
|
|
2019-04-24 00:17:05 +00:00
|
|
|
/// Returns the number of nodes in the tree.
|
|
|
|
///
|
|
|
|
/// Note: this is distinct from `num_chunks`, which returns the total number of chunks in
|
|
|
|
/// this tree.
|
2019-04-21 02:12:47 +00:00
|
|
|
pub fn num_nodes(&self) -> usize {
|
|
|
|
2 * self.num_leaf_nodes() - 1
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn num_internal_nodes(&self) -> usize {
|
|
|
|
self.num_leaf_nodes() - 1
|
|
|
|
}
|
2019-04-15 01:37:29 +00:00
|
|
|
|
2019-04-21 02:12:47 +00:00
|
|
|
fn first_node(&self) -> usize {
|
|
|
|
self.offset
|
2019-04-15 01:37:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn root(&self) -> usize {
|
2019-04-21 02:12:47 +00:00
|
|
|
self.first_node()
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn next_node(&self) -> usize {
|
2019-04-21 23:20:13 +00:00
|
|
|
self.first_node() + self.num_internal_nodes() + self.num_leaf_nodes() - self.lengths.len()
|
|
|
|
+ self.lengths.iter().sum::<usize>()
|
2019-04-15 01:37:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn height(&self) -> usize {
|
2019-04-21 02:12:47 +00:00
|
|
|
self.num_leaf_nodes().trailing_zeros() as usize
|
2019-04-15 01:37:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn chunk_range(&self) -> Range<usize> {
|
2019-04-21 02:12:47 +00:00
|
|
|
self.first_node()..self.next_node()
|
2019-04-15 01:37:29 +00:00
|
|
|
}
|
|
|
|
|
2019-04-24 00:17:05 +00:00
|
|
|
/// Returns the number of chunks inside this tree (including subtrees).
|
|
|
|
///
|
|
|
|
/// Note: this is distinct from `num_nodes` which returns the number of nodes in the binary
|
|
|
|
/// tree.
|
|
|
|
pub fn num_chunks(&self) -> usize {
|
2019-04-21 02:12:47 +00:00
|
|
|
self.next_node() - self.first_node()
|
2019-04-15 01:37:29 +00:00
|
|
|
}
|
|
|
|
|
2019-04-21 02:12:47 +00:00
|
|
|
pub fn first_leaf_node(&self) -> usize {
|
|
|
|
self.offset + self.num_internal_nodes()
|
2019-04-15 01:37:29 +00:00
|
|
|
}
|
|
|
|
|
2019-04-22 06:09:29 +00:00
|
|
|
/// Returns the chunk-range for a given leaf node.
|
|
|
|
///
|
|
|
|
/// Returns `None` if:
|
|
|
|
/// - The specified node is internal.
|
|
|
|
/// - The specified node is padding.
|
|
|
|
/// - The specified node is OOB of the tree.
|
2019-04-21 02:12:47 +00:00
|
|
|
pub fn get_leaf_node(&self, i: usize) -> Result<Option<Range<usize>>, Error> {
|
2019-04-22 06:09:29 +00:00
|
|
|
if i >= self.num_nodes() - self.num_padding_leaves() {
|
|
|
|
Ok(None)
|
|
|
|
} else if (i == self.num_internal_nodes()) && (self.num_items == 0) {
|
|
|
|
// If this is the first leaf node and the overlay contains zero items, return `None` as
|
|
|
|
// this node must be padding.
|
2019-04-21 02:12:47 +00:00
|
|
|
Ok(None)
|
|
|
|
} else {
|
2019-04-22 06:09:29 +00:00
|
|
|
let i = i - self.num_internal_nodes();
|
|
|
|
|
|
|
|
let first_node = self.offset
|
|
|
|
+ self.num_internal_nodes()
|
|
|
|
+ self.lengths.iter().take(i).sum::<usize>();
|
2019-04-21 02:12:47 +00:00
|
|
|
let last_node = first_node + self.lengths[i];
|
2019-04-22 06:09:29 +00:00
|
|
|
|
2019-04-21 02:12:47 +00:00
|
|
|
Ok(Some(first_node..last_node))
|
|
|
|
}
|
2019-04-15 01:37:29 +00:00
|
|
|
}
|
|
|
|
|
2019-04-21 23:20:13 +00:00
|
|
|
pub fn child_chunks(&self, parent: usize) -> (usize, usize) {
|
|
|
|
let children = children(parent);
|
|
|
|
|
|
|
|
if children.1 < self.num_internal_nodes() {
|
|
|
|
(children.0 + self.offset, children.1 + self.offset)
|
|
|
|
} else {
|
|
|
|
let chunks = self.n_leaf_node_chunks(children.1);
|
|
|
|
(chunks[chunks.len() - 2], chunks[chunks.len() - 1])
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// (parent, (left_child, right_child))
|
2019-04-21 02:12:47 +00:00
|
|
|
pub fn internal_parents_and_children(&self) -> Vec<(usize, (usize, usize))> {
|
2019-04-21 23:20:13 +00:00
|
|
|
let mut chunks = Vec::with_capacity(self.num_nodes());
|
|
|
|
chunks.append(&mut self.internal_node_chunks());
|
|
|
|
chunks.append(&mut self.leaf_node_chunks());
|
|
|
|
|
2019-04-21 02:12:47 +00:00
|
|
|
(0..self.num_internal_nodes())
|
|
|
|
.into_iter()
|
|
|
|
.map(|parent| {
|
|
|
|
let children = children(parent);
|
2019-04-21 23:20:13 +00:00
|
|
|
(chunks[parent], (chunks[children.0], chunks[children.1]))
|
2019-04-21 02:12:47 +00:00
|
|
|
})
|
|
|
|
.collect()
|
2019-04-15 01:37:29 +00:00
|
|
|
}
|
|
|
|
|
2019-04-21 02:12:47 +00:00
|
|
|
// Returns a `Vec` of chunk indices for each internal node of the tree.
|
|
|
|
pub fn internal_node_chunks(&self) -> Vec<usize> {
|
|
|
|
(self.offset..self.offset + self.num_internal_nodes()).collect()
|
2019-04-15 01:37:29 +00:00
|
|
|
}
|
2019-04-21 23:20:13 +00:00
|
|
|
|
|
|
|
// Returns a `Vec` of the first chunk index for each leaf node of the tree.
|
|
|
|
pub fn leaf_node_chunks(&self) -> Vec<usize> {
|
|
|
|
self.n_leaf_node_chunks(self.num_leaf_nodes())
|
|
|
|
}
|
|
|
|
|
|
|
|
// Returns a `Vec` of the first chunk index for the first `n` leaf nodes of the tree.
|
|
|
|
fn n_leaf_node_chunks(&self, n: usize) -> Vec<usize> {
|
|
|
|
let mut chunks = Vec::with_capacity(n);
|
|
|
|
|
|
|
|
let mut chunk = self.offset + self.num_internal_nodes();
|
|
|
|
for i in 0..n {
|
|
|
|
chunks.push(chunk);
|
|
|
|
|
|
|
|
match self.lengths.get(i) {
|
|
|
|
Some(len) => {
|
|
|
|
chunk += len;
|
|
|
|
}
|
|
|
|
None => chunk += 1,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
chunks
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-04-25 23:55:03 +00:00
|
|
|
fn children(parent: usize) -> (usize, usize) {
|
|
|
|
((2 * parent + 1), (2 * parent + 2))
|
|
|
|
}
|
|
|
|
|
2019-04-21 23:20:13 +00:00
|
|
|
#[cfg(test)]
|
|
|
|
mod test {
|
|
|
|
use super::*;
|
|
|
|
|
|
|
|
fn get_tree_a(n: usize) -> BTreeOverlay {
|
2019-04-23 23:29:32 +00:00
|
|
|
BTreeOverlay::from_lengths(0, n, 0, vec![1; n]).unwrap()
|
2019-04-21 23:20:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn leaf_node_chunks() {
|
|
|
|
let tree = get_tree_a(4);
|
|
|
|
|
|
|
|
assert_eq!(tree.leaf_node_chunks(), vec![3, 4, 5, 6])
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn internal_node_chunks() {
|
|
|
|
let tree = get_tree_a(4);
|
|
|
|
|
|
|
|
assert_eq!(tree.internal_node_chunks(), vec![0, 1, 2])
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn internal_parents_and_children() {
|
|
|
|
let tree = get_tree_a(4);
|
|
|
|
|
|
|
|
assert_eq!(
|
|
|
|
tree.internal_parents_and_children(),
|
|
|
|
vec![(0, (1, 2)), (1, (3, 4)), (2, (5, 6))]
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn chunk_range() {
|
|
|
|
let tree = get_tree_a(4);
|
|
|
|
assert_eq!(tree.chunk_range(), 0..7);
|
|
|
|
|
|
|
|
let tree = get_tree_a(1);
|
|
|
|
assert_eq!(tree.chunk_range(), 0..1);
|
|
|
|
|
|
|
|
let tree = get_tree_a(2);
|
|
|
|
assert_eq!(tree.chunk_range(), 0..3);
|
|
|
|
|
2019-04-23 23:29:32 +00:00
|
|
|
let tree = BTreeOverlay::from_lengths(11, 4, 0, vec![1, 1]).unwrap();
|
2019-04-21 23:20:13 +00:00
|
|
|
assert_eq!(tree.chunk_range(), 11..14);
|
|
|
|
}
|
|
|
|
|
2019-04-22 06:09:29 +00:00
|
|
|
#[test]
|
|
|
|
fn get_leaf_node() {
|
|
|
|
let tree = get_tree_a(4);
|
|
|
|
|
|
|
|
assert_eq!(tree.get_leaf_node(3), Ok(Some(3..4)));
|
|
|
|
assert_eq!(tree.get_leaf_node(4), Ok(Some(4..5)));
|
|
|
|
assert_eq!(tree.get_leaf_node(5), Ok(Some(5..6)));
|
|
|
|
assert_eq!(tree.get_leaf_node(6), Ok(Some(6..7)));
|
|
|
|
}
|
|
|
|
|
2019-04-21 23:20:13 +00:00
|
|
|
#[test]
|
|
|
|
fn root_of_one_node() {
|
|
|
|
let tree = get_tree_a(1);
|
|
|
|
|
|
|
|
assert_eq!(tree.root(), 0);
|
|
|
|
assert_eq!(tree.num_internal_nodes(), 0);
|
|
|
|
assert_eq!(tree.num_leaf_nodes(), 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn child_chunks() {
|
|
|
|
let tree = get_tree_a(4);
|
|
|
|
|
|
|
|
assert_eq!(tree.child_chunks(0), (1, 2))
|
|
|
|
}
|
2019-04-15 01:37:29 +00:00
|
|
|
}
|