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