Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

refactor PathIterator::check_validity; add tests #642

Open
wants to merge 3 commits into
base: main
Choose a base branch
from
Open
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
81 changes: 58 additions & 23 deletions storage/src/mmr/iterator.rs
Original file line number Diff line number Diff line change
Expand Up @@ -49,33 +49,32 @@
///
/// The implementation verifies that peaks in the MMR of the given size have strictly decreasing
/// height, which is a necessary condition for MMR validity.
pub(crate) fn check_validity(size: u64) -> bool {
if size == 0 {
pub(crate) fn check_validity(mut size: u64) -> bool {
if size <= 1 {
return true;
}
let start = u64::MAX >> size.leading_zeros();
let mut two_h = 1 << start.trailing_ones();
let mut node_pos = start - 1;
while two_h > 1 {
if node_pos < size {
if two_h == 2 {
// If this peak is a leaf yet there are more nodes remaining, then this MMR is
// invalid.
return node_pos == size - 1;
}
// move to the right sibling
node_pos += two_h - 1;
if node_pos < size {
// If the right sibling is in the MMR, then it is invalid.
return false;
}
continue;

// Consider each mountain from left to right.
// For each, subtract its size from `size`.
// We should find that the size eventually reaches 0
// (the last mountain has more than 1 element) or 1
// (the last mountain has 1 element).
// If the size never reaches 0, there are "left over"
// nodes and there isn't an MMR with the given `size`.

// Height of the root of the smallest perfect binary
// tree containing `size` (leaf is height 0)
let mut height = 63 - size.leading_zeros();
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

While this is more clear in isolation, given the rest of the iterator implementations maintain 2^(height+1) as "two_h" instead of converting height directly with each iteration, it might be best to stick with that & save a few cycles of computation each iteration.

while height > 0 && size > 1 {
// Subtract the size of the next mountain.
// If it's bigger than the remaining size, it's not in this range.
let mountain_size = (1 << (height + 1)) - 1;
if size >= mountain_size {
size -= mountain_size;
Copy link
Collaborator

@roberto-bayardo roberto-bayardo Mar 22, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

why not instead return immediately if mountain_size > size? [Edit: Oh does this mean there's no "peak" with the current height and hence no mountain of this size actually exists? If so please add a comment to the effect.]

Ah I see I didn't interpret "range" in your comment as "mountain range" but rather a more generic "range of values"! Perhaps reword your comment as:

// Subtract the size of the next mountain if it's part of this mountain range (mountain_size < size)

}
// descend to the left child
two_h >>= 1;
node_pos -= two_h;
height -= 1;
}
true
size <= 1
}
}

Expand Down Expand Up @@ -364,4 +363,40 @@
last_leaf_pos = *leaf_pos;
}
}

#[test]
fn test_check_validity() {
// Test cases for check_validity function
let valid_sizes = vec![
0, 1, 3, 4, 7, 8, 10, 11, 15, 16, 18, 19, 22, 23, 25, 26, 31, 32, 34, 35, 38, 39, 41,
42, 46, 47, 49, 50, 53, 54, 56, 57, 63, 64, 127, 128, 255, 256, 511, 512, 1023, 1024,
2047, 2048, 4095, 4096,
];
let invalid_sizes = vec![
2, 5, 6, 9, 12, 13, 14, 17, 20, 21, 24, 27, 28, 29, 30, 36, 37, 40, 43, 44, 45, 48, 51,
52, 55, 58, 65, 129, 257, 513,
];

for size in valid_sizes {
assert!(
PeakIterator::check_validity(size),
"Expected validity for size {}",

Check warning on line 383 in storage/src/mmr/iterator.rs

View check run for this annotation

Codecov / codecov/patch

storage/src/mmr/iterator.rs#L383

Added line #L383 was not covered by tests
size
);
}

for size in invalid_sizes {
assert!(
!PeakIterator::check_validity(size),
"Expected invalidity for size {}",

Check warning on line 391 in storage/src/mmr/iterator.rs

View check run for this annotation

Codecov / codecov/patch

storage/src/mmr/iterator.rs#L391

Added line #L391 was not covered by tests
size
);
}

for i in 2..63 {
assert!(PeakIterator::check_validity((1 << i) - 1));
assert!(PeakIterator::check_validity(1 << i));
assert!(!PeakIterator::check_validity((1 << i) + 1));
}
}
}