summaryrefslogtreecommitdiffstats
path: root/js/src/threading/ConditionVariable.h
diff options
context:
space:
mode:
Diffstat (limited to 'js/src/threading/ConditionVariable.h')
-rw-r--r--js/src/threading/ConditionVariable.h138
1 files changed, 138 insertions, 0 deletions
diff --git a/js/src/threading/ConditionVariable.h b/js/src/threading/ConditionVariable.h
new file mode 100644
index 0000000000..e2b8865cbf
--- /dev/null
+++ b/js/src/threading/ConditionVariable.h
@@ -0,0 +1,138 @@
+/* -*- 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 <stdint.h>
+#include <utility>
+#if !defined(XP_WIN) && !defined(__wasi__)
+# include <pthread.h>
+#endif
+
+#include "threading/LockGuard.h"
+#include "threading/Mutex.h"
+
+namespace js {
+
+template <class T>
+class ExclusiveData;
+
+enum class CVStatus { NoTimeout, Timeout };
+
+template <typename T>
+using UniqueLock = LockGuard<T>;
+
+// 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<Mutex>& 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 <typename Predicate>
+ void wait(UniqueLock<Mutex>& 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<Mutex>& 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 <typename Predicate>
+ bool wait_until(UniqueLock<Mutex>& 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<Mutex>& 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 <typename Predicate>
+ bool wait_for(UniqueLock<Mutex>& 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 <class T>
+ friend class ExclusiveWaitableData;
+
+ mozilla::detail::ConditionVariableImpl impl_;
+};
+
+} // namespace js
+
+#endif // threading_ConditionVariable_h