| 
									
										
										
										
											2022-04-28 18:24:11 +02:00
										 |  |  | // SPDX-FileCopyrightText: 2013 Dolphin Emulator Project
 | 
					
						
							|  |  |  | // SPDX-FileCopyrightText: 2014 Citra Emulator Project
 | 
					
						
							|  |  |  | // SPDX-License-Identifier: GPL-2.0-or-later
 | 
					
						
							| 
									
										
										
										
											2013-09-04 20:17:46 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-08-17 13:45:50 -04:00
										 |  |  | #pragma once
 | 
					
						
							| 
									
										
										
										
											2013-09-04 20:17:46 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-26 01:41:46 +00:00
										 |  |  | #include <atomic>
 | 
					
						
							| 
									
										
										
										
											2016-12-11 23:26:23 +02:00
										 |  |  | #include <chrono>
 | 
					
						
							| 
									
										
										
										
											2014-09-02 22:05:45 -07:00
										 |  |  | #include <condition_variable>
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | #include <cstddef>
 | 
					
						
							| 
									
										
										
										
											2014-09-02 22:05:45 -07:00
										 |  |  | #include <mutex>
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | #include <thread>
 | 
					
						
							| 
									
										
										
										
											2020-02-08 12:48:57 -04:00
										 |  |  | #include "common/common_types.h"
 | 
					
						
							| 
									
										
										
										
											2022-12-06 16:13:42 -05:00
										 |  |  | #include "common/polyfill_thread.h"
 | 
					
						
							| 
									
										
										
										
											2013-09-04 20:17:46 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-14 12:53:05 +01:00
										 |  |  | namespace Common { | 
					
						
							| 
									
										
										
										
											2013-09-04 20:17:46 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-16 17:55:30 -03:00
										 |  |  | class Event { | 
					
						
							| 
									
										
										
										
											2013-09-04 20:17:46 -04:00
										 |  |  | public: | 
					
						
							| 
									
										
										
										
											2015-04-16 17:55:30 -03:00
										 |  |  |     void Set() { | 
					
						
							| 
									
										
										
										
											2022-04-07 19:30:55 +01:00
										 |  |  |         std::scoped_lock lk{mutex}; | 
					
						
							| 
									
										
										
										
											2015-04-16 17:55:30 -03:00
										 |  |  |         if (!is_set) { | 
					
						
							| 
									
										
										
										
											2014-04-01 18:20:08 -04:00
										 |  |  |             is_set = true; | 
					
						
							| 
									
										
										
										
											2016-04-14 12:53:05 +01:00
										 |  |  |             condvar.notify_one(); | 
					
						
							| 
									
										
										
										
											2014-04-01 18:20:08 -04:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-16 17:55:30 -03:00
										 |  |  |     void Wait() { | 
					
						
							| 
									
										
										
										
											2019-04-01 12:29:59 -04:00
										 |  |  |         std::unique_lock lk{mutex}; | 
					
						
							| 
									
										
										
										
											2020-08-26 01:41:46 +00:00
										 |  |  |         condvar.wait(lk, [&] { return is_set.load(); }); | 
					
						
							| 
									
										
										
										
											2014-04-01 18:20:08 -04:00
										 |  |  |         is_set = false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-08 12:48:57 -04:00
										 |  |  |     bool WaitFor(const std::chrono::nanoseconds& time) { | 
					
						
							| 
									
										
										
										
											2019-11-03 08:07:04 +01:00
										 |  |  |         std::unique_lock lk{mutex}; | 
					
						
							| 
									
										
										
										
											2020-08-26 01:41:46 +00:00
										 |  |  |         if (!condvar.wait_for(lk, time, [this] { return is_set.load(); })) | 
					
						
							| 
									
										
										
										
											2019-08-24 15:57:49 +02:00
										 |  |  |             return false; | 
					
						
							|  |  |  |         is_set = false; | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-12-11 23:26:23 +02:00
										 |  |  |     template <class Clock, class Duration> | 
					
						
							|  |  |  |     bool WaitUntil(const std::chrono::time_point<Clock, Duration>& time) { | 
					
						
							| 
									
										
										
										
											2019-04-01 12:29:59 -04:00
										 |  |  |         std::unique_lock lk{mutex}; | 
					
						
							| 
									
										
										
										
											2020-08-26 01:41:46 +00:00
										 |  |  |         if (!condvar.wait_until(lk, time, [this] { return is_set.load(); })) | 
					
						
							| 
									
										
										
										
											2016-12-11 23:26:23 +02:00
										 |  |  |             return false; | 
					
						
							|  |  |  |         is_set = false; | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-16 17:55:30 -03:00
										 |  |  |     void Reset() { | 
					
						
							| 
									
										
										
										
											2019-04-01 12:29:59 -04:00
										 |  |  |         std::unique_lock lk{mutex}; | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  |         // no other action required, since wait loops on the predicate and any lingering signal will
 | 
					
						
							|  |  |  |         // get cleared on the first iteration
 | 
					
						
							| 
									
										
										
										
											2014-04-01 18:20:08 -04:00
										 |  |  |         is_set = false; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2013-09-04 20:17:46 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-06-20 11:41:38 -04:00
										 |  |  |     [[nodiscard]] bool IsSet() const { | 
					
						
							| 
									
										
										
										
											2022-07-27 22:31:41 +01:00
										 |  |  |         return is_set; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-09-04 20:17:46 -04:00
										 |  |  | private: | 
					
						
							| 
									
										
										
										
											2016-04-14 12:53:05 +01:00
										 |  |  |     std::condition_variable condvar; | 
					
						
							|  |  |  |     std::mutex mutex; | 
					
						
							| 
									
										
										
										
											2020-08-26 01:41:46 +00:00
										 |  |  |     std::atomic_bool is_set{false}; | 
					
						
							| 
									
										
										
										
											2013-09-04 20:17:46 -04:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-16 17:55:30 -03:00
										 |  |  | class Barrier { | 
					
						
							| 
									
										
										
										
											2013-09-04 20:17:46 -04:00
										 |  |  | public: | 
					
						
							| 
									
										
										
										
											2018-11-21 21:44:58 -05:00
										 |  |  |     explicit Barrier(std::size_t count_) : count(count_) {} | 
					
						
							| 
									
										
										
										
											2014-04-01 18:20:08 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-16 17:55:30 -03:00
										 |  |  |     /// Blocks until all "count" threads have called Sync()
 | 
					
						
							| 
									
										
										
										
											2022-12-06 16:13:42 -05:00
										 |  |  |     bool Sync(std::stop_token token = {}) { | 
					
						
							| 
									
										
										
										
											2019-04-01 12:29:59 -04:00
										 |  |  |         std::unique_lock lk{mutex}; | 
					
						
							| 
									
										
										
										
											2018-09-15 15:21:06 +02:00
										 |  |  |         const std::size_t current_generation = generation; | 
					
						
							| 
									
										
										
										
											2014-04-01 18:20:08 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-14 12:53:05 +01:00
										 |  |  |         if (++waiting == count) { | 
					
						
							| 
									
										
										
										
											2016-04-14 12:54:06 +01:00
										 |  |  |             generation++; | 
					
						
							| 
									
										
										
										
											2016-04-14 12:53:05 +01:00
										 |  |  |             waiting = 0; | 
					
						
							|  |  |  |             condvar.notify_all(); | 
					
						
							| 
									
										
										
										
											2022-12-06 16:13:42 -05:00
										 |  |  |             return true; | 
					
						
							| 
									
										
										
										
											2015-04-16 17:55:30 -03:00
										 |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2022-12-06 16:13:42 -05:00
										 |  |  |             CondvarWait(condvar, lk, token, | 
					
						
							|  |  |  |                         [this, current_generation] { return current_generation != generation; }); | 
					
						
							|  |  |  |             return !token.stop_requested(); | 
					
						
							| 
									
										
										
										
											2014-04-01 18:20:08 -04:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2013-09-04 20:17:46 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							| 
									
										
										
										
											2022-12-06 16:13:42 -05:00
										 |  |  |     std::condition_variable_any condvar; | 
					
						
							| 
									
										
										
										
											2016-04-14 12:53:05 +01:00
										 |  |  |     std::mutex mutex; | 
					
						
							| 
									
										
										
										
											2018-11-21 21:47:06 -05:00
										 |  |  |     std::size_t count; | 
					
						
							| 
									
										
										
										
											2018-11-21 21:44:58 -05:00
										 |  |  |     std::size_t waiting = 0; | 
					
						
							|  |  |  |     std::size_t generation = 0; // Incremented once each time the barrier is used
 | 
					
						
							| 
									
										
										
										
											2013-09-04 20:17:46 -04:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2014-11-19 08:49:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-05 09:48:53 -04:00
										 |  |  | enum class ThreadPriority : u32 { | 
					
						
							|  |  |  |     Low = 0, | 
					
						
							|  |  |  |     Normal = 1, | 
					
						
							|  |  |  |     High = 2, | 
					
						
							|  |  |  |     VeryHigh = 3, | 
					
						
							| 
									
										
										
										
											2021-11-27 20:31:46 +01:00
										 |  |  |     Critical = 4, | 
					
						
							| 
									
										
										
										
											2020-04-05 09:48:53 -04:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void SetCurrentThreadPriority(ThreadPriority new_priority); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-18 09:38:01 +09:00
										 |  |  | void SetCurrentThreadName(const char* name); | 
					
						
							| 
									
										
										
										
											2014-11-19 08:49:13 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-09-04 20:17:46 -04:00
										 |  |  | } // namespace Common
 |