limbo/core/storage/mod.rs
2024-08-01 11:52:50 +03:00

89 lines
2.2 KiB
Rust

pub(crate) mod btree;
#[cfg(feature = "fs")]
use crate::io::File;
use crate::{error::LimboError, io::Completion, Buffer, Result};
use std::{cell::RefCell, rc::Rc};
pub struct PageSource {
io: Rc<dyn PageIO>,
}
impl Clone for PageSource {
fn clone(&self) -> Self {
Self {
io: self.io.clone(),
}
}
}
impl PageSource {
pub fn from_io(io: Rc<dyn PageIO>) -> Self {
Self { io }
}
#[cfg(feature = "fs")]
pub fn from_file(file: Rc<dyn File>) -> Self {
Self {
io: Rc::new(FileStorage::new(file)),
}
}
pub fn get(&self, page_idx: usize, c: Rc<Completion>) -> Result<()> {
self.io.get(page_idx, c)
}
pub fn write(
&self,
page_idx: usize,
buffer: Rc<RefCell<Buffer>>,
c: Rc<Completion>,
) -> Result<()> {
self.io.write(page_idx, buffer, c)
}
}
pub trait PageIO {
fn get(&self, page_idx: usize, c: Rc<Completion>) -> Result<()>;
fn write(&self, page_idx: usize, buffer: Rc<RefCell<Buffer>>, c: Rc<Completion>) -> Result<()>;
}
#[cfg(feature = "fs")]
struct FileStorage {
file: Rc<dyn crate::io::File>,
}
#[cfg(feature = "fs")]
impl PageIO for FileStorage {
fn get(&self, page_idx: usize, c: Rc<Completion>) -> Result<()> {
let r = match &(*c) {
Completion::Read(r) => r,
Completion::Write(_) => unreachable!(),
};
let size = r.buf().len();
assert!(page_idx > 0);
if size < 512 || size > 65536 || size & (size - 1) != 0 {
return Err(LimboError::NotADB.into());
}
let pos = (page_idx - 1) * size;
self.file.pread(pos, c)?;
Ok(())
}
fn write(&self, page_idx: usize, buffer: Rc<RefCell<Buffer>>, c: Rc<Completion>) -> Result<()> {
let buffer_size = buffer.borrow().len();
assert!(buffer_size >= 512);
assert!(buffer_size <= 65536);
assert!((buffer_size & (buffer_size - 1)) == 0);
let pos = (page_idx - 1) * buffer_size;
self.file.pwrite(pos, buffer, c)?;
Ok(())
}
}
#[cfg(feature = "fs")]
impl FileStorage {
pub fn new(file: Rc<dyn crate::io::File>) -> Self {
Self { file }
}
}