mirror of
				https://github.com/dolphin-emu/dolphin.git
				synced 2025-10-25 01:19:19 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			114 lines
		
	
	
	
		
			2.2 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			114 lines
		
	
	
	
		
			2.2 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| // Copyright 2014 Dolphin Emulator Project
 | |
| // Licensed under GPLv2
 | |
| // Refer to the license.txt file included.
 | |
| 
 | |
| // Multithreaded event class. This allows waiting in a thread for an event to
 | |
| // be triggered in another thread. While waiting, the CPU will be available for
 | |
| // other tasks.
 | |
| // * Set(): triggers the event and wakes up the waiting thread.
 | |
| // * Wait(): waits for the event to be triggered.
 | |
| // * Reset(): tries to reset the event before the waiting thread sees it was
 | |
| //            triggered. Usually a bad idea.
 | |
| 
 | |
| #pragma once
 | |
| 
 | |
| #ifdef _WIN32
 | |
| #include <concrt.h>
 | |
| #endif
 | |
| 
 | |
| #include <chrono>
 | |
| #include <condition_variable>
 | |
| #include <mutex>
 | |
| 
 | |
| #include "Common/Flag.h"
 | |
| 
 | |
| namespace Common {
 | |
| 
 | |
| // Windows uses a specific implementation because std::condition_variable has
 | |
| // terrible performance for this kind of workload with MSVC++ 2013.
 | |
| #ifndef _WIN32
 | |
| class Event final
 | |
| {
 | |
| public:
 | |
| 	void Set()
 | |
| 	{
 | |
| 		if (m_flag.TestAndSet())
 | |
| 		{
 | |
| 			std::lock_guard<std::mutex> lk(m_mutex);
 | |
| 			m_condvar.notify_one();
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	void Wait()
 | |
| 	{
 | |
| 		if (m_flag.TestAndClear())
 | |
| 			return;
 | |
| 
 | |
| 		std::unique_lock<std::mutex> lk(m_mutex);
 | |
| 		m_condvar.wait(lk, [&]{ return m_flag.IsSet(); });
 | |
| 		m_flag.Clear();
 | |
| 	}
 | |
| 
 | |
| 	template<class Rep, class Period>
 | |
| 	bool WaitFor(const std::chrono::duration<Rep, Period>& rel_time)
 | |
| 	{
 | |
| 		if (m_flag.TestAndClear())
 | |
| 			return true;
 | |
| 
 | |
| 		std::unique_lock<std::mutex> lk(m_mutex);
 | |
| 		bool signaled = m_condvar.wait_for(lk, rel_time,
 | |
| 			[&]{ return m_flag.IsSet(); });
 | |
| 		m_flag.Clear();
 | |
| 
 | |
| 		return signaled;
 | |
| 	}
 | |
| 
 | |
| 	void Reset()
 | |
| 	{
 | |
| 		// no other action required, since wait loops on
 | |
| 		// the predicate and any lingering signal will get
 | |
| 		// cleared on the first iteration
 | |
| 		m_flag.Clear();
 | |
| 	}
 | |
| 
 | |
| private:
 | |
| 	Flag m_flag;
 | |
| 	std::condition_variable m_condvar;
 | |
| 	std::mutex m_mutex;
 | |
| };
 | |
| #else
 | |
| class Event final
 | |
| {
 | |
| public:
 | |
| 	void Set()
 | |
| 	{
 | |
| 		m_event.set();
 | |
| 	}
 | |
| 
 | |
| 	void Wait()
 | |
| 	{
 | |
| 		m_event.wait();
 | |
| 		m_event.reset();
 | |
| 	}
 | |
| 
 | |
| 	template<class Rep, class Period>
 | |
| 	bool WaitFor(const std::chrono::duration<Rep, Period>& rel_time)
 | |
| 	{
 | |
| 		bool signaled = m_event.wait(
 | |
| 			(u32)std::chrono::duration_cast<std::chrono::milliseconds>(rel_time).count()
 | |
| 			) == 0;
 | |
| 		m_event.reset();
 | |
| 		return signaled;
 | |
| 	}
 | |
| 
 | |
| 	void Reset()
 | |
| 	{
 | |
| 		m_event.reset();
 | |
| 	}
 | |
| 
 | |
| private:
 | |
| 	concurrency::event m_event;
 | |
| };
 | |
| #endif
 | |
| 
 | |
| }  // namespace Common
 |