Scalable Concurrent Containers

Cargo Crates.io GitHub Workflow Status

A collection of concurrent data structures and building blocks for concurrent programming.

EBR

The ebr module implements epoch-based reclamation and various types of auxiliary data structures to make use of it. Its epoch-based reclamation algorithm is similar to that implemented in crossbeam_epoch, however users may find it easier to use as the lifetime of an instance is automatically managed. For instance, ebr::AtomicArc and ebr::Arc hold a strong reference to the underlying instance, and the instance is passed to the garbage collector when the reference count drops to zero.

Examples

The ebr module can be used without an unsafe block.

```rust use scc::ebr::{Arc, AtomicArc, Barrier, Ptr, Tag}; use std::sync::atomic::Ordering::Relaxed;

// atomic_arc holds a strong reference to 17. let atomic_arc: AtomicArc = AtomicArc::new(17);

// barrier prevents the garbage collector from dropping reachable instances. let barrier: Barrier = Barrier::new();

// ptr cannot outlive barrier. let mut ptr: Ptr = atomicarc.load(Relaxed, &barrier); asserteq!(*ptr.as_ref().unwrap(), 17);

// atomic_arc can be tagged. atomicarc.settag(Tag::First, Relaxed);

// ptr is not tagged, so CAS fails. assert!(atomicarc.compareexchange( ptr, (Some(Arc::new(18)), Tag::First), Relaxed, Relaxed).is_err());

// ptr can be tagged. ptr.set_tag(Tag::First);

// The result of CAS is a handle to the instance that atomic_arc previously owned. let prev: Arc = atomicarc.compareexchange( ptr, (Some(Arc::new(18)), Tag::Second), Relaxed, Relaxed).unwrap().0.unwrap(); assert_eq!(*prev, 17);

// 17 will be garbage-collected later. drop(prev);

// ebr::AtomicArc can be converted into ebr::Arc. let arc: Arc = atomicarc.tryintoarc(Relaxed).unwrap(); asserteq!(*arc, 18);

// 18 will be garbage-collected later. drop(arc);

// 17 is still valid as barrier keeps the garbage collector from dropping it. asserteq!(*ptr.asref().unwrap(), 17); ```

HashMap

HashMap is a scalable in-memory unique key-value store that is targeted at highly concurrent heavy workloads. It applies EBR to its entry array management, thus allowing it to reduce the number of locks and avoid data sharding.

Examples

A unique key can be inserted along with its corresponding value, then it can be updated, read, and removed.

```rust use scc::HashMap;

let hashmap: HashMap = Default::default();

assert!(hashmap.insert(1, 0).isok()); asserteq!(hashmap.update(&1, |v| { *v = 2; *v }).unwrap(), 2); asserteq!(hashmap.read(&1, |, v| *v).unwrap(), 2); assert_eq!(hashmap.remove(&1).unwrap(), (1, 2)); ```

It supports upsert as in database management software; it tries to insert the given key-value pair, and if it fails, it updates the value field of an existing entry corresponding to the key.

```rust use scc::HashMap;

let hashmap: HashMap = Default::default();

hashmap.upsert(1, || 2, |, v| *v = 2); asserteq!(hashmap.read(&1, |, v| *v).unwrap(), 2); hashmap.upsert(1, || 2, |, v| *v = 3); asserteq!(hashmap.read(&1, |, v| *v).unwrap(), 3); ```

There is no method to confine the lifetime of references derived from an Iterator to the Iterator, and it is illegal for them to live as long as the HashMap due to the lack of global lock inside it. Therefore Iterator is not implemented, instead, it provides two methods that enable a HashMap to iterate over its entries: for_each, and retain.

```rust use scc::HashMap;

let hashmap: HashMap = Default::default();

assert!(hashmap.insert(1, 0).isok()); assert!(hashmap.insert(2, 1).isok());

// Inside for_each, a ebr::Barrier protects the entry array. let mut acc = 0; hashmap.foreach(|k, vmut| { acc += *k; *vmut = 2; }); asserteq!(acc, 3);

// for_each can modify the entries. asserteq!(hashmap.read(&1, |, v| *v).unwrap(), 2); asserteq!(hashmap.read(&2, |, v| *v).unwrap(), 2);

assert!(hashmap.insert(3, 2).is_ok());

// Inside retain, a ebr::Barrier protects the entry array. assert_eq!(hashmap.retain(|key, value| *key == 1 && *value == 0), (1, 2)); ```

