2026-03-16 18:48:12 +08:00
|
|
|
#include "XCEngine/RHI/OpenGL/OpenGLFence.h"
|
|
|
|
|
#include <glad/glad.h>
|
|
|
|
|
|
|
|
|
|
namespace XCEngine {
|
|
|
|
|
namespace RHI {
|
|
|
|
|
|
2026-03-24 01:53:00 +08:00
|
|
|
OpenGLFence::OpenGLFence()
|
2026-03-17 02:17:41 +08:00
|
|
|
: m_sync(nullptr)
|
2026-03-24 01:53:00 +08:00
|
|
|
, m_signaledValue(0)
|
|
|
|
|
, m_completedValue(0) {
|
2026-03-16 18:48:12 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
OpenGLFence::~OpenGLFence() {
|
2026-03-17 02:17:41 +08:00
|
|
|
Shutdown();
|
2026-03-16 18:48:12 +08:00
|
|
|
}
|
|
|
|
|
|
2026-03-24 01:53:00 +08:00
|
|
|
bool OpenGLFence::Initialize(uint64_t initialValue) {
|
|
|
|
|
m_signaledValue.store(initialValue, std::memory_order_release);
|
|
|
|
|
m_completedValue.store(initialValue, std::memory_order_release);
|
2026-03-16 18:48:12 +08:00
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void OpenGLFence::Shutdown() {
|
2026-03-17 02:17:41 +08:00
|
|
|
if (m_sync) {
|
|
|
|
|
glDeleteSync(static_cast<GLsync>(m_sync));
|
|
|
|
|
m_sync = nullptr;
|
|
|
|
|
}
|
2026-03-24 01:53:00 +08:00
|
|
|
m_signaledValue.store(0, std::memory_order_release);
|
|
|
|
|
m_completedValue.store(0, std::memory_order_release);
|
2026-03-16 18:48:12 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void OpenGLFence::Signal() {
|
2026-03-24 01:53:00 +08:00
|
|
|
Signal(1);
|
2026-03-17 02:17:41 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void OpenGLFence::Signal(uint64_t value) {
|
2026-03-23 21:09:15 +08:00
|
|
|
if (m_sync) {
|
2026-03-24 01:53:00 +08:00
|
|
|
glClientWaitSync(static_cast<GLsync>(m_sync), GL_SYNC_FLUSH_COMMANDS_BIT, GL_TIMEOUT_IGNORED);
|
2026-03-23 21:09:15 +08:00
|
|
|
glDeleteSync(static_cast<GLsync>(m_sync));
|
2026-03-24 01:53:00 +08:00
|
|
|
m_sync = nullptr;
|
2026-03-23 21:09:15 +08:00
|
|
|
}
|
2026-03-24 01:53:00 +08:00
|
|
|
|
|
|
|
|
glFlush();
|
|
|
|
|
|
|
|
|
|
m_signaledValue.store(value, std::memory_order_release);
|
2026-03-23 21:09:15 +08:00
|
|
|
m_sync = glFenceSync(GL_SYNC_GPU_COMMANDS_COMPLETE, 0);
|
2026-03-16 18:48:12 +08:00
|
|
|
}
|
|
|
|
|
|
2026-03-24 01:53:00 +08:00
|
|
|
void OpenGLFence::Wait(uint64_t value) {
|
|
|
|
|
uint64_t currentCompleted = m_completedValue.load(std::memory_order_acquire);
|
|
|
|
|
if (currentCompleted >= value) {
|
|
|
|
|
return;
|
2026-03-17 02:17:41 +08:00
|
|
|
}
|
2026-03-16 18:48:12 +08:00
|
|
|
|
2026-03-17 02:17:41 +08:00
|
|
|
if (m_sync) {
|
2026-03-24 01:53:00 +08:00
|
|
|
glClientWaitSync(static_cast<GLsync>(m_sync), GL_SYNC_FLUSH_COMMANDS_BIT, GL_TIMEOUT_IGNORED);
|
2026-03-17 02:17:41 +08:00
|
|
|
glDeleteSync(static_cast<GLsync>(m_sync));
|
|
|
|
|
m_sync = nullptr;
|
|
|
|
|
}
|
2026-03-16 18:48:12 +08:00
|
|
|
|
2026-03-24 01:53:00 +08:00
|
|
|
uint64_t signaled = m_signaledValue.load(std::memory_order_acquire);
|
|
|
|
|
m_completedValue.store(signaled, std::memory_order_release);
|
2026-03-16 18:48:12 +08:00
|
|
|
}
|
|
|
|
|
|
2026-03-24 01:53:00 +08:00
|
|
|
uint64_t OpenGLFence::GetCompletedValue() const {
|
2026-03-17 02:17:41 +08:00
|
|
|
if (!m_sync) {
|
2026-03-24 01:53:00 +08:00
|
|
|
return m_completedValue.load(std::memory_order_acquire);
|
2026-03-17 02:17:41 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
GLint status = 0;
|
2026-03-24 01:53:00 +08:00
|
|
|
glGetSynciv(static_cast<GLsync>(m_sync), GL_SYNC_STATUS, sizeof(status), nullptr, &status);
|
|
|
|
|
|
2026-03-17 02:17:41 +08:00
|
|
|
if (status == GL_SIGNALED) {
|
2026-03-24 01:53:00 +08:00
|
|
|
return m_signaledValue.load(std::memory_order_acquire);
|
2026-03-17 02:17:41 +08:00
|
|
|
}
|
|
|
|
|
|
2026-03-24 01:53:00 +08:00
|
|
|
return m_completedValue.load(std::memory_order_acquire);
|
2026-03-17 02:17:41 +08:00
|
|
|
}
|
|
|
|
|
|
2026-03-23 21:09:15 +08:00
|
|
|
void* OpenGLFence::GetNativeHandle() {
|
|
|
|
|
if (!m_sync) {
|
|
|
|
|
m_sync = glFenceSync(GL_SYNC_GPU_COMMANDS_COMPLETE, 0);
|
|
|
|
|
}
|
|
|
|
|
return m_sync;
|
|
|
|
|
}
|
|
|
|
|
|
2026-03-16 18:48:12 +08:00
|
|
|
} // namespace RHI
|
2026-03-24 01:53:00 +08:00
|
|
|
} // namespace XCEngine
|