2014-04-13 23:15:23 +00:00
|
|
|
// Copyright 2014 Dolphin Emulator Project
|
2015-05-17 23:08:10 +00:00
|
|
|
// Licensed under GPLv2+
|
2014-04-13 23:15:23 +00:00
|
|
|
// 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
|
|
|
|
|
2014-04-14 00:30:40 +00:00
|
|
|
#ifdef _WIN32
|
|
|
|
#include <concrt.h>
|
|
|
|
#endif
|
|
|
|
|
2014-08-13 05:53:17 +00:00
|
|
|
#include <chrono>
|
2014-08-14 05:14:35 +00:00
|
|
|
#include <condition_variable>
|
|
|
|
#include <mutex>
|
|
|
|
|
2014-04-13 23:42:03 +00:00
|
|
|
#include "Common/Flag.h"
|
2014-04-13 23:15:23 +00:00
|
|
|
|
|
|
|
namespace Common {
|
|
|
|
|
2014-04-14 00:30:40 +00:00
|
|
|
class Event final
|
2014-04-13 23:15:23 +00:00
|
|
|
{
|
|
|
|
public:
|
|
|
|
void Set()
|
|
|
|
{
|
2014-04-13 23:42:03 +00:00
|
|
|
if (m_flag.TestAndSet())
|
2014-04-13 23:15:23 +00:00
|
|
|
{
|
2014-04-13 23:42:03 +00:00
|
|
|
std::lock_guard<std::mutex> lk(m_mutex);
|
2014-04-13 23:15:23 +00:00
|
|
|
m_condvar.notify_one();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void Wait()
|
|
|
|
{
|
2014-04-13 23:42:03 +00:00
|
|
|
if (m_flag.TestAndClear())
|
|
|
|
return;
|
|
|
|
|
2014-04-13 23:15:23 +00:00
|
|
|
std::unique_lock<std::mutex> lk(m_mutex);
|
2016-05-12 09:17:17 +00:00
|
|
|
m_condvar.wait(lk, [&]{ return m_flag.TestAndClear(); });
|
2014-04-13 23:15:23 +00:00
|
|
|
}
|
|
|
|
|
2014-08-13 05:53:17 +00:00
|
|
|
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,
|
2016-05-12 09:17:17 +00:00
|
|
|
[&]{ return m_flag.TestAndClear(); });
|
2014-08-13 05:53:17 +00:00
|
|
|
|
|
|
|
return signaled;
|
|
|
|
}
|
|
|
|
|
2014-04-13 23:15:23 +00:00
|
|
|
void Reset()
|
|
|
|
{
|
|
|
|
// no other action required, since wait loops on
|
|
|
|
// the predicate and any lingering signal will get
|
|
|
|
// cleared on the first iteration
|
2014-04-13 23:42:03 +00:00
|
|
|
m_flag.Clear();
|
2014-04-13 23:15:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
2014-04-13 23:42:03 +00:00
|
|
|
Flag m_flag;
|
2014-04-13 23:15:23 +00:00
|
|
|
std::condition_variable m_condvar;
|
|
|
|
std::mutex m_mutex;
|
|
|
|
};
|
|
|
|
|
|
|
|
} // namespace Common
|