2 个版本
0.1.1 | 2022年2月10日 |
---|---|
0.1.0 | 2022年2月10日 |
#2 in #loom
6KB
loomy
一个用于通过 loom
容易测试代码的适配器包。
// std or loom, chosen at compile time by crate feature.
use loomy::{thread, cell::UnsafeCell};
struct Foo {
cell: UnsafeCell<...>,
}
#[test]
fn test_example() {
// When using `std`, `loomy::model` only invokes the closure and nothing
// more.
loomy::model(|| {
// ...
thread::spawn(|| {
// ...
});
// ...
});
}
使用 std
运行测试
$ cargo test
使用 loom
运行测试
$ cargo test --features loomy/enable
lib.rs
:
一个用于通过 loom
容易测试代码的适配器包。
从本库导入常用的类型和模块,如 UnsafeCell
、thread
、Arc
、AtomicI32
等,并在命令行中使用 cargo test --features loomy/enable
运行 loom 测试。
示例
以下模块可以通过两种方式测试
$ cargo test
$ cargo test --features loomy/enable
当 loomy/enable
被设置时,代码将作为 loomy 模型进行测试;否则,所有类型默认为它们的 std
等效类型,代码将按正常方式测试。
// Note the use of `loomy` instead of `std` or `loom`.
use loomy::{
hint,
cell::UnsafeCell,
sync::atomic::{AtomicBool, Ordering},
};
pub struct SpinLock<T> {
flag: AtomicBool,
data: UnsafeCell<T>,
}
unsafe impl<T> Send for SpinLock<T> {}
unsafe impl<T> Sync for SpinLock<T> {}
impl<T> SpinLock<T> {
pub fn new(t: T) -> Self {
Self {
flag: AtomicBool::new(false),
data: UnsafeCell::new(t),
}
}
pub fn with<R, F: FnOnce(&mut T) -> R>(&self, f: F) -> R {
while let Err(_) = self
.flag
.compare_exchange_weak(false, true, Ordering::Acquire, Ordering::Relaxed)
{
hint::spin_loop()
}
let out = self.data.with_mut(move |t| unsafe { f(&mut *t) });
self.flag.store(false, Ordering::Release);
out
}
}
#[cfg(test)]
mod tests {
// Also using `loomy` instead of `loom` or `std`.
use loomy::{thread, sync::Arc};
# mod tmp {
use super::*;
# }
#[test]
# fn mock() {}
fn test_simple() {
loomy::model(|| {
let lock = Arc::new(SpinLock::new(123));
let lock2 = Arc::clone(&lock);
let t = thread::spawn(move || {
lock2.with(|n| *n += 1);
});
lock.with(|n| *n = 456);
let out = lock.with(|n| *n);
t.join().unwrap();
assert!(out == 456 || out == 457);
});
}
# mod dummy {
}
关于 UnsafeCell
的说明
loom
中的 UnsafeCell
使用基于闭包的 API。当使用 std
类型时,UnsafeCell
被包裹以提供相同的 API。
依赖
~0–26MB
~328K SLoC