HashIndex

HashIndex is a read-optimized version of HashMap. It applies EBR to its entry management as well, enabling it to perform read operations without acquiring locks.

Examples

Its read methods does not modify any shared data.

```rust use scc::HashIndex;

let hashindex: HashIndex = Default::default();

assert!(hashindex.insert(1, 0).isok()); asserteq!(hashindex.read(&1, |_, v| *v).unwrap(), 0); ```

An Iterator is implemented for HashIndex, because entry derived references can survive as long as the supplied ebr::Barrier survives.

```rust use scc::ebr::Barrier; use scc::HashIndex;

let hashindex: HashIndex = Default::default();

assert!(hashindex.insert(1, 0).is_ok());

let barrier = Barrier::new();

// An ebr::Barrier has to be given to iter. let mut iter = hashindex.iter(&barrier);

// The derived reference can live as long as barrier. let entryref = iter.next().unwrap(); asserteq!(iter.next(), None);

drop(hashindex);

// The entry can be read after hashindex is dropped. asserteq!(entryref, (&1, &0)); ```

TreeIndex

TreeIndex is a B+ tree variant optimized for read operations. The ebr module enables it to implement lock-free read and scan methods.

Examples

Key-value pairs can be inserted, read, and removed.

```rust use scc::TreeIndex;

let treeindex: TreeIndex = TreeIndex::new();

assert!(treeindex.insert(1, 10).isok()); asserteq!(treeindex.read(&1, |_, value| *value).unwrap(), 10); assert!(treeindex.remove(&1)); ```

Key-value pairs can be scanned.

```rust use scc::ebr::Barrier; use scc::TreeIndex;

let treeindex: TreeIndex = TreeIndex::new();

assert!(treeindex.insert(1, 10).isok()); assert!(treeindex.insert(2, 11).isok()); assert!(treeindex.insert(3, 13).is_ok());

let barrier = Barrier::new();

let mut visitor = treeindex.iter(&barrier); asserteq!(visitor.next().unwrap(), (&1, &10)); asserteq!(visitor.next().unwrap(), (&2, &11)); asserteq!(visitor.next().unwrap(), (&3, &13)); assert!(visitor.next().isnone()); ```

Key-value pairs in a specific range can be scanned.

```rust use scc::ebr::Barrier; use scc::TreeIndex;

let treeindex: TreeIndex = TreeIndex::new();

for i in 0..10 { assert!(treeindex.insert(i, 10).is_ok()); }

let barrier = Barrier::new();

asserteq!(treeindex.range(1..1, &barrier).count(), 0); asserteq!(treeindex.range(4..8, &barrier).count(), 4); assert_eq!(treeindex.range(4..=8, &barrier).count(), 5); ```

Performance

Test setup

Test data

Test workload: local

Test workload: local-remote

Test Results

| | 11 threads | 22 threads | 44 threads | |---------|------------|------------|------------| | InsertL | 186.564s | 192.279s | 258.713s | | ReadL | 102.086s | 108.321s | 117.399s | | ScanL | 35.874s | 63.281s | 134.607s | | RemoveL | 118.931s | 127.682s | 135.774s | | InsertR | 334.535s | 342.914s | 359.023s | | MixedR | 363.165s | 384.775s | 396.604s | | RemoveR | 260.543s | 277.702s | 302.858s |

| | 11 threads | 22 threads | 44 threads | |---------|------------|------------|------------| | InsertL | 4.172s | 4.693s | 5.947s | | ReadL | 2.291s | 2.337s | 2.647s | | ScanL | 0.703s | 1.459s | 3.04s | | RemoveL | 2.719s | 2.877s | 3.706s | | InsertR | 7.201s | 7.853s | 9.105s | | MixedR | 12.343s | 13.367s | 15.063s | | RemoveR | 8.726s | 9.357s | 10.94s |

Changelog

0.5.0