7 releases
0.4.0 |
|
---|---|
0.3.3 | Oct 30, 2023 |
0.3.2 | Sep 11, 2023 |
0.3.1 | Aug 27, 2022 |
0.1.0 | Mar 4, 2021 |
#286 in Embedded development
38 downloads per month
Used in 2 crates
(via memac)
24KB
495 lines
Synchronization Tools for no_std environments in Rust
MCS Lock
MCS lock is a fair and scalable mutual lock. This can be used as std::sync::Mutex.
use synctools::mcs;
use std::sync::Arc;
use std::vec::Vec;
const NUM_LOOP: usize = 10000000;
const NUM_THREADS: usize = 4;
fn main() {
// create a MCSLock object
let n = Arc::new(mcs::MCSLock::new(0));
let mut v = Vec::new();
for _ in 0..NUM_THREADS {
let n0 = n.clone();
let t = std::thread::spawn(move || {
let mut node = mcs::MCSNode::new();
for _ in 0..NUM_LOOP {
// lock and acquire the reference
let mut r = n0.lock(&mut node);
// increment atomically
*r += 1;
}
});
v.push(t);
}
for t in v {
t.join().unwrap();
}
let mut node = mcs::MCSNode::new();
let r = n.lock(&mut node);
assert_eq!(NUM_LOOP * NUM_THREADS, *r);
}
Readers Writer Lock
Spin lock based readers writer lock can be used as std::sync:RwLock.
use synctools::rwlock;
use std::sync::Arc;
use std::vec::Vec;
const NUM_LOOP: usize = 10000000;
const NUM_THREADS: usize = 4;
fn main() {
// create a RwLock object
let n = Arc::new(rwlock::RwLock::new(0));
let mut v = Vec::new();
// reader
for _ in 0..(NUM_THREADS - 1) {
let n0 = n.clone();
let t = std::thread::spawn(move || {
for _ in 0..NUM_LOOP {
// read lock
let r = n0.read();
assert_eq!(*r, 0);
}
});
v.push(t);
}
// writer
let n0 = n.clone();
let wr = std::thread::spawn(move || {
for _ in 0..NUM_LOOP {
{
// write lock
let mut r = n0.write();
*r += 1;
*r -= 1;
}
}
});
v.push(wr);
for t in v {
t.join().unwrap();
}
}
Lock Free Stack (AArch64 only)
Lock free stack is a concurrent data structure. This can be used only AArch64 and nightly because this uses LL/SC instructions in inline assembly internally.
use synctools::lfstack;
use std::sync::Arc;
use std::vec::Vec;
const NUM_LOOP: usize = 10000000;
const NUM_THREADS: usize = 4;
fn main() {
// create a stack
let mut stack = Arc::new(lfstack::LFStack::<usize>::new());
let mut v = Vec::new();
for i in 0..NUM_THREADS {
let mut stack0 = stack.clone();
let t = std::thread::spawn(move || {
if i & 1 == 0 { // even thread
for j in 0..NUM_LOOP {
let k = i * NUM_LOOP + j;
// push k to the stack
stack0.get_mut().push(k);
}
} else { // odd thread
for _ in 0..NUM_LOOP {
loop {
// pop from the stack
if let Some(k) = stack0.get_mut().pop() {
break;
}
}
}
}
});
v.push(t);
}
for t in v {
t.join().unwrap();
}
assert_eq!(stack.get_mut().pop(), None);
}
How to Test
Run
$ cargo +nightly test
for AArch64, or
$ cargo +nightly test --no-default-features
for non AArch64 environments.
Dependencies
~0–24MB
~330K SLoC