1
0
mirror of https://github.com/EduApps-CDG/OpenDX synced 2024-12-30 09:45:37 +01:00
OpenDX/src/util/sync/sync_signal.h
2020-03-19 23:22:13 +01:00

94 lines
1.9 KiB
C++

#pragma once
#include <atomic>
#include <condition_variable>
#include <mutex>
#include "../rc/util_rc.h"
namespace dxvk::sync {
/**
* \brief Signal
*
* Interface for a CPU-side fence. Can be signaled
* to a given value, and any thread waiting for a
* lower value will be woken up.
*/
class Signal : public RcObject {
public:
virtual ~Signal() { }
/**
* \brief Last signaled value
* \returns Last signaled value
*/
virtual uint64_t value() const = 0;
/**
* \brief Notifies signal
*
* Wakes up all threads currently waiting for
* a value lower than \c value. Note that
* \c value must monotonically increase.
* \param [in] value Value to signal to
*/
virtual void signal(uint64_t value) = 0;
/**
* \brief Waits for signal
*
* Blocks the calling thread until another
* thread signals it with a value equal to
* or greater than \c value.
* \param [in] value The value to wait for
*/
virtual void wait(uint64_t value) = 0;
};
/**
* \brief Fence
*
* Simple CPU-side fence.
*/
class Fence final : public Signal {
public:
Fence()
: m_value(0ull) { }
explicit Fence(uint64_t value)
: m_value(value) { }
uint64_t value() const {
return m_value.load(std::memory_order_acquire);
}
void signal(uint64_t value) {
std::unique_lock<std::mutex> lock(m_mutex);
m_value.store(value, std::memory_order_release);
m_cond.notify_all();
}
void wait(uint64_t value) {
std::unique_lock<std::mutex> lock(m_mutex);
m_cond.wait(lock, [this, value] {
return value <= m_value.load(std::memory_order_acquire);
});
}
private:
std::atomic<uint64_t> m_value;
std::mutex m_mutex;
std::condition_variable m_cond;
};
}