mirror of
https://github.com/aljazceru/turso.git
synced 2025-12-27 04:54:21 +01:00
`balance_non_root` should be as close as possible to `balance_non_root` in SQLite. This commits extract `balance_non_root` from `balance` and renames `balance_leaf` to `balance` as it enables future work on a complete `balance_non_root` procedure.
2012 lines
86 KiB
Rust
2012 lines
86 KiB
Rust
use log::debug;
|
|
|
|
use crate::storage::pager::Pager;
|
|
use crate::storage::sqlite3_ondisk::{
|
|
read_btree_cell, read_varint, write_varint, BTreeCell, DatabaseHeader, PageContent, PageType,
|
|
TableInteriorCell, TableLeafCell,
|
|
};
|
|
use crate::types::{Cursor, CursorResult, OwnedRecord, OwnedValue, SeekKey, SeekOp};
|
|
use crate::Result;
|
|
|
|
use std::cell::{Ref, RefCell};
|
|
use std::pin::Pin;
|
|
use std::rc::Rc;
|
|
|
|
use super::pager::PageRef;
|
|
use super::sqlite3_ondisk::{
|
|
write_varint_to_vec, IndexInteriorCell, IndexLeafCell, OverflowCell, DATABASE_HEADER_SIZE,
|
|
};
|
|
|
|
/*
|
|
These are offsets of fields in the header of a b-tree page.
|
|
*/
|
|
|
|
/// type of btree page -> u8
|
|
const PAGE_HEADER_OFFSET_PAGE_TYPE: usize = 0;
|
|
/// pointer to first freeblock -> u16
|
|
/// The second field of the b-tree page header is the offset of the first freeblock, or zero if there are no freeblocks on the page.
|
|
/// A freeblock is a structure used to identify unallocated space within a b-tree page.
|
|
/// Freeblocks are organized as a chain.
|
|
///
|
|
/// To be clear, freeblocks do not mean the regular unallocated free space to the left of the cell content area pointer, but instead
|
|
/// blocks of at least 4 bytes WITHIN the cell content area that are not in use due to e.g. deletions.
|
|
const PAGE_HEADER_OFFSET_FIRST_FREEBLOCK: usize = 1;
|
|
/// number of cells in the page -> u16
|
|
const PAGE_HEADER_OFFSET_CELL_COUNT: usize = 3;
|
|
/// pointer to first byte of cell allocated content from top -> u16
|
|
/// SQLite strives to place cells as far toward the end of the b-tree page as it can,
|
|
/// in order to leave space for future growth of the cell pointer array.
|
|
/// = the cell content area pointer moves leftward as cells are added to the page
|
|
const PAGE_HEADER_OFFSET_CELL_CONTENT_AREA: usize = 5;
|
|
/// number of fragmented bytes -> u8
|
|
/// Fragments are isolated groups of 1, 2, or 3 unused bytes within the cell content area.
|
|
const PAGE_HEADER_OFFSET_FRAGMENTED_BYTES_COUNT: usize = 7;
|
|
/// if internalnode, pointer right most pointer (saved separately from cells) -> u32
|
|
const PAGE_HEADER_OFFSET_RIGHTMOST_PTR: usize = 8;
|
|
|
|
/// Maximum depth of an SQLite B-Tree structure. Any B-Tree deeper than
|
|
/// this will be declared corrupt. This value is calculated based on a
|
|
/// maximum database size of 2^31 pages a minimum fanout of 2 for a
|
|
/// root-node and 3 for all other internal nodes.
|
|
///
|
|
/// If a tree that appears to be taller than this is encountered, it is
|
|
/// assumed that the database is corrupt.
|
|
pub const BTCURSOR_MAX_DEPTH: usize = 20;
|
|
|
|
/// Evaluate a Result<CursorResult<T>>, if IO return IO.
|
|
macro_rules! return_if_io {
|
|
($expr:expr) => {
|
|
match $expr? {
|
|
CursorResult::Ok(v) => v,
|
|
CursorResult::IO => return Ok(CursorResult::IO),
|
|
}
|
|
};
|
|
}
|
|
|
|
/// Check if the page is unlocked, if not return IO.
|
|
macro_rules! return_if_locked {
|
|
($expr:expr) => {{
|
|
if $expr.is_locked() {
|
|
return Ok(CursorResult::IO);
|
|
}
|
|
}};
|
|
}
|
|
|
|
/// State machine of a write operation.
|
|
/// May involve balancing due to overflow.
|
|
#[derive(Debug)]
|
|
enum WriteState {
|
|
Start,
|
|
BalanceStart,
|
|
BalanceNonRoot,
|
|
BalanceGetParentPage,
|
|
BalanceMoveUp,
|
|
Finish,
|
|
}
|
|
|
|
struct WriteInfo {
|
|
/// State of the write operation state machine.
|
|
state: WriteState,
|
|
/// Pages allocated during the write operation due to balancing.
|
|
new_pages: RefCell<Vec<PageRef>>,
|
|
/// Scratch space used during balancing.
|
|
scratch_cells: RefCell<Vec<&'static [u8]>>,
|
|
/// Bookkeeping of the rightmost pointer so the PAGE_HEADER_OFFSET_RIGHTMOST_PTR can be updated.
|
|
rightmost_pointer: RefCell<Option<u32>>,
|
|
/// Copy of the current page needed for buffer references.
|
|
page_copy: RefCell<Option<PageContent>>,
|
|
}
|
|
|
|
pub struct BTreeCursor {
|
|
pager: Rc<Pager>,
|
|
/// Page id of the root page used to go back up fast.
|
|
root_page: usize,
|
|
/// Rowid and record are stored before being consumed.
|
|
rowid: RefCell<Option<u64>>,
|
|
record: RefCell<Option<OwnedRecord>>,
|
|
null_flag: bool,
|
|
database_header: Rc<RefCell<DatabaseHeader>>,
|
|
/// Index internal pages are consumed on the way up, so we store going upwards flag in case
|
|
/// we just moved to a parent page and the parent page is an internal index page which requires
|
|
/// to be consumed.
|
|
going_upwards: bool,
|
|
/// Write information kept in case of write yields due to I/O. Needs to be stored somewhere
|
|
/// right :).
|
|
write_info: WriteInfo,
|
|
/// Page stack used to traverse the btree.
|
|
/// Each cursor has a stack because each cursor traverses the btree independently.
|
|
stack: PageStack,
|
|
}
|
|
|
|
/// Stack of pages representing the tree traversal order.
|
|
/// current_page represents the current page being used in the tree and current_page - 1 would be
|
|
/// the parent. Using current_page + 1 or higher is undefined behaviour.
|
|
struct PageStack {
|
|
/// Pointer to the currenet page being consumed
|
|
current_page: RefCell<i32>,
|
|
/// List of pages in the stack. Root page will be in index 0
|
|
stack: RefCell<[Option<PageRef>; BTCURSOR_MAX_DEPTH + 1]>,
|
|
/// List of cell indices in the stack.
|
|
/// cell_indices[current_page] is the current cell index being consumed. Similarly
|
|
/// cell_indices[current_page-1] is the cell index of the parent of the current page
|
|
/// that we save in case of going back up.
|
|
/// There are two points that need special attention:
|
|
/// If cell_indices[current_page] = -1, it indicates that the current iteration has reached the start of the current_page
|
|
/// If cell_indices[current_page] = `cell_count`, it means that the current iteration has reached the end of the current_page
|
|
cell_indices: RefCell<[i32; BTCURSOR_MAX_DEPTH + 1]>,
|
|
}
|
|
|
|
impl BTreeCursor {
|
|
pub fn new(
|
|
pager: Rc<Pager>,
|
|
root_page: usize,
|
|
database_header: Rc<RefCell<DatabaseHeader>>,
|
|
) -> Self {
|
|
Self {
|
|
pager,
|
|
root_page,
|
|
rowid: RefCell::new(None),
|
|
record: RefCell::new(None),
|
|
null_flag: false,
|
|
database_header,
|
|
going_upwards: false,
|
|
write_info: WriteInfo {
|
|
state: WriteState::Start,
|
|
new_pages: RefCell::new(Vec::with_capacity(4)),
|
|
scratch_cells: RefCell::new(Vec::new()),
|
|
rightmost_pointer: RefCell::new(None),
|
|
page_copy: RefCell::new(None),
|
|
},
|
|
stack: PageStack {
|
|
current_page: RefCell::new(-1),
|
|
cell_indices: RefCell::new([0; BTCURSOR_MAX_DEPTH + 1]),
|
|
stack: RefCell::new([const { None }; BTCURSOR_MAX_DEPTH + 1]),
|
|
},
|
|
}
|
|
}
|
|
|
|
/// Check if the table is empty.
|
|
/// This is done by checking if the root page has no cells.
|
|
fn is_empty_table(&mut self) -> Result<CursorResult<bool>> {
|
|
let page = self.pager.read_page(self.root_page)?;
|
|
return_if_locked!(page);
|
|
|
|
let cell_count = page.get().contents.as_ref().unwrap().cell_count();
|
|
Ok(CursorResult::Ok(cell_count == 0))
|
|
}
|
|
|
|
/// Move the cursor to the previous record and return it.
|
|
/// Used in backwards iteration.
|
|
fn get_prev_record(&mut self) -> Result<CursorResult<(Option<u64>, Option<OwnedRecord>)>> {
|
|
loop {
|
|
let page = self.stack.top();
|
|
let cell_idx = self.stack.current_cell_index();
|
|
|
|
// moved to beginning of current page
|
|
// todo: find a better way to flag moved to end or begin of page
|
|
if self.stack.current_cell_index_less_than_min() {
|
|
loop {
|
|
if self.stack.current_cell_index() > 0 {
|
|
self.stack.retreat();
|
|
break;
|
|
}
|
|
if self.stack.has_parent() {
|
|
self.stack.pop();
|
|
} else {
|
|
// moved to begin of btree
|
|
return Ok(CursorResult::Ok((None, None)));
|
|
}
|
|
}
|
|
// continue to next loop to get record from the new page
|
|
continue;
|
|
}
|
|
|
|
let cell_idx = cell_idx as usize;
|
|
debug!(
|
|
"get_prev_record current id={} cell={}",
|
|
page.get().id,
|
|
cell_idx
|
|
);
|
|
return_if_locked!(page);
|
|
if !page.is_loaded() {
|
|
self.pager.load_page(page.clone())?;
|
|
return Ok(CursorResult::IO);
|
|
}
|
|
let contents = page.get().contents.as_ref().unwrap();
|
|
|
|
let cell_count = contents.cell_count();
|
|
let cell_idx = if cell_idx >= cell_count {
|
|
self.stack.set_cell_index(cell_count as i32 - 1);
|
|
cell_count - 1
|
|
} else {
|
|
cell_idx
|
|
};
|
|
|
|
let cell = contents.cell_get(
|
|
cell_idx,
|
|
self.pager.clone(),
|
|
self.payload_overflow_threshold_max(contents.page_type()),
|
|
self.payload_overflow_threshold_min(contents.page_type()),
|
|
self.usable_space(),
|
|
)?;
|
|
|
|
match cell {
|
|
BTreeCell::TableInteriorCell(TableInteriorCell {
|
|
_left_child_page,
|
|
_rowid,
|
|
}) => {
|
|
let mem_page = self.pager.read_page(_left_child_page as usize)?;
|
|
self.stack.push(mem_page);
|
|
// use cell_index = i32::MAX to tell next loop to go to the end of the current page
|
|
self.stack.set_cell_index(i32::MAX);
|
|
continue;
|
|
}
|
|
BTreeCell::TableLeafCell(TableLeafCell {
|
|
_rowid, _payload, ..
|
|
}) => {
|
|
self.stack.retreat();
|
|
let record: OwnedRecord =
|
|
crate::storage::sqlite3_ondisk::read_record(&_payload)?;
|
|
return Ok(CursorResult::Ok((Some(_rowid), Some(record))));
|
|
}
|
|
BTreeCell::IndexInteriorCell(_) => todo!(),
|
|
BTreeCell::IndexLeafCell(_) => todo!(),
|
|
}
|
|
}
|
|
}
|
|
|
|
/// Move the cursor to the next record and return it.
|
|
/// Used in forwards iteration, which is the default.
|
|
fn get_next_record(
|
|
&mut self,
|
|
predicate: Option<(SeekKey<'_>, SeekOp)>,
|
|
) -> Result<CursorResult<(Option<u64>, Option<OwnedRecord>)>> {
|
|
loop {
|
|
let mem_page_rc = self.stack.top();
|
|
let cell_idx = self.stack.current_cell_index() as usize;
|
|
|
|
debug!("current id={} cell={}", mem_page_rc.get().id, cell_idx);
|
|
return_if_locked!(mem_page_rc);
|
|
if !mem_page_rc.is_loaded() {
|
|
self.pager.load_page(mem_page_rc.clone())?;
|
|
return Ok(CursorResult::IO);
|
|
}
|
|
let mem_page = mem_page_rc.get();
|
|
|
|
let contents = mem_page.contents.as_ref().unwrap();
|
|
|
|
if cell_idx == contents.cell_count() {
|
|
// do rightmost
|
|
let has_parent = self.stack.has_parent();
|
|
match contents.rightmost_pointer() {
|
|
Some(right_most_pointer) => {
|
|
self.stack.advance();
|
|
let mem_page = self.pager.read_page(right_most_pointer as usize)?;
|
|
self.stack.push(mem_page);
|
|
continue;
|
|
}
|
|
None => {
|
|
if has_parent {
|
|
debug!("moving simple upwards");
|
|
self.going_upwards = true;
|
|
self.stack.pop();
|
|
continue;
|
|
} else {
|
|
return Ok(CursorResult::Ok((None, None)));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if cell_idx > contents.cell_count() {
|
|
// end
|
|
let has_parent = self.stack.current() > 0;
|
|
if has_parent {
|
|
debug!("moving upwards");
|
|
self.going_upwards = true;
|
|
self.stack.pop();
|
|
continue;
|
|
} else {
|
|
return Ok(CursorResult::Ok((None, None)));
|
|
}
|
|
}
|
|
assert!(cell_idx < contents.cell_count());
|
|
|
|
let cell = contents.cell_get(
|
|
cell_idx,
|
|
self.pager.clone(),
|
|
self.payload_overflow_threshold_max(contents.page_type()),
|
|
self.payload_overflow_threshold_min(contents.page_type()),
|
|
self.usable_space(),
|
|
)?;
|
|
match &cell {
|
|
BTreeCell::TableInteriorCell(TableInteriorCell {
|
|
_left_child_page,
|
|
_rowid,
|
|
}) => {
|
|
assert!(predicate.is_none());
|
|
self.stack.advance();
|
|
let mem_page = self.pager.read_page(*_left_child_page as usize)?;
|
|
self.stack.push(mem_page);
|
|
continue;
|
|
}
|
|
BTreeCell::TableLeafCell(TableLeafCell {
|
|
_rowid,
|
|
_payload,
|
|
first_overflow_page: _,
|
|
}) => {
|
|
assert!(predicate.is_none());
|
|
self.stack.advance();
|
|
let record = crate::storage::sqlite3_ondisk::read_record(_payload)?;
|
|
return Ok(CursorResult::Ok((Some(*_rowid), Some(record))));
|
|
}
|
|
BTreeCell::IndexInteriorCell(IndexInteriorCell {
|
|
payload,
|
|
left_child_page,
|
|
..
|
|
}) => {
|
|
if !self.going_upwards {
|
|
let mem_page = self.pager.read_page(*left_child_page as usize)?;
|
|
self.stack.push(mem_page);
|
|
continue;
|
|
}
|
|
|
|
self.going_upwards = false;
|
|
self.stack.advance();
|
|
|
|
let record = crate::storage::sqlite3_ondisk::read_record(payload)?;
|
|
if predicate.is_none() {
|
|
let rowid = match record.values.last() {
|
|
Some(OwnedValue::Integer(rowid)) => *rowid as u64,
|
|
_ => unreachable!("index cells should have an integer rowid"),
|
|
};
|
|
return Ok(CursorResult::Ok((Some(rowid), Some(record))));
|
|
}
|
|
|
|
let (key, op) = predicate.as_ref().unwrap();
|
|
let SeekKey::IndexKey(index_key) = key else {
|
|
unreachable!("index seek key should be a record");
|
|
};
|
|
let found = match op {
|
|
SeekOp::GT => &record > *index_key,
|
|
SeekOp::GE => &record >= *index_key,
|
|
SeekOp::EQ => &record == *index_key,
|
|
};
|
|
if found {
|
|
let rowid = match record.values.last() {
|
|
Some(OwnedValue::Integer(rowid)) => *rowid as u64,
|
|
_ => unreachable!("index cells should have an integer rowid"),
|
|
};
|
|
return Ok(CursorResult::Ok((Some(rowid), Some(record))));
|
|
} else {
|
|
continue;
|
|
}
|
|
}
|
|
BTreeCell::IndexLeafCell(IndexLeafCell { payload, .. }) => {
|
|
self.stack.advance();
|
|
let record = crate::storage::sqlite3_ondisk::read_record(payload)?;
|
|
if predicate.is_none() {
|
|
let rowid = match record.values.last() {
|
|
Some(OwnedValue::Integer(rowid)) => *rowid as u64,
|
|
_ => unreachable!("index cells should have an integer rowid"),
|
|
};
|
|
return Ok(CursorResult::Ok((Some(rowid), Some(record))));
|
|
}
|
|
let (key, op) = predicate.as_ref().unwrap();
|
|
let SeekKey::IndexKey(index_key) = key else {
|
|
unreachable!("index seek key should be a record");
|
|
};
|
|
let found = match op {
|
|
SeekOp::GT => &record > *index_key,
|
|
SeekOp::GE => &record >= *index_key,
|
|
SeekOp::EQ => &record == *index_key,
|
|
};
|
|
if found {
|
|
let rowid = match record.values.last() {
|
|
Some(OwnedValue::Integer(rowid)) => *rowid as u64,
|
|
_ => unreachable!("index cells should have an integer rowid"),
|
|
};
|
|
return Ok(CursorResult::Ok((Some(rowid), Some(record))));
|
|
} else {
|
|
continue;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/// Move the cursor to the record that matches the seek key and seek operation.
|
|
/// This may be used to seek to a specific record in a point query (e.g. SELECT * FROM table WHERE col = 10)
|
|
/// or e.g. find the first record greater than the seek key in a range query (e.g. SELECT * FROM table WHERE col > 10).
|
|
/// We don't include the rowid in the comparison and that's why the last value from the record is not included.
|
|
fn seek(
|
|
&mut self,
|
|
key: SeekKey<'_>,
|
|
op: SeekOp,
|
|
) -> Result<CursorResult<(Option<u64>, Option<OwnedRecord>)>> {
|
|
return_if_io!(self.move_to(key.clone(), op.clone()));
|
|
|
|
{
|
|
let page = self.stack.top();
|
|
return_if_locked!(page);
|
|
|
|
let contents = page.get().contents.as_ref().unwrap();
|
|
|
|
for cell_idx in 0..contents.cell_count() {
|
|
let cell = contents.cell_get(
|
|
cell_idx,
|
|
self.pager.clone(),
|
|
self.payload_overflow_threshold_max(contents.page_type()),
|
|
self.payload_overflow_threshold_min(contents.page_type()),
|
|
self.usable_space(),
|
|
)?;
|
|
match &cell {
|
|
BTreeCell::TableLeafCell(TableLeafCell {
|
|
_rowid: cell_rowid,
|
|
_payload: payload,
|
|
first_overflow_page: _,
|
|
}) => {
|
|
let SeekKey::TableRowId(rowid_key) = key else {
|
|
unreachable!("table seek key should be a rowid");
|
|
};
|
|
let found = match op {
|
|
SeekOp::GT => *cell_rowid > rowid_key,
|
|
SeekOp::GE => *cell_rowid >= rowid_key,
|
|
SeekOp::EQ => *cell_rowid == rowid_key,
|
|
};
|
|
self.stack.advance();
|
|
if found {
|
|
let record = crate::storage::sqlite3_ondisk::read_record(payload)?;
|
|
return Ok(CursorResult::Ok((Some(*cell_rowid), Some(record))));
|
|
}
|
|
}
|
|
BTreeCell::IndexLeafCell(IndexLeafCell { payload, .. }) => {
|
|
let SeekKey::IndexKey(index_key) = key else {
|
|
unreachable!("index seek key should be a record");
|
|
};
|
|
let record = crate::storage::sqlite3_ondisk::read_record(payload)?;
|
|
let found = match op {
|
|
SeekOp::GT => {
|
|
&record.values[..record.values.len() - 1] > &index_key.values
|
|
}
|
|
SeekOp::GE => {
|
|
&record.values[..record.values.len() - 1] >= &index_key.values
|
|
}
|
|
SeekOp::EQ => {
|
|
&record.values[..record.values.len() - 1] == &index_key.values
|
|
}
|
|
};
|
|
self.stack.advance();
|
|
if found {
|
|
let rowid = match record.values.last() {
|
|
Some(OwnedValue::Integer(rowid)) => *rowid as u64,
|
|
_ => unreachable!("index cells should have an integer rowid"),
|
|
};
|
|
return Ok(CursorResult::Ok((Some(rowid), Some(record))));
|
|
}
|
|
}
|
|
cell_type => {
|
|
unreachable!("unexpected cell type: {:?}", cell_type);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// We have now iterated over all cells in the leaf page and found no match.
|
|
let is_index = matches!(key, SeekKey::IndexKey(_));
|
|
if is_index {
|
|
// Unlike tables, indexes store payloads in interior cells as well. self.move_to() always moves to a leaf page, so there are cases where we need to
|
|
// move back up to the parent interior cell and get the next record from there to perform a correct seek.
|
|
// an example of how this can occur:
|
|
//
|
|
// we do an index seek for key K with cmp = SeekOp::GT, meaning we want to seek to the first key that is greater than K.
|
|
// in self.move_to(), we encounter an interior cell with key K' = K+2, and move the left child page, which is a leaf page.
|
|
// the reason we move to the left child page is that we know that in an index, all keys in the left child page are less than K' i.e. less than K+2,
|
|
// meaning that the left subtree may contain a key greater than K, e.g. K+1. however, it is possible that it doesn't, in which case the correct
|
|
// next key is K+2, which is in the parent interior cell.
|
|
//
|
|
// In the seek() method, once we have landed in the leaf page and find that there is no cell with a key greater than K,
|
|
// if we were to return Ok(CursorResult::Ok((None, None))), self.record would be None, which is incorrect, because we already know
|
|
// that there is a record with a key greater than K (K' = K+2) in the parent interior cell. Hence, we need to move back up the tree
|
|
// and get the next matching record from there.
|
|
return self.get_next_record(Some((key, op)));
|
|
}
|
|
|
|
Ok(CursorResult::Ok((None, None)))
|
|
}
|
|
|
|
/// Move the cursor to the root page of the btree.
|
|
fn move_to_root(&mut self) {
|
|
let mem_page = self.pager.read_page(self.root_page).unwrap();
|
|
self.stack.clear();
|
|
self.stack.push(mem_page);
|
|
}
|
|
|
|
/// Move the cursor to the rightmost record in the btree.
|
|
fn move_to_rightmost(&mut self) -> Result<CursorResult<()>> {
|
|
self.move_to_root();
|
|
|
|
loop {
|
|
let mem_page = self.stack.top();
|
|
let page_idx = mem_page.get().id;
|
|
let page = self.pager.read_page(page_idx)?;
|
|
return_if_locked!(page);
|
|
let contents = page.get().contents.as_ref().unwrap();
|
|
if contents.is_leaf() {
|
|
if contents.cell_count() > 0 {
|
|
self.stack.set_cell_index(contents.cell_count() as i32 - 1);
|
|
}
|
|
return Ok(CursorResult::Ok(()));
|
|
}
|
|
|
|
match contents.rightmost_pointer() {
|
|
Some(right_most_pointer) => {
|
|
self.stack.set_cell_index(contents.cell_count() as i32 + 1);
|
|
let mem_page = self.pager.read_page(right_most_pointer as usize).unwrap();
|
|
self.stack.push(mem_page);
|
|
continue;
|
|
}
|
|
|
|
None => {
|
|
unreachable!("interior page should have a rightmost pointer");
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
pub fn move_to(&mut self, key: SeekKey<'_>, cmp: SeekOp) -> Result<CursorResult<()>> {
|
|
// For a table with N rows, we can find any row by row id in O(log(N)) time by starting at the root page and following the B-tree pointers.
|
|
// B-trees consist of interior pages and leaf pages. Interior pages contain pointers to other pages, while leaf pages contain the actual row data.
|
|
//
|
|
// Conceptually, each Interior Cell in a interior page has a rowid and a left child node, and the page itself has a right-most child node.
|
|
// Example: consider an interior page that contains cells C1(rowid=10), C2(rowid=20), C3(rowid=30).
|
|
// - All rows with rowids <= 10 are in the left child node of C1.
|
|
// - All rows with rowids > 10 and <= 20 are in the left child node of C2.
|
|
// - All rows with rowids > 20 and <= 30 are in the left child node of C3.
|
|
// - All rows with rowids > 30 are in the right-most child node of the page.
|
|
//
|
|
// There will generally be multiple levels of interior pages before we reach a leaf page,
|
|
// so we need to follow the interior page pointers until we reach the leaf page that contains the row we are looking for (if it exists).
|
|
//
|
|
// Here's a high-level overview of the algorithm:
|
|
// 1. Since we start at the root page, its cells are all interior cells.
|
|
// 2. We scan the interior cells until we find a cell whose rowid is greater than or equal to the rowid we are looking for.
|
|
// 3. Follow the left child pointer of the cell we found in step 2.
|
|
// a. In case none of the cells in the page have a rowid greater than or equal to the rowid we are looking for,
|
|
// we follow the right-most child pointer of the page instead (since all rows with rowids greater than the rowid we are looking for are in the right-most child node).
|
|
// 4. We are now at a new page. If it's another interior page, we repeat the process from step 2. If it's a leaf page, we continue to step 5.
|
|
// 5. We scan the leaf cells in the leaf page until we find the cell whose rowid is equal to the rowid we are looking for.
|
|
// This cell contains the actual data we are looking for.
|
|
// 6. If we find the cell, we return the record. Otherwise, we return an empty result.
|
|
self.move_to_root();
|
|
|
|
loop {
|
|
let page = self.stack.top();
|
|
return_if_locked!(page);
|
|
|
|
let contents = page.get().contents.as_ref().unwrap();
|
|
if contents.is_leaf() {
|
|
return Ok(CursorResult::Ok(()));
|
|
}
|
|
|
|
let mut found_cell = false;
|
|
for cell_idx in 0..contents.cell_count() {
|
|
match &contents.cell_get(
|
|
cell_idx,
|
|
self.pager.clone(),
|
|
self.payload_overflow_threshold_max(contents.page_type()),
|
|
self.payload_overflow_threshold_min(contents.page_type()),
|
|
self.usable_space(),
|
|
)? {
|
|
BTreeCell::TableInteriorCell(TableInteriorCell {
|
|
_left_child_page,
|
|
_rowid,
|
|
}) => {
|
|
let SeekKey::TableRowId(rowid_key) = key else {
|
|
unreachable!("table seek key should be a rowid");
|
|
};
|
|
let target_leaf_page_is_in_left_subtree = match cmp {
|
|
SeekOp::GT => rowid_key < *_rowid,
|
|
SeekOp::GE => rowid_key <= *_rowid,
|
|
SeekOp::EQ => rowid_key <= *_rowid,
|
|
};
|
|
self.stack.advance();
|
|
if target_leaf_page_is_in_left_subtree {
|
|
let mem_page = self.pager.read_page(*_left_child_page as usize)?;
|
|
self.stack.push(mem_page);
|
|
found_cell = true;
|
|
break;
|
|
}
|
|
}
|
|
BTreeCell::TableLeafCell(TableLeafCell {
|
|
_rowid: _,
|
|
_payload: _,
|
|
first_overflow_page: _,
|
|
}) => {
|
|
unreachable!(
|
|
"we don't iterate leaf cells while trying to move to a leaf cell"
|
|
);
|
|
}
|
|
BTreeCell::IndexInteriorCell(IndexInteriorCell {
|
|
left_child_page,
|
|
payload,
|
|
..
|
|
}) => {
|
|
let SeekKey::IndexKey(index_key) = key else {
|
|
unreachable!("index seek key should be a record");
|
|
};
|
|
let record = crate::storage::sqlite3_ondisk::read_record(payload)?;
|
|
let target_leaf_page_is_in_the_left_subtree = match cmp {
|
|
SeekOp::GT => index_key < &record,
|
|
SeekOp::GE => index_key <= &record,
|
|
SeekOp::EQ => index_key <= &record,
|
|
};
|
|
if target_leaf_page_is_in_the_left_subtree {
|
|
// we don't advance in case of index tree internal nodes because we will visit this node going up
|
|
let mem_page = self.pager.read_page(*left_child_page as usize).unwrap();
|
|
self.stack.push(mem_page);
|
|
found_cell = true;
|
|
break;
|
|
} else {
|
|
self.stack.advance();
|
|
}
|
|
}
|
|
BTreeCell::IndexLeafCell(_) => {
|
|
unreachable!(
|
|
"we don't iterate leaf cells while trying to move to a leaf cell"
|
|
);
|
|
}
|
|
}
|
|
}
|
|
|
|
if !found_cell {
|
|
match contents.rightmost_pointer() {
|
|
Some(right_most_pointer) => {
|
|
self.stack.advance();
|
|
let mem_page = self.pager.read_page(right_most_pointer as usize).unwrap();
|
|
self.stack.push(mem_page);
|
|
continue;
|
|
}
|
|
None => {
|
|
unreachable!("we shall not go back up! The only way is down the slope");
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/// Insert a record into the btree.
|
|
/// If the insert operation overflows the page, it will be split and the btree will be balanced.
|
|
fn insert_into_page(
|
|
&mut self,
|
|
key: &OwnedValue,
|
|
record: &OwnedRecord,
|
|
) -> Result<CursorResult<()>> {
|
|
loop {
|
|
let state = &self.write_info.state;
|
|
match state {
|
|
WriteState::Start => {
|
|
let page = self.stack.top();
|
|
let int_key = match key {
|
|
OwnedValue::Integer(i) => *i as u64,
|
|
_ => unreachable!("btree tables are indexed by integers!"),
|
|
};
|
|
|
|
// get page and find cell
|
|
let (cell_idx, page_type) = {
|
|
return_if_locked!(page);
|
|
|
|
page.set_dirty();
|
|
self.pager.add_dirty(page.get().id);
|
|
|
|
let page = page.get().contents.as_mut().unwrap();
|
|
assert!(matches!(page.page_type(), PageType::TableLeaf));
|
|
|
|
// find cell
|
|
(self.find_cell(page, int_key), page.page_type())
|
|
};
|
|
|
|
// TODO: if overwrite drop cell
|
|
|
|
// insert cell
|
|
|
|
let mut cell_payload: Vec<u8> = Vec::new();
|
|
self.fill_cell_payload(page_type, Some(int_key), &mut cell_payload, record);
|
|
|
|
// insert
|
|
let overflow = {
|
|
let contents = page.get().contents.as_mut().unwrap();
|
|
log::debug!(
|
|
"insert_into_page(overflow, cell_count={})",
|
|
contents.cell_count()
|
|
);
|
|
|
|
self.insert_into_cell(contents, cell_payload.as_slice(), cell_idx);
|
|
contents.overflow_cells.len()
|
|
};
|
|
if overflow > 0 {
|
|
self.write_info.state = WriteState::BalanceStart;
|
|
} else {
|
|
self.write_info.state = WriteState::Finish;
|
|
}
|
|
}
|
|
WriteState::BalanceStart
|
|
| WriteState::BalanceNonRoot
|
|
| WriteState::BalanceMoveUp
|
|
| WriteState::BalanceGetParentPage => {
|
|
return_if_io!(self.balance());
|
|
}
|
|
WriteState::Finish => {
|
|
self.write_info.state = WriteState::Start;
|
|
return Ok(CursorResult::Ok(()));
|
|
}
|
|
};
|
|
}
|
|
}
|
|
|
|
/// Insert a record into a cell.
|
|
/// If the cell overflows, an overflow cell is created.
|
|
/// insert_into_cell() is called from insert_into_page(),
|
|
/// and the overflow cell count is used to determine if the page overflows,
|
|
/// i.e. whether we need to balance the btree after the insert.
|
|
fn insert_into_cell(&self, page: &mut PageContent, payload: &[u8], cell_idx: usize) {
|
|
let free = self.compute_free_space(page, RefCell::borrow(&self.database_header));
|
|
const CELL_POINTER_SIZE_BYTES: usize = 2;
|
|
let enough_space = payload.len() + CELL_POINTER_SIZE_BYTES <= free as usize;
|
|
if !enough_space {
|
|
// add to overflow cell
|
|
page.overflow_cells.push(OverflowCell {
|
|
index: cell_idx,
|
|
payload: Pin::new(Vec::from(payload)),
|
|
});
|
|
return;
|
|
}
|
|
|
|
// TODO: insert into cell payload in internal page
|
|
let new_cell_data_pointer = self.allocate_cell_space(page, payload.len() as u16);
|
|
let buf = page.as_ptr();
|
|
|
|
// copy data
|
|
buf[new_cell_data_pointer as usize..new_cell_data_pointer as usize + payload.len()]
|
|
.copy_from_slice(payload);
|
|
// memmove(pIns+2, pIns, 2*(pPage->nCell - i));
|
|
let (cell_pointer_array_start, _) = page.cell_pointer_array_offset_and_size();
|
|
let cell_pointer_cur_idx = cell_pointer_array_start + (CELL_POINTER_SIZE_BYTES * cell_idx);
|
|
|
|
// move existing pointers forward by CELL_POINTER_SIZE_BYTES...
|
|
let n_cells_forward = page.cell_count() - cell_idx;
|
|
let n_bytes_forward = CELL_POINTER_SIZE_BYTES * n_cells_forward;
|
|
if n_bytes_forward > 0 {
|
|
buf.copy_within(
|
|
cell_pointer_cur_idx..cell_pointer_cur_idx + n_bytes_forward,
|
|
cell_pointer_cur_idx + CELL_POINTER_SIZE_BYTES,
|
|
);
|
|
}
|
|
// ...and insert new cell pointer at the current index
|
|
page.write_u16(cell_pointer_cur_idx - page.offset, new_cell_data_pointer);
|
|
|
|
// update first byte of content area (cell data always appended to the left, so cell content area pointer moves to point to the new cell data)
|
|
page.write_u16(PAGE_HEADER_OFFSET_CELL_CONTENT_AREA, new_cell_data_pointer);
|
|
|
|
// update cell count
|
|
let new_n_cells = (page.cell_count() + 1) as u16;
|
|
page.write_u16(PAGE_HEADER_OFFSET_CELL_COUNT, new_n_cells);
|
|
}
|
|
|
|
/// Free the range of bytes that a cell occupies.
|
|
/// This function also updates the freeblock list in the page.
|
|
/// Freeblocks are used to keep track of free space in the page,
|
|
/// and are organized as a linked list.
|
|
fn free_cell_range(&self, page: &mut PageContent, offset: u16, len: u16) {
|
|
// if the freeblock list is empty, we set this block as the first freeblock in the page header.
|
|
if page.first_freeblock() == 0 {
|
|
page.write_u16(offset as usize, 0); // next freeblock = null
|
|
page.write_u16(offset as usize + 2, len); // size of this freeblock
|
|
page.write_u16(PAGE_HEADER_OFFSET_FIRST_FREEBLOCK, offset); // first freeblock in page = this block
|
|
return;
|
|
}
|
|
let first_block = page.first_freeblock();
|
|
|
|
// if the freeblock list is not empty, and the offset is less than the first freeblock,
|
|
// we insert this block at the head of the list
|
|
if offset < first_block {
|
|
page.write_u16(offset as usize, first_block); // next freeblock = previous first freeblock
|
|
page.write_u16(offset as usize + 2, len); // size of this freeblock
|
|
page.write_u16(PAGE_HEADER_OFFSET_FIRST_FREEBLOCK, offset); // first freeblock in page = this block
|
|
return;
|
|
}
|
|
|
|
// if we clear space that is at the start of the cell content area,
|
|
// we need to update the cell content area pointer forward to account for the removed space
|
|
// FIXME: is offset ever < cell_content_area? cell content area grows leftwards and the pointer
|
|
// is to the start of the last allocated cell. should we assert!(offset >= page.cell_content_area())
|
|
// and change this to if offset == page.cell_content_area()?
|
|
if offset <= page.cell_content_area() {
|
|
// FIXME: remove the line directly below this, it does not change anything.
|
|
page.write_u16(PAGE_HEADER_OFFSET_FIRST_FREEBLOCK, page.first_freeblock());
|
|
page.write_u16(PAGE_HEADER_OFFSET_CELL_CONTENT_AREA, offset + len);
|
|
return;
|
|
}
|
|
|
|
// if the freeblock list is not empty, and the offset is greater than the first freeblock,
|
|
// then we need to do some more calculation to figure out where to insert the freeblock
|
|
// in the freeblock linked list.
|
|
let maxpc = {
|
|
let db_header = self.database_header.borrow();
|
|
let usable_space = (db_header.page_size - db_header.reserved_space as u16) as usize;
|
|
usable_space as u16
|
|
};
|
|
|
|
let mut pc = first_block;
|
|
let mut prev = first_block;
|
|
|
|
while pc <= maxpc && pc < offset {
|
|
let next = page.read_u16(pc as usize);
|
|
prev = pc;
|
|
pc = next;
|
|
}
|
|
|
|
if pc >= maxpc {
|
|
// insert into tail
|
|
let offset = offset as usize;
|
|
let prev = prev as usize;
|
|
page.write_u16(prev, offset as u16);
|
|
page.write_u16(offset, 0);
|
|
page.write_u16(offset + 2, len);
|
|
} else {
|
|
// insert in between
|
|
let next = page.read_u16(pc as usize);
|
|
let offset = offset as usize;
|
|
let prev = prev as usize;
|
|
page.write_u16(prev, offset as u16);
|
|
page.write_u16(offset, next);
|
|
page.write_u16(offset + 2, len);
|
|
}
|
|
}
|
|
|
|
/// Drop a cell from a page.
|
|
/// This is done by freeing the range of bytes that the cell occupies.
|
|
fn drop_cell(&self, page: &mut PageContent, cell_idx: usize) {
|
|
let (cell_start, cell_len) = page.cell_get_raw_region(
|
|
cell_idx,
|
|
self.payload_overflow_threshold_max(page.page_type()),
|
|
self.payload_overflow_threshold_min(page.page_type()),
|
|
self.usable_space(),
|
|
);
|
|
self.free_cell_range(page, cell_start as u16, cell_len as u16);
|
|
page.write_u16(PAGE_HEADER_OFFSET_CELL_COUNT, page.cell_count() as u16 - 1);
|
|
}
|
|
|
|
/// Balance a leaf page.
|
|
/// Balancing is done when a page overflows.
|
|
/// see e.g. https://en.wikipedia.org/wiki/B-tree
|
|
///
|
|
/// This is a naive algorithm that doesn't try to distribute cells evenly by content.
|
|
/// It will try to split the page in half by keys not by content.
|
|
/// Sqlite tries to have a page at least 40% full.
|
|
fn balance(&mut self) -> Result<CursorResult<()>> {
|
|
let state = &self.write_info.state;
|
|
match state {
|
|
WriteState::BalanceStart => {
|
|
// drop divider cells and find right pointer
|
|
// NOTE: since we are doing a simple split we only finding the pointer we want to update (right pointer).
|
|
// Right pointer means cell that points to the last page, as we don't really want to drop this one. This one
|
|
// can be a "rightmost pointer" or a "cell".
|
|
// we always asumme there is a parent
|
|
let current_page = self.stack.top();
|
|
{
|
|
// check if we don't need to balance
|
|
// don't continue if there are no overflow cells
|
|
let page = current_page.get().contents.as_mut().unwrap();
|
|
if page.overflow_cells.is_empty() {
|
|
self.write_info.state = WriteState::Finish;
|
|
return Ok(CursorResult::Ok(()));
|
|
}
|
|
}
|
|
|
|
if !self.stack.has_parent() {
|
|
self.balance_root();
|
|
return Ok(CursorResult::Ok(()));
|
|
}
|
|
|
|
self.write_info.state = WriteState::BalanceNonRoot;
|
|
self.balance_non_root()
|
|
}
|
|
WriteState::BalanceNonRoot
|
|
| WriteState::BalanceGetParentPage
|
|
| WriteState::BalanceMoveUp => self.balance_non_root(),
|
|
|
|
_ => unreachable!("invalid balance leaf state {:?}", state),
|
|
}
|
|
}
|
|
|
|
fn balance_non_root(&mut self) -> Result<CursorResult<()>> {
|
|
let state = &self.write_info.state;
|
|
match state {
|
|
WriteState::Start => todo!(),
|
|
WriteState::BalanceStart => todo!(),
|
|
WriteState::BalanceNonRoot => {
|
|
// drop divider cells and find right pointer
|
|
// NOTE: since we are doing a simple split we only finding the pointer we want to update (right pointer).
|
|
// Right pointer means cell that points to the last page, as we don't really want to drop this one. This one
|
|
// can be a "rightmost pointer" or a "cell".
|
|
// we always asumme there is a parent
|
|
let current_page = self.stack.top();
|
|
debug!("balance_non_root(page={})", current_page.get().id);
|
|
|
|
// Copy of page used to reference cell bytes.
|
|
// This needs to be saved somewhere safe so taht references still point to here,
|
|
// this will be store in write_info below
|
|
let page_copy = current_page.get().contents.as_ref().unwrap().clone();
|
|
|
|
// In memory in order copy of all cells in pages we want to balance. For now let's do a 2 page split.
|
|
// Right pointer in interior cells should be converted to regular cells if more than 2 pages are used for balancing.
|
|
let mut scratch_cells = self.write_info.scratch_cells.borrow_mut();
|
|
scratch_cells.clear();
|
|
|
|
for cell_idx in 0..page_copy.cell_count() {
|
|
let (start, len) = page_copy.cell_get_raw_region(
|
|
cell_idx,
|
|
self.payload_overflow_threshold_max(page_copy.page_type()),
|
|
self.payload_overflow_threshold_min(page_copy.page_type()),
|
|
self.usable_space(),
|
|
);
|
|
let buf = page_copy.as_ptr();
|
|
scratch_cells.push(to_static_buf(&buf[start..start + len]));
|
|
}
|
|
for overflow_cell in &page_copy.overflow_cells {
|
|
scratch_cells
|
|
.insert(overflow_cell.index, to_static_buf(&overflow_cell.payload));
|
|
}
|
|
*self.write_info.rightmost_pointer.borrow_mut() = page_copy.rightmost_pointer();
|
|
|
|
self.write_info.page_copy.replace(Some(page_copy));
|
|
|
|
// allocate new pages and move cells to those new pages
|
|
// split procedure
|
|
let page = current_page.get().contents.as_mut().unwrap();
|
|
assert!(
|
|
matches!(
|
|
page.page_type(),
|
|
PageType::TableLeaf | PageType::TableInterior
|
|
),
|
|
"indexes still not supported "
|
|
);
|
|
|
|
let right_page = self.allocate_page(page.page_type(), 0);
|
|
let right_page_id = right_page.get().id;
|
|
|
|
self.write_info.new_pages.borrow_mut().clear();
|
|
self.write_info
|
|
.new_pages
|
|
.borrow_mut()
|
|
.push(current_page.clone());
|
|
self.write_info
|
|
.new_pages
|
|
.borrow_mut()
|
|
.push(right_page.clone());
|
|
|
|
debug!(
|
|
"splitting left={} right={}",
|
|
current_page.get().id,
|
|
right_page_id
|
|
);
|
|
|
|
self.write_info.state = WriteState::BalanceGetParentPage;
|
|
Ok(CursorResult::Ok(()))
|
|
}
|
|
WriteState::BalanceGetParentPage => {
|
|
let parent = self.stack.parent();
|
|
let loaded = parent.is_loaded();
|
|
return_if_locked!(parent);
|
|
|
|
if !loaded {
|
|
debug!("balance_leaf(loading page)");
|
|
self.pager.load_page(parent.clone())?;
|
|
return Ok(CursorResult::IO);
|
|
}
|
|
parent.set_dirty();
|
|
self.write_info.state = WriteState::BalanceMoveUp;
|
|
Ok(CursorResult::Ok(()))
|
|
}
|
|
WriteState::BalanceMoveUp => {
|
|
let parent = self.stack.parent();
|
|
|
|
let (page_type, current_idx) = {
|
|
let current_page = self.stack.top();
|
|
let contents = current_page.get().contents.as_ref().unwrap();
|
|
(contents.page_type().clone(), current_page.get().id)
|
|
};
|
|
|
|
parent.set_dirty();
|
|
self.pager.add_dirty(parent.get().id);
|
|
let parent_contents = parent.get().contents.as_mut().unwrap();
|
|
// if this isn't empty next loop won't work
|
|
assert_eq!(parent_contents.overflow_cells.len(), 0);
|
|
|
|
// Right page pointer is u32 in right most pointer, and in cell is u32 too, so we can use a *u32 to hold where we want to change this value
|
|
let mut right_pointer = PAGE_HEADER_OFFSET_RIGHTMOST_PTR;
|
|
for cell_idx in 0..parent_contents.cell_count() {
|
|
let cell = parent_contents
|
|
.cell_get(
|
|
cell_idx,
|
|
self.pager.clone(),
|
|
self.payload_overflow_threshold_max(page_type.clone()),
|
|
self.payload_overflow_threshold_min(page_type.clone()),
|
|
self.usable_space(),
|
|
)
|
|
.unwrap();
|
|
let found = match cell {
|
|
BTreeCell::TableInteriorCell(interior) => {
|
|
interior._left_child_page as usize == current_idx
|
|
}
|
|
_ => unreachable!("Parent should always be a "),
|
|
};
|
|
if found {
|
|
let (start, _len) = parent_contents.cell_get_raw_region(
|
|
cell_idx,
|
|
self.payload_overflow_threshold_max(page_type.clone()),
|
|
self.payload_overflow_threshold_min(page_type.clone()),
|
|
self.usable_space(),
|
|
);
|
|
right_pointer = start;
|
|
break;
|
|
}
|
|
}
|
|
|
|
let mut new_pages = self.write_info.new_pages.borrow_mut();
|
|
let scratch_cells = self.write_info.scratch_cells.borrow();
|
|
|
|
// reset pages
|
|
for page in new_pages.iter() {
|
|
assert!(page.is_dirty());
|
|
let contents = page.get().contents.as_mut().unwrap();
|
|
|
|
contents.write_u16(PAGE_HEADER_OFFSET_FIRST_FREEBLOCK, 0);
|
|
contents.write_u16(PAGE_HEADER_OFFSET_CELL_COUNT, 0);
|
|
|
|
let db_header = RefCell::borrow(&self.database_header);
|
|
let cell_content_area_start =
|
|
db_header.page_size - db_header.reserved_space as u16;
|
|
contents.write_u16(
|
|
PAGE_HEADER_OFFSET_CELL_CONTENT_AREA,
|
|
cell_content_area_start,
|
|
);
|
|
|
|
contents.write_u8(PAGE_HEADER_OFFSET_FRAGMENTED_BYTES_COUNT, 0);
|
|
if !contents.is_leaf() {
|
|
contents.write_u32(PAGE_HEADER_OFFSET_RIGHTMOST_PTR, 0);
|
|
}
|
|
}
|
|
|
|
// distribute cells
|
|
let new_pages_len = new_pages.len();
|
|
let cells_per_page = scratch_cells.len() / new_pages.len();
|
|
let mut current_cell_index = 0_usize;
|
|
let mut divider_cells_index = Vec::new(); /* index to scratch cells that will be used as dividers in order */
|
|
|
|
debug!(
|
|
"balance_leaf::distribute(cells={}, cells_per_page={})",
|
|
scratch_cells.len(),
|
|
cells_per_page
|
|
);
|
|
|
|
for (i, page) in new_pages.iter_mut().enumerate() {
|
|
let page_id = page.get().id;
|
|
let contents = page.get().contents.as_mut().unwrap();
|
|
|
|
let last_page = i == new_pages_len - 1;
|
|
let cells_to_copy = if last_page {
|
|
// last cells is remaining pages if division was odd
|
|
scratch_cells.len() - current_cell_index
|
|
} else {
|
|
cells_per_page
|
|
};
|
|
debug!(
|
|
"balance_leaf::distribute(page={}, cells_to_copy={})",
|
|
page_id, cells_to_copy
|
|
);
|
|
|
|
let cell_index_range = current_cell_index..current_cell_index + cells_to_copy;
|
|
for (j, cell_idx) in cell_index_range.enumerate() {
|
|
debug!("balance_leaf::distribute_in_page(page={}, cells_to_copy={}, j={}, cell_idx={})", page_id, cells_to_copy, j, cell_idx);
|
|
|
|
let cell = scratch_cells[cell_idx];
|
|
self.insert_into_cell(contents, cell, j);
|
|
}
|
|
divider_cells_index.push(current_cell_index + cells_to_copy - 1);
|
|
current_cell_index += cells_to_copy;
|
|
}
|
|
let is_leaf = {
|
|
let page = self.stack.top();
|
|
let page = page.get().contents.as_ref().unwrap();
|
|
page.is_leaf()
|
|
};
|
|
|
|
// update rightmost pointer for each page if we are in interior page
|
|
if !is_leaf {
|
|
for page in new_pages.iter_mut().take(new_pages_len - 1) {
|
|
let contents = page.get().contents.as_mut().unwrap();
|
|
|
|
assert!(contents.cell_count() == 1);
|
|
let last_cell = contents
|
|
.cell_get(
|
|
contents.cell_count() - 1,
|
|
self.pager.clone(),
|
|
self.payload_overflow_threshold_max(contents.page_type()),
|
|
self.payload_overflow_threshold_min(contents.page_type()),
|
|
self.usable_space(),
|
|
)
|
|
.unwrap();
|
|
let last_cell_pointer = match last_cell {
|
|
BTreeCell::TableInteriorCell(interior) => interior._left_child_page,
|
|
_ => unreachable!(),
|
|
};
|
|
self.drop_cell(contents, contents.cell_count() - 1);
|
|
contents.write_u32(PAGE_HEADER_OFFSET_RIGHTMOST_PTR, last_cell_pointer);
|
|
}
|
|
// last page right most pointer points to previous right most pointer before splitting
|
|
let last_page = new_pages.last().unwrap();
|
|
let last_page_contents = last_page.get().contents.as_mut().unwrap();
|
|
last_page_contents.write_u32(
|
|
PAGE_HEADER_OFFSET_RIGHTMOST_PTR,
|
|
self.write_info.rightmost_pointer.borrow().unwrap(),
|
|
);
|
|
}
|
|
|
|
// insert dividers in parent
|
|
// we can consider dividers the first cell of each page starting from the second page
|
|
for (page_id_index, page) in
|
|
new_pages.iter_mut().take(new_pages_len - 1).enumerate()
|
|
{
|
|
let contents = page.get().contents.as_mut().unwrap();
|
|
let divider_cell_index = divider_cells_index[page_id_index];
|
|
let cell_payload = scratch_cells[divider_cell_index];
|
|
let cell = read_btree_cell(
|
|
cell_payload,
|
|
&contents.page_type(),
|
|
0,
|
|
self.pager.clone(),
|
|
self.payload_overflow_threshold_max(contents.page_type()),
|
|
self.payload_overflow_threshold_min(contents.page_type()),
|
|
self.usable_space(),
|
|
)
|
|
.unwrap();
|
|
|
|
if is_leaf {
|
|
// create a new divider cell and push
|
|
let key = match cell {
|
|
BTreeCell::TableLeafCell(leaf) => leaf._rowid,
|
|
_ => unreachable!(),
|
|
};
|
|
let mut divider_cell = Vec::new();
|
|
divider_cell.extend_from_slice(&(page.get().id as u32).to_be_bytes());
|
|
divider_cell.extend(std::iter::repeat(0).take(9));
|
|
let n = write_varint(&mut divider_cell.as_mut_slice()[4..], key);
|
|
divider_cell.truncate(4 + n);
|
|
let parent_cell_idx = self.find_cell(parent_contents, key);
|
|
self.insert_into_cell(
|
|
parent_contents,
|
|
divider_cell.as_slice(),
|
|
parent_cell_idx,
|
|
);
|
|
} else {
|
|
// move cell
|
|
let key = match cell {
|
|
BTreeCell::TableInteriorCell(interior) => interior._rowid,
|
|
_ => unreachable!(),
|
|
};
|
|
let parent_cell_idx = self.find_cell(contents, key);
|
|
self.insert_into_cell(parent_contents, cell_payload, parent_cell_idx);
|
|
// self.drop_cell(*page, 0);
|
|
}
|
|
}
|
|
|
|
{
|
|
// copy last page id to right pointer
|
|
let last_pointer = new_pages.last().unwrap().get().id as u32;
|
|
parent_contents.write_u32(right_pointer, last_pointer);
|
|
}
|
|
self.stack.pop();
|
|
self.write_info.state = WriteState::BalanceStart;
|
|
let _ = self.write_info.page_copy.take();
|
|
Ok(CursorResult::Ok(()))
|
|
}
|
|
WriteState::Finish => todo!(),
|
|
}
|
|
}
|
|
|
|
/// Balance the root page.
|
|
/// This is done when the root page overflows, and we need to create a new root page.
|
|
/// See e.g. https://en.wikipedia.org/wiki/B-tree
|
|
fn balance_root(&mut self) {
|
|
/* todo: balance deeper, create child and copy contents of root there. Then split root */
|
|
/* if we are in root page then we just need to create a new root and push key there */
|
|
|
|
let is_page_1 = {
|
|
let current_root = self.stack.top();
|
|
current_root.get().id == 1
|
|
};
|
|
|
|
let offset = if is_page_1 { DATABASE_HEADER_SIZE } else { 0 };
|
|
let new_root_page = self.allocate_page(PageType::TableInterior, offset);
|
|
{
|
|
let current_root = self.stack.top();
|
|
let current_root_contents = current_root.get().contents.as_ref().unwrap();
|
|
|
|
let new_root_page_id = new_root_page.get().id;
|
|
let new_root_page_contents = new_root_page.get().contents.as_mut().unwrap();
|
|
if is_page_1 {
|
|
// Copy header
|
|
let current_root_buf = current_root_contents.as_ptr();
|
|
let new_root_buf = new_root_page_contents.as_ptr();
|
|
new_root_buf[0..DATABASE_HEADER_SIZE]
|
|
.copy_from_slice(¤t_root_buf[0..DATABASE_HEADER_SIZE]);
|
|
}
|
|
// point new root right child to previous root
|
|
new_root_page_contents
|
|
.write_u32(PAGE_HEADER_OFFSET_RIGHTMOST_PTR, new_root_page_id as u32);
|
|
new_root_page_contents.write_u16(PAGE_HEADER_OFFSET_CELL_COUNT, 0);
|
|
}
|
|
|
|
/* swap splitted page buffer with new root buffer so we don't have to update page idx */
|
|
{
|
|
let (root_id, child_id, child) = {
|
|
let page_ref = self.stack.top();
|
|
let child = page_ref.clone();
|
|
|
|
// Swap the entire Page structs
|
|
std::mem::swap(&mut child.get().id, &mut new_root_page.get().id);
|
|
// TODO:: shift bytes by offset to left on child because now child has offset 100
|
|
// and header bytes
|
|
// Also change the offset of page
|
|
//
|
|
if is_page_1 {
|
|
// Remove header from child and set offset to 0
|
|
let contents = child.get().contents.as_mut().unwrap();
|
|
let (cell_pointer_offset, _) = contents.cell_pointer_array_offset_and_size();
|
|
// change cell pointers
|
|
for cell_idx in 0..contents.cell_count() {
|
|
let cell_pointer_offset = cell_pointer_offset + (2 * cell_idx) - offset;
|
|
let pc = contents.read_u16(cell_pointer_offset);
|
|
contents.write_u16(cell_pointer_offset, pc - offset as u16);
|
|
}
|
|
|
|
contents.offset = 0;
|
|
let buf = contents.as_ptr();
|
|
buf.copy_within(DATABASE_HEADER_SIZE.., 0);
|
|
}
|
|
|
|
self.pager.add_dirty(new_root_page.get().id);
|
|
self.pager.add_dirty(child.get().id);
|
|
(new_root_page.get().id, child.get().id, child)
|
|
};
|
|
|
|
debug!("Balancing root. root={}, rightmost={}", root_id, child_id);
|
|
let root = new_root_page.clone();
|
|
|
|
self.root_page = root_id;
|
|
self.stack.clear();
|
|
self.stack.push(root.clone());
|
|
self.stack.push(child.clone());
|
|
|
|
self.pager.put_loaded_page(root_id, root);
|
|
self.pager.put_loaded_page(child_id, child);
|
|
}
|
|
}
|
|
|
|
/// Allocate a new page to the btree via the pager.
|
|
/// This marks the page as dirty and writes the page header.
|
|
fn allocate_page(&self, page_type: PageType, offset: usize) -> PageRef {
|
|
let page = self.pager.allocate_page().unwrap();
|
|
btree_init_page(&page, page_type, &self.database_header.borrow(), offset);
|
|
page
|
|
}
|
|
|
|
/// Allocate a new overflow page.
|
|
/// This is done when a cell overflows and new space is needed.
|
|
fn allocate_overflow_page(&self) -> PageRef {
|
|
let page = self.pager.allocate_page().unwrap();
|
|
|
|
// setup overflow page
|
|
let contents = page.get().contents.as_mut().unwrap();
|
|
let buf = contents.as_ptr();
|
|
buf.fill(0);
|
|
|
|
page
|
|
}
|
|
|
|
/// Allocate space for a cell on a page.
|
|
fn allocate_cell_space(&self, page_ref: &PageContent, amount: u16) -> u16 {
|
|
let amount = amount as usize;
|
|
|
|
let (cell_offset, _) = page_ref.cell_pointer_array_offset_and_size();
|
|
let gap = cell_offset + 2 * page_ref.cell_count();
|
|
let mut top = page_ref.cell_content_area() as usize;
|
|
|
|
// there are free blocks and enough space
|
|
if page_ref.first_freeblock() != 0 && gap + 2 <= top {
|
|
// find slot
|
|
let db_header = RefCell::borrow(&self.database_header);
|
|
let pc = find_free_cell(page_ref, db_header, amount);
|
|
if pc != 0 {
|
|
return pc as u16;
|
|
}
|
|
/* fall through, we might need to defragment */
|
|
}
|
|
|
|
if gap + 2 + amount > top {
|
|
// defragment
|
|
self.defragment_page(page_ref, RefCell::borrow(&self.database_header));
|
|
top = page_ref.read_u16(PAGE_HEADER_OFFSET_CELL_CONTENT_AREA) as usize;
|
|
}
|
|
|
|
let db_header = RefCell::borrow(&self.database_header);
|
|
top -= amount;
|
|
|
|
page_ref.write_u16(PAGE_HEADER_OFFSET_CELL_CONTENT_AREA, top as u16);
|
|
|
|
let usable_space = (db_header.page_size - db_header.reserved_space as u16) as usize;
|
|
assert!(top + amount <= usable_space);
|
|
top as u16
|
|
}
|
|
|
|
/// Defragment a page. This means packing all the cells to the end of the page.
|
|
fn defragment_page(&self, page: &PageContent, db_header: Ref<DatabaseHeader>) {
|
|
log::debug!("defragment_page");
|
|
let cloned_page = page.clone();
|
|
// TODO(pere): usable space should include offset probably
|
|
let usable_space = (db_header.page_size - db_header.reserved_space as u16) as u64;
|
|
let mut cbrk = usable_space;
|
|
|
|
// TODO: implement fast algorithm
|
|
|
|
let last_cell = usable_space - 4;
|
|
let first_cell = cloned_page.unallocated_region_start() as u64;
|
|
|
|
if cloned_page.cell_count() > 0 {
|
|
let page_type = page.page_type();
|
|
let read_buf = cloned_page.as_ptr();
|
|
let write_buf = page.as_ptr();
|
|
|
|
for i in 0..cloned_page.cell_count() {
|
|
let cell_offset = page.offset + 8;
|
|
let cell_idx = cell_offset + i * 2;
|
|
|
|
let pc = u16::from_be_bytes([read_buf[cell_idx], read_buf[cell_idx + 1]]) as u64;
|
|
if pc > last_cell {
|
|
unimplemented!("corrupted page");
|
|
}
|
|
|
|
assert!(pc <= last_cell);
|
|
|
|
let size = match page_type {
|
|
PageType::TableInterior => {
|
|
let (_, nr_key) = match read_varint(&read_buf[pc as usize ..]) {
|
|
Ok(v) => v,
|
|
Err(_) => todo!(
|
|
"error while parsing varint from cell, probably treat this as corruption?"
|
|
),
|
|
};
|
|
4 + nr_key as u64
|
|
}
|
|
PageType::TableLeaf => {
|
|
let (payload_size, nr_payload) = match read_varint(&read_buf[pc as usize..]) {
|
|
Ok(v) => v,
|
|
Err(_) => todo!(
|
|
"error while parsing varint from cell, probably treat this as corruption?"
|
|
),
|
|
};
|
|
let (_, nr_key) = match read_varint(&read_buf[pc as usize + nr_payload..]) {
|
|
Ok(v) => v,
|
|
Err(_) => todo!(
|
|
"error while parsing varint from cell, probably treat this as corruption?"
|
|
),
|
|
};
|
|
// TODO: add overflow page calculation
|
|
payload_size + nr_payload as u64 + nr_key as u64
|
|
}
|
|
PageType::IndexInterior => todo!(),
|
|
PageType::IndexLeaf => todo!(),
|
|
};
|
|
cbrk -= size;
|
|
if cbrk < first_cell || pc + size > usable_space {
|
|
todo!("corrupt");
|
|
}
|
|
assert!(cbrk + size <= usable_space && cbrk >= first_cell);
|
|
// set new pointer
|
|
write_buf[cell_idx..cell_idx + 2].copy_from_slice(&(cbrk as u16).to_be_bytes());
|
|
// copy payload
|
|
write_buf[cbrk as usize..cbrk as usize + size as usize]
|
|
.copy_from_slice(&read_buf[pc as usize..pc as usize + size as usize]);
|
|
}
|
|
}
|
|
|
|
// assert!( nfree >= 0 );
|
|
// if( data[hdr+7]+cbrk-iCellFirst!=pPage->nFree ){
|
|
// return SQLITE_CORRUPT_PAGE(pPage);
|
|
// }
|
|
assert!(cbrk >= first_cell);
|
|
let write_buf = page.as_ptr();
|
|
|
|
// set new first byte of cell content
|
|
page.write_u16(PAGE_HEADER_OFFSET_CELL_CONTENT_AREA, cbrk as u16);
|
|
// set free block to 0, unused spaced can be retrieved from gap between cell pointer end and content start
|
|
page.write_u16(PAGE_HEADER_OFFSET_FIRST_FREEBLOCK, 0);
|
|
// set unused space to 0
|
|
let first_cell = cloned_page.cell_content_area() as u64;
|
|
assert!(first_cell <= cbrk);
|
|
write_buf[first_cell as usize..cbrk as usize].fill(0);
|
|
}
|
|
|
|
/// Free blocks can be zero, meaning the "real free space" that can be used to allocate is expected to be between first cell byte
|
|
/// and end of cell pointer area.
|
|
#[allow(unused_assignments)]
|
|
fn compute_free_space(&self, page: &PageContent, db_header: Ref<DatabaseHeader>) -> u16 {
|
|
// TODO(pere): maybe free space is not calculated correctly with offset
|
|
|
|
// Usable space, not the same as free space, simply means:
|
|
// space that is not reserved for extensions by sqlite. Usually reserved_space is 0.
|
|
let usable_space = (db_header.page_size - db_header.reserved_space as u16) as usize;
|
|
|
|
let mut cell_content_area_start = page.cell_content_area();
|
|
// A zero value for the cell content area pointer is interpreted as 65536.
|
|
// See https://www.sqlite.org/fileformat.html
|
|
// The max page size for a sqlite database is 64kiB i.e. 65536 bytes.
|
|
// 65536 is u16::MAX + 1, and since cell content grows from right to left, this means
|
|
// the cell content area pointer is at the end of the page,
|
|
// i.e.
|
|
// 1. the page size is 64kiB
|
|
// 2. there are no cells on the page
|
|
// 3. there is no reserved space at the end of the page
|
|
if cell_content_area_start == 0 {
|
|
cell_content_area_start = u16::MAX;
|
|
}
|
|
|
|
// The amount of free space is the sum of:
|
|
// #1. the size of the unallocated region
|
|
// #2. fragments (isolated 1-3 byte chunks of free space within the cell content area)
|
|
// #3. freeblocks (linked list of blocks of at least 4 bytes within the cell content area that are not in use due to e.g. deletions)
|
|
|
|
let mut free_space_bytes =
|
|
page.unallocated_region_size() + page.num_frag_free_bytes() as usize;
|
|
|
|
// #3 is computed by iterating over the freeblocks linked list
|
|
let mut cur_freeblock_ptr = page.first_freeblock() as usize;
|
|
let page_buf = page.as_ptr();
|
|
if cur_freeblock_ptr > 0 {
|
|
if cur_freeblock_ptr < cell_content_area_start as usize {
|
|
// Freeblocks exist in the cell content area e.g. after deletions
|
|
// They should never exist in the unused area of the page.
|
|
todo!("corrupted page");
|
|
}
|
|
|
|
let mut next = 0;
|
|
let mut size = 0;
|
|
loop {
|
|
// TODO: check corruption icellast
|
|
next = u16::from_be_bytes(
|
|
page_buf[cur_freeblock_ptr..cur_freeblock_ptr + 2]
|
|
.try_into()
|
|
.unwrap(),
|
|
) as usize; // first 2 bytes in freeblock = next freeblock pointer
|
|
size = u16::from_be_bytes(
|
|
page_buf[cur_freeblock_ptr + 2..cur_freeblock_ptr + 4]
|
|
.try_into()
|
|
.unwrap(),
|
|
) as usize; // next 2 bytes in freeblock = size of current freeblock
|
|
free_space_bytes += size;
|
|
// Freeblocks are in order from left to right on the page,
|
|
// so next pointer should > current pointer + its size, or 0 if no next block exists.
|
|
if next <= cur_freeblock_ptr + size + 3 {
|
|
break;
|
|
}
|
|
cur_freeblock_ptr = next;
|
|
}
|
|
|
|
// Next should always be 0 (NULL) at this point since we have reached the end of the freeblocks linked list
|
|
assert!(
|
|
next == 0,
|
|
"corrupted page: freeblocks list not in ascending order"
|
|
);
|
|
|
|
assert!(
|
|
cur_freeblock_ptr + size <= usable_space,
|
|
"corrupted page: last freeblock extends last page end"
|
|
);
|
|
}
|
|
|
|
assert!(
|
|
free_space_bytes <= usable_space,
|
|
"corrupted page: free space is greater than usable space"
|
|
);
|
|
|
|
// if( nFree>usableSize || nFree<iCellFirst ){
|
|
// return SQLITE_CORRUPT_PAGE(pPage);
|
|
// }
|
|
|
|
free_space_bytes as u16
|
|
}
|
|
|
|
/// Fill in the cell payload with the record.
|
|
/// If the record is too large to fit in the cell, it will spill onto overflow pages.
|
|
fn fill_cell_payload(
|
|
&self,
|
|
page_type: PageType,
|
|
int_key: Option<u64>,
|
|
cell_payload: &mut Vec<u8>,
|
|
record: &OwnedRecord,
|
|
) {
|
|
assert!(matches!(
|
|
page_type,
|
|
PageType::TableLeaf | PageType::IndexLeaf
|
|
));
|
|
// TODO: make record raw from start, having to serialize is not good
|
|
let mut record_buf = Vec::new();
|
|
record.serialize(&mut record_buf);
|
|
|
|
// fill in header
|
|
if matches!(page_type, PageType::TableLeaf) {
|
|
let int_key = int_key.unwrap();
|
|
write_varint_to_vec(record_buf.len() as u64, cell_payload);
|
|
write_varint_to_vec(int_key, cell_payload);
|
|
} else {
|
|
write_varint_to_vec(record_buf.len() as u64, cell_payload);
|
|
}
|
|
|
|
let payload_overflow_threshold_max = self.payload_overflow_threshold_max(page_type.clone());
|
|
log::debug!(
|
|
"fill_cell_payload(record_size={}, payload_overflow_threshold_max={})",
|
|
record_buf.len(),
|
|
payload_overflow_threshold_max
|
|
);
|
|
if record_buf.len() <= payload_overflow_threshold_max {
|
|
// enough allowed space to fit inside a btree page
|
|
cell_payload.extend_from_slice(record_buf.as_slice());
|
|
return;
|
|
}
|
|
log::debug!("fill_cell_payload(overflow)");
|
|
|
|
let payload_overflow_threshold_min = self.payload_overflow_threshold_min(page_type);
|
|
// see e.g. https://github.com/sqlite/sqlite/blob/9591d3fe93936533c8c3b0dc4d025ac999539e11/src/dbstat.c#L371
|
|
let mut space_left = payload_overflow_threshold_min
|
|
+ (record_buf.len() - payload_overflow_threshold_min) % (self.usable_space() - 4);
|
|
|
|
if space_left > payload_overflow_threshold_max {
|
|
space_left = payload_overflow_threshold_min;
|
|
}
|
|
|
|
// cell_size must be equal to first value of space_left as this will be the bytes copied to non-overflow page.
|
|
let cell_size = space_left + cell_payload.len() + 4; // 4 is the number of bytes of pointer to first overflow page
|
|
let mut to_copy_buffer = record_buf.as_slice();
|
|
|
|
let prev_size = cell_payload.len();
|
|
cell_payload.resize(prev_size + space_left + 4, 0);
|
|
let mut pointer = unsafe { cell_payload.as_mut_ptr().add(prev_size) };
|
|
let mut pointer_to_next = unsafe { cell_payload.as_mut_ptr().add(prev_size + space_left) };
|
|
let mut overflow_pages = Vec::new();
|
|
|
|
loop {
|
|
let to_copy = space_left.min(to_copy_buffer.len());
|
|
unsafe { std::ptr::copy(to_copy_buffer.as_ptr(), pointer, to_copy) };
|
|
|
|
let left = to_copy_buffer.len() - to_copy;
|
|
if left == 0 {
|
|
break;
|
|
}
|
|
|
|
// we still have bytes to add, we will need to allocate new overflow page
|
|
let overflow_page = self.allocate_overflow_page();
|
|
overflow_pages.push(overflow_page.clone());
|
|
{
|
|
let id = overflow_page.get().id as u32;
|
|
let contents = overflow_page.get().contents.as_mut().unwrap();
|
|
|
|
// TODO: take into account offset here?
|
|
let buf = contents.as_ptr();
|
|
let as_bytes = id.to_be_bytes();
|
|
// update pointer to new overflow page
|
|
unsafe { std::ptr::copy(as_bytes.as_ptr(), pointer_to_next, 4) };
|
|
|
|
pointer = unsafe { buf.as_mut_ptr().add(4) };
|
|
pointer_to_next = buf.as_mut_ptr();
|
|
space_left = self.usable_space() - 4;
|
|
}
|
|
|
|
to_copy_buffer = &to_copy_buffer[to_copy..];
|
|
}
|
|
|
|
assert_eq!(cell_size, cell_payload.len());
|
|
}
|
|
|
|
/// Returns the maximum payload size (X) that can be stored directly on a b-tree page without spilling to overflow pages.
|
|
///
|
|
/// For table leaf pages: X = usable_size - 35
|
|
/// For index pages: X = ((usable_size - 12) * 64/255) - 23
|
|
///
|
|
/// The usable size is the total page size less the reserved space at the end of each page.
|
|
/// These thresholds are designed to:
|
|
/// - Give a minimum fanout of 4 for index b-trees
|
|
/// - Ensure enough payload is on the b-tree page that the record header can usually be accessed
|
|
/// without consulting an overflow page
|
|
fn payload_overflow_threshold_max(&self, page_type: PageType) -> usize {
|
|
let usable_size = self.usable_space();
|
|
match page_type {
|
|
PageType::IndexInterior | PageType::IndexLeaf => {
|
|
((usable_size - 12) * 64 / 255) - 23 // Index page formula
|
|
}
|
|
PageType::TableInterior | PageType::TableLeaf => {
|
|
usable_size - 35 // Table leaf page formula
|
|
}
|
|
}
|
|
}
|
|
|
|
/// Returns the minimum payload size (M) that must be stored on the b-tree page before spilling to overflow pages is allowed.
|
|
///
|
|
/// For all page types: M = ((usable_size - 12) * 32/255) - 23
|
|
///
|
|
/// When payload size P exceeds max_local():
|
|
/// - If K = M + ((P-M) % (usable_size-4)) <= max_local(): store K bytes on page
|
|
/// - Otherwise: store M bytes on page
|
|
///
|
|
/// The remaining bytes are stored on overflow pages in both cases.
|
|
fn payload_overflow_threshold_min(&self, _page_type: PageType) -> usize {
|
|
let usable_size = self.usable_space();
|
|
// Same formula for all page types
|
|
((usable_size - 12) * 32 / 255) - 23
|
|
}
|
|
|
|
/// The "usable size" of a database page is the page size specified by the 2-byte integer at offset 16
|
|
/// in the header, minus the "reserved" space size recorded in the 1-byte integer at offset 20 in the header.
|
|
/// The usable size of a page might be an odd number. However, the usable size is not allowed to be less than 480.
|
|
/// In other words, if the page size is 512, then the reserved space size cannot exceed 32.
|
|
fn usable_space(&self) -> usize {
|
|
let db_header = RefCell::borrow(&self.database_header);
|
|
(db_header.page_size - db_header.reserved_space as u16) as usize
|
|
}
|
|
|
|
/// Find the index of the cell in the page that contains the given rowid.
|
|
/// BTree tables only.
|
|
fn find_cell(&self, page: &PageContent, int_key: u64) -> usize {
|
|
let mut cell_idx = 0;
|
|
let cell_count = page.cell_count();
|
|
while cell_idx < cell_count {
|
|
match page
|
|
.cell_get(
|
|
cell_idx,
|
|
self.pager.clone(),
|
|
self.payload_overflow_threshold_max(page.page_type()),
|
|
self.payload_overflow_threshold_min(page.page_type()),
|
|
self.usable_space(),
|
|
)
|
|
.unwrap()
|
|
{
|
|
BTreeCell::TableLeafCell(cell) => {
|
|
if int_key <= cell._rowid {
|
|
break;
|
|
}
|
|
}
|
|
BTreeCell::TableInteriorCell(cell) => {
|
|
if int_key <= cell._rowid {
|
|
break;
|
|
}
|
|
}
|
|
_ => todo!(),
|
|
}
|
|
cell_idx += 1;
|
|
}
|
|
cell_idx
|
|
}
|
|
}
|
|
|
|
impl PageStack {
|
|
/// Push a new page onto the stack.
|
|
/// This effectively means traversing to a child page.
|
|
fn push(&self, page: PageRef) {
|
|
debug!(
|
|
"pagestack::push(current={}, new_page_id={})",
|
|
self.current_page.borrow(),
|
|
page.get().id
|
|
);
|
|
*self.current_page.borrow_mut() += 1;
|
|
let current = *self.current_page.borrow();
|
|
assert!(
|
|
current < BTCURSOR_MAX_DEPTH as i32,
|
|
"corrupted database, stack is bigger than expected"
|
|
);
|
|
self.stack.borrow_mut()[current as usize] = Some(page);
|
|
self.cell_indices.borrow_mut()[current as usize] = 0;
|
|
}
|
|
|
|
/// Pop a page off the stack.
|
|
/// This effectively means traversing back up to a parent page.
|
|
fn pop(&self) {
|
|
let current = *self.current_page.borrow();
|
|
debug!("pagestack::pop(current={})", current);
|
|
self.cell_indices.borrow_mut()[current as usize] = 0;
|
|
self.stack.borrow_mut()[current as usize] = None;
|
|
*self.current_page.borrow_mut() -= 1;
|
|
}
|
|
|
|
/// Get the top page on the stack.
|
|
/// This is the page that is currently being traversed.
|
|
fn top(&self) -> PageRef {
|
|
let current = *self.current_page.borrow();
|
|
let page = self.stack.borrow()[current as usize]
|
|
.as_ref()
|
|
.unwrap()
|
|
.clone();
|
|
debug!(
|
|
"pagestack::top(current={}, page_id={})",
|
|
current,
|
|
page.get().id
|
|
);
|
|
page
|
|
}
|
|
|
|
/// Get the parent page of the current page.
|
|
fn parent(&self) -> PageRef {
|
|
let current = *self.current_page.borrow();
|
|
self.stack.borrow()[current as usize - 1]
|
|
.as_ref()
|
|
.unwrap()
|
|
.clone()
|
|
}
|
|
|
|
/// Current page pointer being used
|
|
fn current(&self) -> usize {
|
|
*self.current_page.borrow() as usize
|
|
}
|
|
|
|
/// Cell index of the current page
|
|
fn current_cell_index(&self) -> i32 {
|
|
let current = self.current();
|
|
self.cell_indices.borrow()[current]
|
|
}
|
|
|
|
/// Check if the current cell index is less than 0.
|
|
/// This means we have been iterating backwards and have reached the start of the page.
|
|
fn current_cell_index_less_than_min(&self) -> bool {
|
|
let cell_idx = self.current_cell_index();
|
|
cell_idx < 0
|
|
}
|
|
|
|
/// Advance the current cell index of the current page to the next cell.
|
|
fn advance(&self) {
|
|
let current = self.current();
|
|
self.cell_indices.borrow_mut()[current] += 1;
|
|
}
|
|
|
|
fn retreat(&self) {
|
|
let current = self.current();
|
|
self.cell_indices.borrow_mut()[current] -= 1;
|
|
}
|
|
|
|
fn set_cell_index(&self, idx: i32) {
|
|
let current = self.current();
|
|
self.cell_indices.borrow_mut()[current] = idx
|
|
}
|
|
|
|
fn has_parent(&self) -> bool {
|
|
*self.current_page.borrow() > 0
|
|
}
|
|
|
|
fn clear(&self) {
|
|
*self.current_page.borrow_mut() = -1;
|
|
}
|
|
}
|
|
|
|
fn find_free_cell(page_ref: &PageContent, db_header: Ref<DatabaseHeader>, amount: usize) -> usize {
|
|
// NOTE: freelist is in ascending order of keys and pc
|
|
// unuse_space is reserved bytes at the end of page, therefore we must substract from maxpc
|
|
let mut pc = page_ref.first_freeblock() as usize;
|
|
|
|
let buf = page_ref.as_ptr();
|
|
|
|
let usable_space = (db_header.page_size - db_header.reserved_space as u16) as usize;
|
|
let maxpc = usable_space - amount;
|
|
let mut found = false;
|
|
while pc <= maxpc {
|
|
let next = u16::from_be_bytes(buf[pc..pc + 2].try_into().unwrap());
|
|
let size = u16::from_be_bytes(buf[pc + 2..pc + 4].try_into().unwrap());
|
|
if amount <= size as usize {
|
|
found = true;
|
|
break;
|
|
}
|
|
pc = next as usize;
|
|
}
|
|
if !found {
|
|
0
|
|
} else {
|
|
pc
|
|
}
|
|
}
|
|
|
|
impl Cursor for BTreeCursor {
|
|
fn seek_to_last(&mut self) -> Result<CursorResult<()>> {
|
|
return_if_io!(self.move_to_rightmost());
|
|
let (rowid, record) = return_if_io!(self.get_next_record(None));
|
|
if rowid.is_none() {
|
|
let is_empty = return_if_io!(self.is_empty_table());
|
|
assert!(is_empty);
|
|
return Ok(CursorResult::Ok(()));
|
|
}
|
|
self.rowid.replace(rowid);
|
|
self.record.replace(record);
|
|
Ok(CursorResult::Ok(()))
|
|
}
|
|
|
|
fn is_empty(&self) -> bool {
|
|
self.record.borrow().is_none()
|
|
}
|
|
|
|
fn root_page(&self) -> usize {
|
|
self.root_page
|
|
}
|
|
|
|
fn rewind(&mut self) -> Result<CursorResult<()>> {
|
|
self.move_to_root();
|
|
|
|
let (rowid, record) = return_if_io!(self.get_next_record(None));
|
|
self.rowid.replace(rowid);
|
|
self.record.replace(record);
|
|
Ok(CursorResult::Ok(()))
|
|
}
|
|
|
|
fn last(&mut self) -> Result<CursorResult<()>> {
|
|
match self.move_to_rightmost()? {
|
|
CursorResult::Ok(_) => self.prev(),
|
|
CursorResult::IO => Ok(CursorResult::IO),
|
|
}
|
|
}
|
|
|
|
fn next(&mut self) -> Result<CursorResult<()>> {
|
|
let (rowid, record) = return_if_io!(self.get_next_record(None));
|
|
self.rowid.replace(rowid);
|
|
self.record.replace(record);
|
|
Ok(CursorResult::Ok(()))
|
|
}
|
|
|
|
fn prev(&mut self) -> Result<CursorResult<()>> {
|
|
match self.get_prev_record()? {
|
|
CursorResult::Ok((rowid, record)) => {
|
|
self.rowid.replace(rowid);
|
|
self.record.replace(record);
|
|
Ok(CursorResult::Ok(()))
|
|
}
|
|
CursorResult::IO => Ok(CursorResult::IO),
|
|
}
|
|
}
|
|
|
|
fn wait_for_completion(&mut self) -> Result<()> {
|
|
// TODO: Wait for pager I/O to complete
|
|
Ok(())
|
|
}
|
|
|
|
fn rowid(&self) -> Result<Option<u64>> {
|
|
Ok(*self.rowid.borrow())
|
|
}
|
|
|
|
fn seek(&mut self, key: SeekKey<'_>, op: SeekOp) -> Result<CursorResult<bool>> {
|
|
let (rowid, record) = return_if_io!(self.seek(key, op));
|
|
self.rowid.replace(rowid);
|
|
self.record.replace(record);
|
|
Ok(CursorResult::Ok(rowid.is_some()))
|
|
}
|
|
|
|
fn record(&self) -> Result<Ref<Option<OwnedRecord>>> {
|
|
Ok(self.record.borrow())
|
|
}
|
|
|
|
fn insert(
|
|
&mut self,
|
|
key: &OwnedValue,
|
|
_record: &OwnedRecord,
|
|
moved_before: bool, /* Indicate whether it's necessary to traverse to find the leaf page */
|
|
) -> Result<CursorResult<()>> {
|
|
let int_key = match key {
|
|
OwnedValue::Integer(i) => i,
|
|
_ => unreachable!("btree tables are indexed by integers!"),
|
|
};
|
|
if !moved_before {
|
|
return_if_io!(self.move_to(SeekKey::TableRowId(*int_key as u64), SeekOp::EQ));
|
|
}
|
|
|
|
return_if_io!(self.insert_into_page(key, _record));
|
|
self.rowid.replace(Some(*int_key as u64));
|
|
Ok(CursorResult::Ok(()))
|
|
}
|
|
|
|
fn delete(&mut self) -> Result<CursorResult<()>> {
|
|
println!("rowid: {:?}", self.rowid.borrow());
|
|
Ok(CursorResult::Ok(()))
|
|
}
|
|
|
|
fn set_null_flag(&mut self, flag: bool) {
|
|
self.null_flag = flag;
|
|
}
|
|
|
|
fn get_null_flag(&self) -> bool {
|
|
self.null_flag
|
|
}
|
|
|
|
fn exists(&mut self, key: &OwnedValue) -> Result<CursorResult<bool>> {
|
|
let int_key = match key {
|
|
OwnedValue::Integer(i) => i,
|
|
_ => unreachable!("btree tables are indexed by integers!"),
|
|
};
|
|
return_if_io!(self.move_to(SeekKey::TableRowId(*int_key as u64), SeekOp::EQ));
|
|
let page = self.stack.top();
|
|
// TODO(pere): request load
|
|
return_if_locked!(page);
|
|
|
|
let contents = page.get().contents.as_ref().unwrap();
|
|
|
|
// find cell
|
|
let int_key = match key {
|
|
OwnedValue::Integer(i) => *i as u64,
|
|
_ => unreachable!("btree tables are indexed by integers!"),
|
|
};
|
|
let cell_idx = self.find_cell(contents, int_key);
|
|
if cell_idx >= contents.cell_count() {
|
|
Ok(CursorResult::Ok(false))
|
|
} else {
|
|
let equals = match &contents.cell_get(
|
|
cell_idx,
|
|
self.pager.clone(),
|
|
self.payload_overflow_threshold_max(contents.page_type()),
|
|
self.payload_overflow_threshold_min(contents.page_type()),
|
|
self.usable_space(),
|
|
)? {
|
|
BTreeCell::TableLeafCell(l) => l._rowid == int_key,
|
|
_ => unreachable!(),
|
|
};
|
|
Ok(CursorResult::Ok(equals))
|
|
}
|
|
}
|
|
|
|
fn btree_create(&mut self, flags: usize) -> u32 {
|
|
let page_type = match flags {
|
|
1 => PageType::TableLeaf,
|
|
2 => PageType::IndexLeaf,
|
|
_ => unreachable!(
|
|
"wrong create table falgs, should be 1 for table and 2 for index, got {}",
|
|
flags,
|
|
),
|
|
};
|
|
let page = self.allocate_page(page_type, 0);
|
|
let id = page.get().id;
|
|
id as u32
|
|
}
|
|
}
|
|
|
|
pub fn btree_init_page(
|
|
page: &PageRef,
|
|
page_type: PageType,
|
|
db_header: &DatabaseHeader,
|
|
offset: usize,
|
|
) {
|
|
// setup btree page
|
|
let contents = page.get();
|
|
debug!("btree_init_page(id={}, offset={})", contents.id, offset);
|
|
let contents = contents.contents.as_mut().unwrap();
|
|
contents.offset = offset;
|
|
let id = page_type as u8;
|
|
contents.write_u8(PAGE_HEADER_OFFSET_PAGE_TYPE, id);
|
|
contents.write_u16(PAGE_HEADER_OFFSET_FIRST_FREEBLOCK, 0);
|
|
contents.write_u16(PAGE_HEADER_OFFSET_CELL_COUNT, 0);
|
|
|
|
let cell_content_area_start = db_header.page_size - db_header.reserved_space as u16;
|
|
contents.write_u16(
|
|
PAGE_HEADER_OFFSET_CELL_CONTENT_AREA,
|
|
cell_content_area_start,
|
|
);
|
|
|
|
contents.write_u8(PAGE_HEADER_OFFSET_FRAGMENTED_BYTES_COUNT, 0);
|
|
contents.write_u32(PAGE_HEADER_OFFSET_RIGHTMOST_PTR, 0);
|
|
}
|
|
|
|
fn to_static_buf(buf: &[u8]) -> &'static [u8] {
|
|
unsafe { std::mem::transmute::<&[u8], &'static [u8]>(buf) }
|
|
}
|