rust/tests/pass/atomic.rs

117 lines
4.3 KiB
Rust
Raw Normal View History

2020-01-25 13:48:26 -06:00
use std::sync::atomic::{compiler_fence, fence, AtomicBool, AtomicIsize, AtomicU64, Ordering::*};
2017-03-14 13:05:51 +01:00
fn main() {
2019-08-02 19:50:54 +02:00
atomic_bool();
atomic_isize();
2019-08-02 20:08:04 +02:00
atomic_u64();
2019-10-05 08:09:23 -03:00
atomic_fences();
2021-01-25 02:41:03 -08:00
weak_sometimes_fails();
2019-08-02 19:50:54 +02:00
}
fn atomic_bool() {
static mut ATOMIC: AtomicBool = AtomicBool::new(false);
unsafe {
assert_eq!(*ATOMIC.get_mut(), false);
ATOMIC.store(true, SeqCst);
assert_eq!(*ATOMIC.get_mut(), true);
ATOMIC.fetch_or(false, SeqCst);
assert_eq!(*ATOMIC.get_mut(), true);
ATOMIC.fetch_and(false, SeqCst);
assert_eq!(*ATOMIC.get_mut(), false);
ATOMIC.fetch_nand(true, SeqCst);
assert_eq!(*ATOMIC.get_mut(), true);
ATOMIC.fetch_xor(true, SeqCst);
assert_eq!(*ATOMIC.get_mut(), false);
}
}
// There isn't a trait to use to make this generic, so just use a macro
macro_rules! compare_exchange_weak_loop {
($atom:expr, $from:expr, $to:expr, $succ_order:expr, $fail_order:expr) => {
loop {
match $atom.compare_exchange_weak($from, $to, $succ_order, $fail_order) {
Ok(n) => {
assert_eq!(n, $from);
break;
}
Err(n) => assert_eq!(n, $from),
}
}
};
}
2019-08-02 19:50:54 +02:00
fn atomic_isize() {
static ATOMIC: AtomicIsize = AtomicIsize::new(0);
2017-03-14 13:05:51 +01:00
// Make sure trans can emit all the intrinsics correctly
assert_eq!(ATOMIC.compare_exchange(0, 1, Relaxed, Relaxed), Ok(0));
assert_eq!(ATOMIC.compare_exchange(0, 2, Acquire, Relaxed), Err(1));
assert_eq!(ATOMIC.compare_exchange(0, 1, Release, Relaxed), Err(1));
assert_eq!(ATOMIC.compare_exchange(1, 0, AcqRel, Relaxed), Ok(1));
2017-03-14 13:05:51 +01:00
ATOMIC.compare_exchange(0, 1, SeqCst, Relaxed).ok();
ATOMIC.compare_exchange(0, 1, Acquire, Acquire).ok();
ATOMIC.compare_exchange(0, 1, AcqRel, Acquire).ok();
ATOMIC.compare_exchange(0, 1, SeqCst, Acquire).ok();
ATOMIC.compare_exchange(0, 1, SeqCst, SeqCst).ok();
ATOMIC.store(0, SeqCst);
compare_exchange_weak_loop!(ATOMIC, 0, 1, Relaxed, Relaxed);
assert_eq!(ATOMIC.compare_exchange_weak(0, 2, Acquire, Relaxed), Err(1));
assert_eq!(ATOMIC.compare_exchange_weak(0, 1, Release, Relaxed), Err(1));
compare_exchange_weak_loop!(ATOMIC, 1, 0, AcqRel, Relaxed);
assert_eq!(ATOMIC.load(Relaxed), 0);
2017-03-14 13:05:51 +01:00
ATOMIC.compare_exchange_weak(0, 1, AcqRel, Relaxed).ok();
ATOMIC.compare_exchange_weak(0, 1, SeqCst, Relaxed).ok();
ATOMIC.compare_exchange_weak(0, 1, Acquire, Acquire).ok();
ATOMIC.compare_exchange_weak(0, 1, AcqRel, Acquire).ok();
ATOMIC.compare_exchange_weak(0, 1, SeqCst, Acquire).ok();
ATOMIC.compare_exchange_weak(0, 1, SeqCst, SeqCst).ok();
}
2019-08-02 20:08:04 +02:00
fn atomic_u64() {
static ATOMIC: AtomicU64 = AtomicU64::new(0);
ATOMIC.store(1, SeqCst);
assert_eq!(ATOMIC.compare_exchange(0, 0x100, AcqRel, Acquire), Err(1));
compare_exchange_weak_loop!(ATOMIC, 1, 0x100, AcqRel, Acquire);
2019-08-02 20:08:04 +02:00
assert_eq!(ATOMIC.load(Relaxed), 0x100);
assert_eq!(ATOMIC.fetch_max(0x10, SeqCst), 0x100);
assert_eq!(ATOMIC.fetch_max(0x100, SeqCst), 0x100);
assert_eq!(ATOMIC.fetch_max(0x1000, SeqCst), 0x100);
assert_eq!(ATOMIC.fetch_max(0x1000, SeqCst), 0x1000);
assert_eq!(ATOMIC.fetch_max(0x2000, SeqCst), 0x1000);
assert_eq!(ATOMIC.fetch_max(0x2000, SeqCst), 0x2000);
assert_eq!(ATOMIC.fetch_min(0x2000, SeqCst), 0x2000);
assert_eq!(ATOMIC.fetch_min(0x2000, SeqCst), 0x2000);
assert_eq!(ATOMIC.fetch_min(0x1000, SeqCst), 0x2000);
assert_eq!(ATOMIC.fetch_min(0x1000, SeqCst), 0x1000);
assert_eq!(ATOMIC.fetch_min(0x100, SeqCst), 0x1000);
assert_eq!(ATOMIC.fetch_min(0x10, SeqCst), 0x100);
2019-08-02 20:08:04 +02:00
}
2019-10-05 08:09:23 -03:00
fn atomic_fences() {
fence(SeqCst);
fence(Release);
fence(Acquire);
2019-10-09 10:29:08 +02:00
fence(AcqRel);
2020-01-25 13:48:26 -06:00
compiler_fence(SeqCst);
compiler_fence(Release);
compiler_fence(Acquire);
compiler_fence(AcqRel);
2019-10-05 08:09:23 -03:00
}
2021-01-25 02:41:03 -08:00
fn weak_sometimes_fails() {
2021-01-25 02:52:55 -08:00
let atomic = AtomicBool::new(false);
let tries = 100;
2021-01-25 02:41:03 -08:00
for _ in 0..tries {
2021-01-25 02:52:55 -08:00
let cur = atomic.load(Relaxed);
2021-01-25 02:41:03 -08:00
// Try (weakly) to flip the flag.
2021-01-25 02:52:55 -08:00
if atomic.compare_exchange_weak(cur, !cur, Relaxed, Relaxed).is_err() {
// We failed, so return and skip the panic.
2021-01-25 02:41:03 -08:00
return;
}
}
panic!("compare_exchange_weak succeeded {} tries in a row", tries);
}