use vstd::prelude::*;
verus! {
spec fn sqlite_page_size() -> nat {
16_384
}
spec fn segment_page_count() -> nat {
256
}
spec fn page_table_entry_len() -> nat {
8
}
spec fn u64_max() -> nat {
18_446_744_073_709_551_615
}
spec fn page_count_for_size(size: nat) -> nat {
if size == 0 {
0
} else {
(((size - 1) / (sqlite_page_size() as int)) + 1) as nat
}
}
spec fn segment_count_for_pages(page_count: nat) -> nat {
if page_count == 0 {
0
} else {
(((page_count - 1) / (segment_page_count() as int)) + 1) as nat
}
}
spec fn segment_no(page_no: nat) -> nat {
page_no / segment_page_count()
}
spec fn segment_index(page_no: nat) -> nat {
page_no % segment_page_count()
}
spec fn root_table_bytes(entry_count: nat) -> nat {
entry_count * page_table_entry_len()
}
spec fn import_offset_fits(base: nat, offset: nat) -> bool {
base + offset <= u64_max()
}
proof fn page_count_zero()
ensures
page_count_for_size(0) == 0,
{
}
proof fn page_count_within_first_page(size: nat)
by (nonlinear_arith)
requires
0 < size <= sqlite_page_size(),
ensures
page_count_for_size(size) == 1,
{
}
proof fn segment_count_zero()
ensures
segment_count_for_pages(0) == 0,
{
}
proof fn segment_index_is_bounded(page_no: nat)
by (nonlinear_arith)
ensures
segment_index(page_no) < segment_page_count(),
{
}
proof fn root_table_bytes_scales_by_entry_count(entry_count: nat)
ensures
root_table_bytes(entry_count) == entry_count * 8,
{
}
proof fn import_offset_at_u64_max_fits()
ensures
import_offset_fits(u64_max(), 0),
{
}
proof fn import_offset_past_u64_max_overflows()
by (nonlinear_arith)
ensures
!import_offset_fits(u64_max(), 1),
{
}
}