//! A concurrency library for high concurrency reads.
//!
//! This library is named after the 2 (identical) tables that we hold
//! internally:
//! - Active - this is the table that all Readers view. This table will never be
//! write locked, so readers never face contention.
//! - Standby - this is the table the the Writer mutates. A writer should face
//! minimal contention retrieving this table for mutation since Readers move
//! to the Active table when the tables are swapped.
//!
//! The cost of providing no contention to readers, and minimal contention to
//! writers is:
//! 1. Memory - Internally we hold 2 copies of the underlying type the user
//! created. This is needed to allow there to always be a table that Readers
//! can check out without contention.
//! 2. Writer thread CPU usage - The writer must apply all updates twice, once
//! to each table. Lock contention for the writer should be less than with a
//! plain RwLock due to Readers using the activetable.
//!
//! The usage is meant to be similar to a RwLock. Some of the inspiration came
//! from the leftright crate, so feel
//! free to check that out. The main differences focus on trying to simplify the
//! client (creating data structures) and user (using data structures)
//! experiences; primarily focused on trying to mimic the API/usage of an
//! RwLock.
//!
//! There are 2 flavors of this algorithm that we offer:
//! 1. Lockless - this variant trades off increased performance against changing
//! the API to be less like an RwLock. This avoids the cost of performing
//! synchronization on reads, but this requires that each thread/task that is
//! going to access the tables, registers in advance. Therefore this centers
//! around the AsLockHandle, which is conceptually similar to Arcrust
//! use std::thread::sleep;
//! use std::time::Duration;
//! use std::sync::Arc;
//! use active_standby::primitives::UpdateTables;
//!
//! // Client's must implement the mutable interface that they want to offer users
//! // of their active standby data structure. This is not automatically generated.
//! struct AddOne {}
//! impl<'a> UpdateTables<'a, i32, ()> for AddOne {
//! fn apply_first(&mut self, table: &'a mut i32) {
//! *table = *table + 1;
//! }
//! fn apply_second(mut self, table: &mut i32) {
//! self.apply_first(table);
//! }
//! }
//!
//! pub mod lockless {
//! active_standby::generate_lockless_aslockhandle!(i32);
//!
//! impl<'w> WriteGuard<'w> {
//! pub fn add_one(&mut self) {
//! self.guard.update_tables(super::AddOne {})
//! }
//! }
//! }
//!
//! pub mod shared {
//! active_standby::generate_shared_aslock!(i32);
//!
//! impl<'w> WriteGuard<'w> {
//! pub fn add_one(&mut self) {
//! self.guard.update_tables(super::AddOne {})
//! }
//! }
//! }
//!
//! fn run_lockless() {
//! let table = lockless::AsLockHandle::new(0);
//! let table2 = table.clone();
//! let handle = std::thread::spawn(move || {
//! while *table2.read() != 1 {
//! sleep(Duration::from_micros(100));
//! }
//! });
//!
//! {
//! let mut wg = table.write();
//! wg.add_one();
//! }
//! handle.join();
//! }
//!
//! fn run_shared() {
//! let table = Arc::new(shared::AsLock::new(0));
//! let table2 = Arc::clone(&table);
//! let handle = std::thread::spawn(move || {
//! while *table2.read() != 1 {
//! sleep(Duration::from_micros(100));
//! }
//! });
//!
//! {
//! let mut wg = table.write();
//! wg.add_one();
//! }
//! handle.join();
//! }
//!
//! fn main() {
//! run_lockless();
//! run_shared();
//! }
//!
//!
//! If your table has large elements, you may want to save memory by only
//! holding each element once (e.g. vec::AsLockHandle