2022-05-07 11:34:18 -05:00
|
|
|
// ignore-windows: Concurrency on Windows is not supported yet.
|
|
|
|
// compile-flags: -Zmiri-ignore-leaks
|
|
|
|
|
|
|
|
// https://plv.mpi-sws.org/scfix/paper.pdf
|
|
|
|
// 2.2 Second Problem: SC Fences are Too Weak
|
|
|
|
// This test should pass under the C++20 model Rust is using.
|
|
|
|
// Unfortunately, Miri's weak memory emulation only follows C++11 model
|
|
|
|
// as we don't know how to correctly emulate C++20's revised SC semantics,
|
|
|
|
// so we have to stick to C++11 emulation from exiting research.
|
|
|
|
|
|
|
|
use std::sync::atomic::Ordering::*;
|
2022-05-17 14:04:18 -05:00
|
|
|
use std::thread::spawn;
|
2022-05-07 11:34:18 -05:00
|
|
|
use std::sync::atomic::{fence, AtomicUsize};
|
|
|
|
|
2022-05-17 14:04:18 -05:00
|
|
|
// Spins until it reads value
|
2022-05-07 11:34:18 -05:00
|
|
|
fn reads_value(loc: &AtomicUsize, val: usize) -> usize {
|
|
|
|
while loc.load(Relaxed) != val {
|
2022-05-17 14:04:18 -05:00
|
|
|
std::hint::spin_loop();
|
2022-05-07 11:34:18 -05:00
|
|
|
}
|
|
|
|
val
|
|
|
|
}
|
|
|
|
|
|
|
|
// We can't create static items because we need to run each test
|
|
|
|
// multiple tests
|
|
|
|
fn static_atomic(val: usize) -> &'static AtomicUsize {
|
|
|
|
let ret = Box::leak(Box::new(AtomicUsize::new(val)));
|
|
|
|
// A workaround to put the initialisation value in the store buffer
|
|
|
|
ret.store(val, Relaxed);
|
|
|
|
ret
|
|
|
|
}
|
|
|
|
|
|
|
|
fn test_cpp20_rwc_syncs() {
|
|
|
|
/*
|
|
|
|
int main() {
|
|
|
|
atomic_int x = 0;
|
|
|
|
atomic_int y = 0;
|
|
|
|
|
|
|
|
{{{ x.store(1,mo_relaxed);
|
|
|
|
||| { r1=x.load(mo_relaxed).readsvalue(1);
|
|
|
|
fence(mo_seq_cst);
|
|
|
|
r2=y.load(mo_relaxed); }
|
|
|
|
||| { y.store(1,mo_relaxed);
|
|
|
|
fence(mo_seq_cst);
|
|
|
|
r3=x.load(mo_relaxed); }
|
|
|
|
}}}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
*/
|
|
|
|
let x = static_atomic(0);
|
|
|
|
let y = static_atomic(0);
|
|
|
|
|
|
|
|
let j1 = spawn(move || {
|
|
|
|
x.store(1, Relaxed);
|
|
|
|
});
|
|
|
|
|
|
|
|
let j2 = spawn(move || {
|
|
|
|
reads_value(&x, 1);
|
|
|
|
fence(SeqCst);
|
|
|
|
y.load(Relaxed)
|
|
|
|
});
|
|
|
|
|
|
|
|
let j3 = spawn(move || {
|
|
|
|
y.store(1, Relaxed);
|
|
|
|
fence(SeqCst);
|
|
|
|
x.load(Relaxed)
|
|
|
|
});
|
|
|
|
|
|
|
|
j1.join().unwrap();
|
|
|
|
let b = j2.join().unwrap();
|
|
|
|
let c = j3.join().unwrap();
|
|
|
|
|
|
|
|
if (b, c) == (0, 0) {
|
|
|
|
// FIXME: the standalone compiletest-rs needs to support
|
|
|
|
// failure-status header to allow us to write assert_ne!((b, c), (0, 0))
|
|
|
|
// https://rustc-dev-guide.rust-lang.org/tests/headers.html#miscellaneous-headers
|
|
|
|
// because panic exits with 101 but compile-rs expects 1
|
|
|
|
let _ = unsafe { std::mem::MaybeUninit::<*const u32>::uninit().assume_init() }; //~ ERROR uninitialized
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn main() {
|
|
|
|
for _ in 0..500 {
|
|
|
|
test_cpp20_rwc_syncs();
|
|
|
|
}
|
|
|
|
}
|