2017-12-16 18:10:55 +01:00
|
|
|
#include "dxvk_device.h"
|
|
|
|
#include "dxvk_queue.h"
|
|
|
|
|
|
|
|
namespace dxvk {
|
|
|
|
|
|
|
|
DxvkSubmissionQueue::DxvkSubmissionQueue(DxvkDevice* device)
|
|
|
|
: m_device(device),
|
2017-12-25 16:05:11 +01:00
|
|
|
m_thread([this] () { threadFunc(); }) {
|
2017-12-16 18:10:55 +01:00
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
DxvkSubmissionQueue::~DxvkSubmissionQueue() {
|
2017-12-25 16:05:11 +01:00
|
|
|
{ std::unique_lock<std::mutex> lock(m_mutex);
|
|
|
|
m_stopped.store(true);
|
|
|
|
}
|
|
|
|
|
2017-12-16 18:10:55 +01:00
|
|
|
m_condOnAdd.notify_one();
|
|
|
|
m_thread.join();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void DxvkSubmissionQueue::submit(
|
|
|
|
const Rc<DxvkFence>& fence,
|
|
|
|
const Rc<DxvkCommandList>& cmdList) {
|
|
|
|
{ std::unique_lock<std::mutex> lock(m_mutex);
|
|
|
|
|
|
|
|
m_condOnTake.wait(lock, [this] {
|
2018-01-14 11:23:14 +01:00
|
|
|
return m_entries.size() < MaxNumQueuedCommandBuffers;
|
2017-12-16 18:10:55 +01:00
|
|
|
});
|
|
|
|
|
|
|
|
m_entries.push({ fence, cmdList });
|
|
|
|
}
|
|
|
|
|
|
|
|
m_condOnAdd.notify_one();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void DxvkSubmissionQueue::threadFunc() {
|
|
|
|
while (!m_stopped.load()) {
|
|
|
|
Entry entry;
|
|
|
|
|
|
|
|
{ std::unique_lock<std::mutex> lock(m_mutex);
|
|
|
|
|
|
|
|
m_condOnAdd.wait(lock, [this] {
|
|
|
|
return m_stopped.load() || (m_entries.size() != 0);
|
|
|
|
});
|
|
|
|
|
|
|
|
if (m_entries.size() != 0) {
|
|
|
|
entry = std::move(m_entries.front());
|
|
|
|
m_entries.pop();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
m_condOnTake.notify_one();
|
|
|
|
|
|
|
|
if (entry.fence != nullptr) {
|
|
|
|
entry.fence->wait(std::numeric_limits<uint64_t>::max());
|
|
|
|
entry.cmdList->reset();
|
|
|
|
m_device->recycleCommandList(entry.cmdList);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|