pub mod error;
pub mod leaf;
pub mod leafnode;
pub mod node;
use crossbeam_epoch::{Atomic, Guard, Owned};
use error::{InsertError, RemoveError, SearchError};
use leaf::{Leaf, LeafScanner};
use node::Node;
use std::fmt;
use std::sync::atomic::Ordering::{Acquire, Relaxed};
pub struct TreeIndex<K, V>
where
K: Clone + Ord + Send + Sync,
V: Clone + Send + Sync,
{
root: Atomic<Node<K, V>>,
}
impl<K, V> Default for TreeIndex<K, V>
where
K: Clone + Ord + Send + Sync,
V: Clone + Send + Sync,
{
fn default() -> Self {
TreeIndex::new()
}
}
impl<K, V> TreeIndex<K, V>
where
K: Clone + Ord + Send + Sync,
V: Clone + Send + Sync,
{
pub fn new() -> TreeIndex<K, V> {
TreeIndex {
root: Atomic::null(),
}
}
pub fn insert(&self, mut key: K, mut value: V) -> Result<(), (K, V)> {
loop {
let guard = crossbeam_epoch::pin();
let mut root_node = self.root.load(Acquire, &guard);
if root_node.is_null() {
let new_root = Owned::new(Node::new(0, true));
match self
.root
.compare_and_set(root_node, new_root, Relaxed, &guard)
{
Ok(new_root) => root_node = new_root,
Err(_) => continue,
}
}
let root_node_ref = unsafe { root_node.deref() };
match root_node_ref.insert(key, value, &guard) {
Ok(_) => return Ok(()),
Err(error) => match error {
InsertError::Duplicated(entry) => return Err(entry),
InsertError::Full(entry) => {
root_node_ref.split_root(&self.root, &guard);
key = entry.0;
value = entry.1;
}
InsertError::Retry(entry) => {
key = entry.0;
value = entry.1;
}
},
}
}
}
pub fn remove(&self, key: &K) -> bool {
let mut has_been_removed = false;
let guard = crossbeam_epoch::pin();
let mut root_node = self.root.load(Acquire, &guard);
loop {
if root_node.is_null() {
return has_been_removed;
}
let root_node_ref = unsafe { root_node.deref() };
match root_node_ref.remove(key, &guard) {
Ok(removed) => return removed || has_been_removed,
Err(remove_error) => match remove_error {
RemoveError::Coalesce(removed) => {
if removed && !has_been_removed {
has_been_removed = true;
}
Node::update_root(root_node, &self.root, &guard);
return has_been_removed;
}
RemoveError::Retry(removed) => {
if removed && !has_been_removed {
has_been_removed = true;
}
}
},
};
let root_node_new = self.root.load(Acquire, &guard);
root_node = root_node_new;
}
}
pub fn read<R, F: FnOnce(&K, &V) -> R>(&self, key: &K, f: F) -> Option<R> {
let guard = crossbeam_epoch::pin();
loop {
let root_node = self.root.load(Acquire, &guard);
if root_node.is_null() {
return None;
}
match unsafe { root_node.deref().search(key, &guard) } {
Ok(result) => {
if let Some(value) = result {
return Some(f(key, value));
} else {
return None;
}
}
Err(err) => match err {
SearchError::Retry => continue,
},
}
}
}
pub fn clear(&self) {
let guard = crossbeam_epoch::pin();
Node::remove_root(&self.root, &guard);
}
pub fn len(&self) -> usize {
self.iter().count()
}
pub fn depth(&self) -> usize {
let guard = crossbeam_epoch::pin();
let root_node = self.root.load(Acquire, &guard);
if !root_node.is_null() {
unsafe { root_node.deref().floor() + 1 }
} else {
0
}
}
pub fn iter(&self) -> Scanner<K, V> {
Scanner::new(self)
}
pub fn from(&self, key: &K) -> Scanner<K, V> {
Scanner::from(self, key)
}
}
impl<K, V> TreeIndex<K, V>
where
K: Clone + fmt::Display + Ord + Send + Sync,
V: Clone + fmt::Display + Send + Sync,
{
pub fn print<T: std::io::Write>(&self, output: &mut T) -> std::io::Result<()> {
output.write_fmt(format_args!("digraph {{\n"))?;
let guard = crossbeam_epoch::pin();
let root_node = self.root.load(Acquire, &guard);
if !root_node.is_null() {
unsafe { root_node.deref().print(output, &guard) }?
}
output.write_fmt(format_args!("}}"))
}
}
impl<K, V> Drop for TreeIndex<K, V>
where
K: Clone + Ord + Send + Sync,
V: Clone + Send + Sync,
{
fn drop(&mut self) {
let guard = unsafe { crossbeam_epoch::unprotected() };
Node::remove_root(&self.root, guard);
}
}
pub struct Scanner<'t, K, V>
where
K: Clone + Ord + Send + Sync,
V: Clone + Send + Sync,
{
tree: &'t TreeIndex<K, V>,
leaf_scanner: Option<LeafScanner<'t, K, V>>,
from_iterator: bool,
guard: Guard,
}
impl<'t, K: Clone + Ord + Send + Sync, V: Clone + Send + Sync> Scanner<'t, K, V> {
fn new(tree: &'t TreeIndex<K, V>) -> Scanner<'t, K, V> {
Scanner::<'t, K, V> {
tree,
leaf_scanner: None,
from_iterator: false,
guard: crossbeam_epoch::pin(),
}
}
fn from(tree: &'t TreeIndex<K, V>, min_allowed_key: &K) -> Scanner<'t, K, V> {
let mut scanner = Scanner::<'t, K, V> {
tree,
leaf_scanner: None,
from_iterator: false,
guard: crossbeam_epoch::pin(),
};
loop {
let root_node = tree.root.load(Acquire, &scanner.guard);
if root_node.is_null() {
scanner.from_iterator = true;
return scanner;
}
if let Ok(leaf_scanner) =
unsafe { &*root_node.as_raw() }.max_less(min_allowed_key, &scanner.guard)
{
scanner.leaf_scanner.replace(unsafe {
std::mem::transmute::<_, LeafScanner<'t, K, V>>(leaf_scanner)
});
break;
}
}
while let Some((key_ref, _)) = scanner.next() {
if key_ref.cmp(min_allowed_key) != std::cmp::Ordering::Less {
scanner.from_iterator = true;
return scanner;
}
}
scanner.leaf_scanner.take();
scanner.from_iterator = true;
scanner
}
}
impl<'t, K, V> Iterator for Scanner<'t, K, V>
where
K: Clone + Ord + Send + Sync,
V: Clone + Send + Sync,
{
type Item = (&'t K, &'t V);
fn next(&mut self) -> Option<Self::Item> {
if self.from_iterator {
self.from_iterator = false;
if let Some(leaf_scanner) = self.leaf_scanner.as_ref() {
return leaf_scanner.get();
}
return None;
}
if self.leaf_scanner.is_none() {
loop {
let root_node = self.tree.root.load(Acquire, &self.guard);
if root_node.is_null() {
return None;
}
if let Ok(leaf_scanner) = unsafe { &*root_node.as_raw() }.min(&self.guard) {
self.leaf_scanner.replace(unsafe {
std::mem::transmute::<_, LeafScanner<'t, K, V>>(leaf_scanner)
});
break;
}
}
}
if let Some(mut scanner) = self.leaf_scanner.take() {
let min_allowed_key = scanner.get().map(|(key, _)| key);
if let Some(result) = scanner.next() {
self.leaf_scanner.replace(scanner);
return Some(result);
}
while let Some(mut new_scanner) = unsafe {
std::mem::transmute::<_, Option<LeafScanner<'t, K, V>>>(scanner.jump(&self.guard))
.take()
} {
while let Some(entry) = new_scanner.next() {
if min_allowed_key
.as_ref()
.map_or_else(|| true, |&key| key.cmp(entry.0) == std::cmp::Ordering::Less)
{
self.leaf_scanner.replace(new_scanner);
return Some(entry);
}
}
scanner = new_scanner;
}
}
None
}
}