| 
									
										
										
										
											2022-11-21 11:31:18 -05:00
										 |  |  | // SPDX-FileCopyrightText: 2022 yuzu Emulator Project
 | 
					
						
							|  |  |  | // SPDX-License-Identifier: GPL-2.0-or-later
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // TODO: remove this file when jthread is supported by all compilation targets
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #pragma once
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <version>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifdef __cpp_lib_jthread
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-25 16:03:32 -05:00
										 |  |  | #include <chrono>
 | 
					
						
							|  |  |  | #include <condition_variable>
 | 
					
						
							| 
									
										
										
										
											2022-11-21 11:31:18 -05:00
										 |  |  | #include <stop_token>
 | 
					
						
							|  |  |  | #include <thread>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace Common { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <typename Condvar, typename Lock, typename Pred> | 
					
						
							| 
									
										
										
										
											2023-09-02 11:43:16 -04:00
										 |  |  | void CondvarWait(Condvar& cv, std::unique_lock<Lock>& lk, std::stop_token token, Pred&& pred) { | 
					
						
							|  |  |  |     cv.wait(lk, token, std::move(pred)); | 
					
						
							| 
									
										
										
										
											2022-11-21 11:31:18 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-25 16:03:32 -05:00
										 |  |  | template <typename Rep, typename Period> | 
					
						
							|  |  |  | bool StoppableTimedWait(std::stop_token token, const std::chrono::duration<Rep, Period>& rel_time) { | 
					
						
							|  |  |  |     std::condition_variable_any cv; | 
					
						
							|  |  |  |     std::mutex m; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Perform the timed wait.
 | 
					
						
							|  |  |  |     std::unique_lock lk{m}; | 
					
						
							|  |  |  |     return !cv.wait_for(lk, token, rel_time, [&] { return token.stop_requested(); }); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-21 11:31:18 -05:00
										 |  |  | } // namespace Common
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <atomic>
 | 
					
						
							| 
									
										
										
										
											2023-01-25 16:03:32 -05:00
										 |  |  | #include <chrono>
 | 
					
						
							|  |  |  | #include <condition_variable>
 | 
					
						
							| 
									
										
										
										
											2022-11-21 11:31:18 -05:00
										 |  |  | #include <functional>
 | 
					
						
							| 
									
										
										
										
											2023-01-26 17:38:34 -05:00
										 |  |  | #include <map>
 | 
					
						
							| 
									
										
										
										
											2022-11-21 11:31:18 -05:00
										 |  |  | #include <memory>
 | 
					
						
							|  |  |  | #include <mutex>
 | 
					
						
							|  |  |  | #include <optional>
 | 
					
						
							|  |  |  | #include <thread>
 | 
					
						
							|  |  |  | #include <type_traits>
 | 
					
						
							| 
									
										
										
										
											2023-01-26 17:38:34 -05:00
										 |  |  | #include <utility>
 | 
					
						
							| 
									
										
										
										
											2022-11-21 11:31:18 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace std { | 
					
						
							|  |  |  | namespace polyfill { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-26 17:38:34 -05:00
										 |  |  | using stop_state_callback = size_t; | 
					
						
							| 
									
										
										
										
											2022-11-21 11:31:18 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | class stop_state { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     stop_state() = default; | 
					
						
							|  |  |  |     ~stop_state() = default; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     bool request_stop() { | 
					
						
							| 
									
										
										
										
											2023-01-26 17:38:34 -05:00
										 |  |  |         unique_lock lk{m_lock}; | 
					
						
							| 
									
										
										
										
											2022-11-21 11:31:18 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-26 17:38:34 -05:00
										 |  |  |         if (m_stop_requested) { | 
					
						
							|  |  |  |             // Already set, nothing to do.
 | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2022-11-21 11:31:18 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-26 17:38:34 -05:00
										 |  |  |         // Mark stop requested.
 | 
					
						
							|  |  |  |         m_stop_requested = true; | 
					
						
							| 
									
										
										
										
											2022-11-21 11:31:18 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-26 17:38:34 -05:00
										 |  |  |         while (!m_callbacks.empty()) { | 
					
						
							|  |  |  |             // Get an iterator to the first element.
 | 
					
						
							|  |  |  |             const auto it = m_callbacks.begin(); | 
					
						
							| 
									
										
										
										
											2022-11-21 11:31:18 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-26 17:38:34 -05:00
										 |  |  |             // Move the callback function out of the map.
 | 
					
						
							|  |  |  |             function<void()> f; | 
					
						
							|  |  |  |             swap(it->second, f); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // Erase the now-empty map element.
 | 
					
						
							|  |  |  |             m_callbacks.erase(it); | 
					
						
							| 
									
										
										
										
											2022-11-21 11:31:18 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-26 17:38:34 -05:00
										 |  |  |             // Run the callback.
 | 
					
						
							|  |  |  |             if (f) { | 
					
						
							|  |  |  |                 f(); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2022-11-21 11:31:18 -05:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     bool stop_requested() const { | 
					
						
							| 
									
										
										
										
											2023-01-26 17:38:34 -05:00
										 |  |  |         unique_lock lk{m_lock}; | 
					
						
							|  |  |  |         return m_stop_requested; | 
					
						
							| 
									
										
										
										
											2022-11-21 11:31:18 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-26 17:38:34 -05:00
										 |  |  |     stop_state_callback insert_callback(function<void()> f) { | 
					
						
							|  |  |  |         unique_lock lk{m_lock}; | 
					
						
							| 
									
										
										
										
											2022-11-21 11:31:18 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-26 17:38:34 -05:00
										 |  |  |         if (m_stop_requested) { | 
					
						
							|  |  |  |             // Stop already requested. Don't insert anything,
 | 
					
						
							|  |  |  |             // just run the callback synchronously.
 | 
					
						
							|  |  |  |             if (f) { | 
					
						
							|  |  |  |                 f(); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             return 0; | 
					
						
							| 
									
										
										
										
											2022-11-21 11:31:18 -05:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-26 17:38:34 -05:00
										 |  |  |         // Insert the callback.
 | 
					
						
							|  |  |  |         stop_state_callback ret = ++m_next_callback; | 
					
						
							|  |  |  |         m_callbacks.emplace(ret, move(f)); | 
					
						
							| 
									
										
										
										
											2022-11-21 11:31:18 -05:00
										 |  |  |         return ret; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-26 17:38:34 -05:00
										 |  |  |     void remove_callback(stop_state_callback cb) { | 
					
						
							|  |  |  |         unique_lock lk{m_lock}; | 
					
						
							|  |  |  |         m_callbacks.erase(cb); | 
					
						
							| 
									
										
										
										
											2022-11-21 11:31:18 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							| 
									
										
										
										
											2023-01-26 17:38:34 -05:00
										 |  |  |     mutable recursive_mutex m_lock; | 
					
						
							|  |  |  |     map<stop_state_callback, function<void()>> m_callbacks; | 
					
						
							|  |  |  |     stop_state_callback m_next_callback{0}; | 
					
						
							|  |  |  |     bool m_stop_requested{false}; | 
					
						
							| 
									
										
										
										
											2022-11-21 11:31:18 -05:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } // namespace polyfill
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class stop_token; | 
					
						
							|  |  |  | class stop_source; | 
					
						
							|  |  |  | struct nostopstate_t { | 
					
						
							|  |  |  |     explicit nostopstate_t() = default; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | inline constexpr nostopstate_t nostopstate{}; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <class Callback> | 
					
						
							|  |  |  | class stop_callback; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class stop_token { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     stop_token() noexcept = default; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     stop_token(const stop_token&) noexcept = default; | 
					
						
							|  |  |  |     stop_token(stop_token&&) noexcept = default; | 
					
						
							|  |  |  |     stop_token& operator=(const stop_token&) noexcept = default; | 
					
						
							|  |  |  |     stop_token& operator=(stop_token&&) noexcept = default; | 
					
						
							|  |  |  |     ~stop_token() = default; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void swap(stop_token& other) noexcept { | 
					
						
							|  |  |  |         m_stop_state.swap(other.m_stop_state); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     [[nodiscard]] bool stop_requested() const noexcept { | 
					
						
							|  |  |  |         return m_stop_state && m_stop_state->stop_requested(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     [[nodiscard]] bool stop_possible() const noexcept { | 
					
						
							|  |  |  |         return m_stop_state != nullptr; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							|  |  |  |     friend class stop_source; | 
					
						
							|  |  |  |     template <typename Callback> | 
					
						
							|  |  |  |     friend class stop_callback; | 
					
						
							|  |  |  |     stop_token(shared_ptr<polyfill::stop_state> stop_state) : m_stop_state(move(stop_state)) {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							|  |  |  |     shared_ptr<polyfill::stop_state> m_stop_state; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class stop_source { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     stop_source() : m_stop_state(make_shared<polyfill::stop_state>()) {} | 
					
						
							|  |  |  |     explicit stop_source(nostopstate_t) noexcept {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     stop_source(const stop_source&) noexcept = default; | 
					
						
							|  |  |  |     stop_source(stop_source&&) noexcept = default; | 
					
						
							|  |  |  |     stop_source& operator=(const stop_source&) noexcept = default; | 
					
						
							|  |  |  |     stop_source& operator=(stop_source&&) noexcept = default; | 
					
						
							|  |  |  |     ~stop_source() = default; | 
					
						
							|  |  |  |     void swap(stop_source& other) noexcept { | 
					
						
							|  |  |  |         m_stop_state.swap(other.m_stop_state); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     [[nodiscard]] stop_token get_token() const noexcept { | 
					
						
							|  |  |  |         return stop_token(m_stop_state); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     [[nodiscard]] bool stop_possible() const noexcept { | 
					
						
							|  |  |  |         return m_stop_state != nullptr; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     [[nodiscard]] bool stop_requested() const noexcept { | 
					
						
							|  |  |  |         return m_stop_state && m_stop_state->stop_requested(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     bool request_stop() noexcept { | 
					
						
							|  |  |  |         return m_stop_state && m_stop_state->request_stop(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							|  |  |  |     friend class jthread; | 
					
						
							|  |  |  |     explicit stop_source(shared_ptr<polyfill::stop_state> stop_state) | 
					
						
							|  |  |  |         : m_stop_state(move(stop_state)) {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							|  |  |  |     shared_ptr<polyfill::stop_state> m_stop_state; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <typename Callback> | 
					
						
							|  |  |  | class stop_callback { | 
					
						
							|  |  |  |     static_assert(is_nothrow_destructible_v<Callback>); | 
					
						
							|  |  |  |     static_assert(is_invocable_v<Callback>); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     using callback_type = Callback; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     template <typename C> | 
					
						
							| 
									
										
										
										
											2023-01-29 13:54:13 -07:00
										 |  |  |         requires constructible_from<Callback, C> | 
					
						
							| 
									
										
										
										
											2022-11-21 11:31:18 -05:00
										 |  |  |     explicit stop_callback(const stop_token& st, | 
					
						
							|  |  |  |                            C&& cb) noexcept(is_nothrow_constructible_v<Callback, C>) | 
					
						
							|  |  |  |         : m_stop_state(st.m_stop_state) { | 
					
						
							|  |  |  |         if (m_stop_state) { | 
					
						
							|  |  |  |             m_callback = m_stop_state->insert_callback(move(cb)); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     template <typename C> | 
					
						
							| 
									
										
										
										
											2023-01-29 13:54:13 -07:00
										 |  |  |         requires constructible_from<Callback, C> | 
					
						
							| 
									
										
										
										
											2022-11-21 11:31:18 -05:00
										 |  |  |     explicit stop_callback(stop_token&& st, | 
					
						
							|  |  |  |                            C&& cb) noexcept(is_nothrow_constructible_v<Callback, C>) | 
					
						
							|  |  |  |         : m_stop_state(move(st.m_stop_state)) { | 
					
						
							|  |  |  |         if (m_stop_state) { | 
					
						
							|  |  |  |             m_callback = m_stop_state->insert_callback(move(cb)); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     ~stop_callback() { | 
					
						
							|  |  |  |         if (m_stop_state && m_callback) { | 
					
						
							| 
									
										
										
										
											2023-01-26 17:38:34 -05:00
										 |  |  |             m_stop_state->remove_callback(m_callback); | 
					
						
							| 
									
										
										
										
											2022-11-21 11:31:18 -05:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     stop_callback(const stop_callback&) = delete; | 
					
						
							|  |  |  |     stop_callback(stop_callback&&) = delete; | 
					
						
							|  |  |  |     stop_callback& operator=(const stop_callback&) = delete; | 
					
						
							|  |  |  |     stop_callback& operator=(stop_callback&&) = delete; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							|  |  |  |     shared_ptr<polyfill::stop_state> m_stop_state; | 
					
						
							| 
									
										
										
										
											2023-01-26 17:38:34 -05:00
										 |  |  |     polyfill::stop_state_callback m_callback; | 
					
						
							| 
									
										
										
										
											2022-11-21 11:31:18 -05:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <typename Callback> | 
					
						
							|  |  |  | stop_callback(stop_token, Callback) -> stop_callback<Callback>; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class jthread { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     using id = thread::id; | 
					
						
							|  |  |  |     using native_handle_type = thread::native_handle_type; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     jthread() noexcept = default; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     template <typename F, typename... Args, | 
					
						
							|  |  |  |               typename = enable_if_t<!is_same_v<remove_cvref_t<F>, jthread>>> | 
					
						
							|  |  |  |     explicit jthread(F&& f, Args&&... args) | 
					
						
							|  |  |  |         : m_stop_state(make_shared<polyfill::stop_state>()), | 
					
						
							|  |  |  |           m_thread(make_thread(move(f), move(args)...)) {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ~jthread() { | 
					
						
							|  |  |  |         if (joinable()) { | 
					
						
							|  |  |  |             request_stop(); | 
					
						
							|  |  |  |             join(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     jthread(const jthread&) = delete; | 
					
						
							|  |  |  |     jthread(jthread&&) noexcept = default; | 
					
						
							|  |  |  |     jthread& operator=(const jthread&) = delete; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     jthread& operator=(jthread&& other) noexcept { | 
					
						
							|  |  |  |         m_thread.swap(other.m_thread); | 
					
						
							|  |  |  |         m_stop_state.swap(other.m_stop_state); | 
					
						
							|  |  |  |         return *this; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void swap(jthread& other) noexcept { | 
					
						
							|  |  |  |         m_thread.swap(other.m_thread); | 
					
						
							|  |  |  |         m_stop_state.swap(other.m_stop_state); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     [[nodiscard]] bool joinable() const noexcept { | 
					
						
							|  |  |  |         return m_thread.joinable(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     void join() { | 
					
						
							|  |  |  |         m_thread.join(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     void detach() { | 
					
						
							|  |  |  |         m_thread.detach(); | 
					
						
							|  |  |  |         m_stop_state.reset(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     [[nodiscard]] id get_id() const noexcept { | 
					
						
							|  |  |  |         return m_thread.get_id(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     [[nodiscard]] native_handle_type native_handle() { | 
					
						
							|  |  |  |         return m_thread.native_handle(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     [[nodiscard]] stop_source get_stop_source() noexcept { | 
					
						
							|  |  |  |         return stop_source(m_stop_state); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     [[nodiscard]] stop_token get_stop_token() const noexcept { | 
					
						
							|  |  |  |         return stop_source(m_stop_state).get_token(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     bool request_stop() noexcept { | 
					
						
							|  |  |  |         return get_stop_source().request_stop(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     [[nodiscard]] static unsigned int hardware_concurrency() noexcept { | 
					
						
							|  |  |  |         return thread::hardware_concurrency(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							|  |  |  |     template <typename F, typename... Args> | 
					
						
							|  |  |  |     thread make_thread(F&& f, Args&&... args) { | 
					
						
							|  |  |  |         if constexpr (is_invocable_v<decay_t<F>, stop_token, decay_t<Args>...>) { | 
					
						
							|  |  |  |             return thread(move(f), get_stop_token(), move(args)...); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             return thread(move(f), move(args)...); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     shared_ptr<polyfill::stop_state> m_stop_state; | 
					
						
							|  |  |  |     thread m_thread; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } // namespace std
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace Common { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | template <typename Condvar, typename Lock, typename Pred> | 
					
						
							| 
									
										
										
										
											2023-09-02 11:43:16 -04:00
										 |  |  | void CondvarWait(Condvar& cv, std::unique_lock<Lock>& lk, std::stop_token token, Pred pred) { | 
					
						
							| 
									
										
										
										
											2022-11-21 11:31:18 -05:00
										 |  |  |     if (token.stop_requested()) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-02 11:43:16 -04:00
										 |  |  |     std::stop_callback callback(token, [&] { | 
					
						
							|  |  |  |         { std::scoped_lock lk2{*lk.mutex()}; } | 
					
						
							|  |  |  |         cv.notify_all(); | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     cv.wait(lk, [&] { return pred() || token.stop_requested(); }); | 
					
						
							| 
									
										
										
										
											2022-11-21 11:31:18 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-25 16:03:32 -05:00
										 |  |  | template <typename Rep, typename Period> | 
					
						
							|  |  |  | bool StoppableTimedWait(std::stop_token token, const std::chrono::duration<Rep, Period>& rel_time) { | 
					
						
							|  |  |  |     if (token.stop_requested()) { | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     bool stop_requested = false; | 
					
						
							|  |  |  |     std::condition_variable cv; | 
					
						
							|  |  |  |     std::mutex m; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     std::stop_callback cb(token, [&] { | 
					
						
							|  |  |  |         // Wake up the waiting thread.
 | 
					
						
							| 
									
										
										
										
											2023-09-02 11:43:16 -04:00
										 |  |  |         { | 
					
						
							|  |  |  |             std::scoped_lock lk{m}; | 
					
						
							|  |  |  |             stop_requested = true; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2023-01-25 16:03:32 -05:00
										 |  |  |         cv.notify_one(); | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Perform the timed wait.
 | 
					
						
							|  |  |  |     std::unique_lock lk{m}; | 
					
						
							|  |  |  |     return !cv.wait_for(lk, rel_time, [&] { return stop_requested; }); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-21 11:31:18 -05:00
										 |  |  | } // namespace Common
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #endif
 |