1 // Copyright 2021, The Android Open Source Project 2 // 3 // Licensed under the Apache License, Version 2.0 (the "License"); 4 // you may not use this file except in compliance with the License. 5 // You may obtain a copy of the License at 6 // 7 // http://www.apache.org/licenses/LICENSE-2.0 8 // 9 // Unless required by applicable law or agreed to in writing, software 10 // distributed under the License is distributed on an "AS IS" BASIS, 11 // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 12 // See the License for the specific language governing permissions and 13 // limitations under the License. 14 15 //! Synchronisation utilities. 16 17 use std::sync::{Arc, Condvar, Mutex}; 18 19 /// A flag which one thread can use to notify other threads when a condition becomes true. This is 20 /// something like a single-use binary semaphore. 21 #[derive(Clone, Debug)] 22 pub struct AtomicFlag { 23 state: Arc<(Mutex<bool>, Condvar)>, 24 } 25 26 impl Default for AtomicFlag { 27 #[allow(clippy::mutex_atomic)] 28 fn default() -> Self { 29 Self { state: Arc::new((Mutex::new(false), Condvar::new())) } 30 } 31 } 32 33 #[allow(clippy::mutex_atomic)] 34 impl AtomicFlag { 35 /// Wait until the flag is set. 36 pub fn wait(&self) { 37 let _flag = self.state.1.wait_while(self.state.0.lock().unwrap(), |flag| !*flag).unwrap(); 38 } 39 40 /// Set the flag, and notify all waiting threads. 41 pub fn raise(&self) { 42 let mut flag = self.state.0.lock().unwrap(); 43 *flag = true; 44 self.state.1.notify_all(); 45 } 46 } 47