2022-07-17 21:54:10 -04:00
|
|
|
//@ignore-target-windows: Concurrency on Windows is not supported yet.
|
2022-06-06 17:44:16 -04:00
|
|
|
// This specifically tests behavior *without* preemption.
|
2022-07-08 16:08:32 +00:00
|
|
|
//@compile-flags: -Zmiri-preemption-rate=0
|
2022-06-05 14:20:22 -04:00
|
|
|
|
2022-06-20 15:30:34 -07:00
|
|
|
use std::cell::Cell;
|
2022-06-05 14:31:44 -04:00
|
|
|
use std::sync::atomic::{AtomicBool, AtomicUsize, Ordering};
|
|
|
|
use std::sync::mpsc;
|
2022-06-20 15:30:34 -07:00
|
|
|
use std::thread;
|
2022-06-05 14:20:22 -04:00
|
|
|
|
2022-06-05 14:31:44 -04:00
|
|
|
/// When a thread yields, Miri's scheduler used to pick the thread with the lowest ID
|
|
|
|
/// that can run. IDs are assigned in thread creation order.
|
|
|
|
/// This means we could make 2 threads infinitely ping-pong with each other while
|
|
|
|
/// really there is a 3rd thread that we should schedule to make progress.
|
|
|
|
fn two_player_ping_pong() {
|
|
|
|
static FLAG: AtomicUsize = AtomicUsize::new(0);
|
2022-06-05 14:20:22 -04:00
|
|
|
|
|
|
|
let waiter1 = thread::spawn(|| {
|
|
|
|
while FLAG.load(Ordering::Acquire) == 0 {
|
|
|
|
// spin and wait
|
|
|
|
thread::yield_now();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
let waiter2 = thread::spawn(|| {
|
|
|
|
while FLAG.load(Ordering::Acquire) == 0 {
|
|
|
|
// spin and wait
|
|
|
|
thread::yield_now();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
let progress = thread::spawn(|| {
|
|
|
|
FLAG.store(1, Ordering::Release);
|
|
|
|
});
|
|
|
|
// The first `join` blocks the main thread and thus takes it out of the equation.
|
|
|
|
waiter1.join().unwrap();
|
|
|
|
waiter2.join().unwrap();
|
|
|
|
progress.join().unwrap();
|
|
|
|
}
|
2022-06-05 14:31:44 -04:00
|
|
|
|
|
|
|
/// Based on a test by @jethrogb.
|
|
|
|
fn launcher() {
|
|
|
|
static THREAD2_LAUNCHED: AtomicBool = AtomicBool::new(false);
|
|
|
|
|
|
|
|
for _ in 0..10 {
|
|
|
|
let (tx, rx) = mpsc::sync_channel(0);
|
|
|
|
THREAD2_LAUNCHED.store(false, Ordering::SeqCst);
|
|
|
|
|
|
|
|
let jh = thread::spawn(move || {
|
|
|
|
struct RecvOnDrop(Cell<Option<mpsc::Receiver<()>>>);
|
|
|
|
|
|
|
|
impl Drop for RecvOnDrop {
|
|
|
|
fn drop(&mut self) {
|
|
|
|
let rx = self.0.take().unwrap();
|
|
|
|
while !THREAD2_LAUNCHED.load(Ordering::SeqCst) {
|
|
|
|
thread::yield_now();
|
|
|
|
}
|
|
|
|
rx.recv().unwrap();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
let tl_rx: RecvOnDrop = RecvOnDrop(Cell::new(None));
|
|
|
|
tl_rx.0.set(Some(rx));
|
|
|
|
});
|
|
|
|
|
|
|
|
let tx_clone = tx.clone();
|
|
|
|
let jh2 = thread::spawn(move || {
|
|
|
|
THREAD2_LAUNCHED.store(true, Ordering::SeqCst);
|
|
|
|
jh.join().unwrap();
|
|
|
|
tx_clone.send(()).expect_err(
|
|
|
|
"Expecting channel to be closed because thread 1 TLS destructors must've run",
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
while !THREAD2_LAUNCHED.load(Ordering::SeqCst) {
|
|
|
|
thread::yield_now();
|
|
|
|
}
|
|
|
|
thread::yield_now();
|
|
|
|
tx.send(()).expect("Expecting channel to be live because thread 2 must block on join");
|
|
|
|
jh2.join().unwrap();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn main() {
|
|
|
|
two_player_ping_pong();
|
|
|
|
launcher();
|
|
|
|
}
|