1ef40d7f6Sopenharmony_ci//! Benchmark the overhead that the synchronization of `OnceCell::get` causes.
2ef40d7f6Sopenharmony_ci//! We do some other operations that write to memory to get an imprecise but somewhat realistic
3ef40d7f6Sopenharmony_ci//! measurement.
4ef40d7f6Sopenharmony_ci
5ef40d7f6Sopenharmony_ciuse once_cell::sync::OnceCell;
6ef40d7f6Sopenharmony_ciuse std::sync::atomic::{AtomicUsize, Ordering};
7ef40d7f6Sopenharmony_ci
8ef40d7f6Sopenharmony_ciconst N_THREADS: usize = 16;
9ef40d7f6Sopenharmony_ciconst N_ROUNDS: usize = 1_000_000;
10ef40d7f6Sopenharmony_ci
11ef40d7f6Sopenharmony_cistatic CELL: OnceCell<usize> = OnceCell::new();
12ef40d7f6Sopenharmony_cistatic OTHER: AtomicUsize = AtomicUsize::new(0);
13ef40d7f6Sopenharmony_ci
14ef40d7f6Sopenharmony_cifn main() {
15ef40d7f6Sopenharmony_ci    let start = std::time::Instant::now();
16ef40d7f6Sopenharmony_ci    let threads =
17ef40d7f6Sopenharmony_ci        (0..N_THREADS).map(|i| std::thread::spawn(move || thread_main(i))).collect::<Vec<_>>();
18ef40d7f6Sopenharmony_ci    for thread in threads {
19ef40d7f6Sopenharmony_ci        thread.join().unwrap();
20ef40d7f6Sopenharmony_ci    }
21ef40d7f6Sopenharmony_ci    println!("{:?}", start.elapsed());
22ef40d7f6Sopenharmony_ci    println!("{:?}", OTHER.load(Ordering::Relaxed));
23ef40d7f6Sopenharmony_ci}
24ef40d7f6Sopenharmony_ci
25ef40d7f6Sopenharmony_ci#[inline(never)]
26ef40d7f6Sopenharmony_cifn thread_main(i: usize) {
27ef40d7f6Sopenharmony_ci    // The operations we do here don't really matter, as long as we do multiple writes, and
28ef40d7f6Sopenharmony_ci    // everything is messy enough to prevent the compiler from optimizing the loop away.
29ef40d7f6Sopenharmony_ci    let mut data = [i; 128];
30ef40d7f6Sopenharmony_ci    let mut accum = 0usize;
31ef40d7f6Sopenharmony_ci    for _ in 0..N_ROUNDS {
32ef40d7f6Sopenharmony_ci        let _value = CELL.get_or_init(|| i + 1);
33ef40d7f6Sopenharmony_ci        let k = OTHER.fetch_add(data[accum & 0x7F] as usize, Ordering::Relaxed);
34ef40d7f6Sopenharmony_ci        for j in data.iter_mut() {
35ef40d7f6Sopenharmony_ci            *j = (*j).wrapping_add(accum);
36ef40d7f6Sopenharmony_ci            accum = accum.wrapping_add(k);
37ef40d7f6Sopenharmony_ci        }
38ef40d7f6Sopenharmony_ci    }
39ef40d7f6Sopenharmony_ci}
40