/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ /* vim: set ts=8 sts=2 et sw=2 tw=80: */ /* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ #ifndef threading_ConditionVariable_h #define threading_ConditionVariable_h #include "mozilla/PlatformConditionVariable.h" #include "mozilla/TimeStamp.h" #include #include #if !defined(XP_WIN) && !defined(__wasi__) # include #endif #include "threading/LockGuard.h" #include "threading/Mutex.h" namespace js { template class ExclusiveData; enum class CVStatus { NoTimeout, Timeout }; template using UniqueLock = LockGuard; // A poly-fill for std::condition_variable. class ConditionVariable { public: struct PlatformData; ConditionVariable() = default; ~ConditionVariable() = default; // Wake one thread that is waiting on this condition. void notify_one() { impl_.notify_one(); } // Wake all threads that are waiting on this condition. void notify_all() { impl_.notify_all(); } // Block the current thread of execution until this condition variable is // woken from another thread via notify_one or notify_all. void wait(Mutex& lock) { #ifdef DEBUG lock.preUnlockChecks(); #endif impl_.wait(lock.impl_); #ifdef DEBUG lock.preLockChecks(); lock.postLockChecks(); #endif } void wait(UniqueLock& lock) { wait(lock.lock); } // As with |wait|, block the current thread of execution until woken from // another thread. This method will resume waiting once woken until the given // Predicate |pred| evaluates to true. template void wait(UniqueLock& lock, Predicate pred) { while (!pred()) { wait(lock); } } // Block the current thread of execution until woken from another thread, or // the given absolute time is reached. The given absolute time is evaluated // when this method is called, so will wake up after (abs_time - now), // independent of system clock changes. While insulated from clock changes, // this API is succeptible to the issues discussed above wait_for. CVStatus wait_until(UniqueLock& lock, const mozilla::TimeStamp& abs_time) { return wait_for(lock, abs_time - mozilla::TimeStamp::Now()); } // As with |wait_until|, block the current thread of execution until woken // from another thread, or the given absolute time is reached. This method // will resume waiting once woken until the given Predicate |pred| evaluates // to true. template bool wait_until(UniqueLock& lock, const mozilla::TimeStamp& abs_time, Predicate pred) { while (!pred()) { if (wait_until(lock, abs_time) == CVStatus::Timeout) { return pred(); } } return true; } // Block the current thread of execution until woken from another thread, or // the given time duration has elapsed. Given that the system may be // interrupted between the callee and the actual wait beginning, this call // has a minimum granularity of the system's scheduling interval, and may // encounter substantially longer delays, depending on system load. CVStatus wait_for(UniqueLock& lock, const mozilla::TimeDuration& rel_time) { #ifdef DEBUG lock.lock.preUnlockChecks(); #endif CVStatus res = impl_.wait_for(lock.lock.impl_, rel_time) == mozilla::CVStatus::Timeout ? CVStatus::Timeout : CVStatus::NoTimeout; #ifdef DEBUG lock.lock.preLockChecks(); lock.lock.postLockChecks(); #endif return res; } // As with |wait_for|, block the current thread of execution until woken from // another thread or the given time duration has elapsed. This method will // resume waiting once woken until the given Predicate |pred| evaluates to // true. template bool wait_for(UniqueLock& lock, const mozilla::TimeDuration& rel_time, Predicate pred) { return wait_until(lock, mozilla::TimeStamp::Now() + rel_time, std::move(pred)); } private: ConditionVariable(const ConditionVariable&) = delete; ConditionVariable& operator=(const ConditionVariable&) = delete; template friend class ExclusiveWaitableData; mozilla::detail::ConditionVariableImpl impl_; }; } // namespace js #endif // threading_ConditionVariable_h