vk_scheduler: Use locks instead of SPSC a queue
This tries to fix a data race where we'd wait forever for the GPU.
This commit is contained in:
parent
56c47951c5
commit
36f1586267
|
@ -47,8 +47,11 @@ VKScheduler::VKScheduler(const Device& device_, StateTracker& state_tracker_)
|
||||||
}
|
}
|
||||||
|
|
||||||
VKScheduler::~VKScheduler() {
|
VKScheduler::~VKScheduler() {
|
||||||
|
{
|
||||||
|
std::lock_guard lock{work_mutex};
|
||||||
quit = true;
|
quit = true;
|
||||||
cv.notify_all();
|
}
|
||||||
|
work_cv.notify_all();
|
||||||
worker_thread.join();
|
worker_thread.join();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -69,20 +72,19 @@ void VKScheduler::WaitWorker() {
|
||||||
MICROPROFILE_SCOPE(Vulkan_WaitForWorker);
|
MICROPROFILE_SCOPE(Vulkan_WaitForWorker);
|
||||||
DispatchWork();
|
DispatchWork();
|
||||||
|
|
||||||
bool finished = false;
|
std::unique_lock lock{work_mutex};
|
||||||
do {
|
wait_cv.wait(lock, [this] { return work_queue.empty(); });
|
||||||
cv.notify_all();
|
|
||||||
std::unique_lock lock{mutex};
|
|
||||||
finished = chunk_queue.Empty();
|
|
||||||
} while (!finished);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void VKScheduler::DispatchWork() {
|
void VKScheduler::DispatchWork() {
|
||||||
if (chunk->Empty()) {
|
if (chunk->Empty()) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
chunk_queue.Push(std::move(chunk));
|
{
|
||||||
cv.notify_all();
|
std::lock_guard lock{work_mutex};
|
||||||
|
work_queue.push(std::move(chunk));
|
||||||
|
}
|
||||||
|
work_cv.notify_one();
|
||||||
AcquireNewChunk();
|
AcquireNewChunk();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -135,22 +137,27 @@ bool VKScheduler::UpdateGraphicsPipeline(GraphicsPipeline* pipeline) {
|
||||||
|
|
||||||
void VKScheduler::WorkerThread() {
|
void VKScheduler::WorkerThread() {
|
||||||
Common::SetCurrentThreadName("yuzu:VulkanWorker");
|
Common::SetCurrentThreadName("yuzu:VulkanWorker");
|
||||||
std::unique_lock lock{mutex};
|
|
||||||
do {
|
do {
|
||||||
cv.wait(lock, [this] { return !chunk_queue.Empty() || quit; });
|
if (work_queue.empty()) {
|
||||||
|
wait_cv.notify_all();
|
||||||
|
}
|
||||||
|
std::unique_ptr<CommandChunk> work;
|
||||||
|
{
|
||||||
|
std::unique_lock lock{work_mutex};
|
||||||
|
work_cv.wait(lock, [this] { return !work_queue.empty() || quit; });
|
||||||
if (quit) {
|
if (quit) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
while (!chunk_queue.Empty()) {
|
work = std::move(work_queue.front());
|
||||||
auto extracted_chunk = std::move(chunk_queue.Front());
|
work_queue.pop();
|
||||||
chunk_queue.Pop();
|
}
|
||||||
const bool has_submit = extracted_chunk->HasSubmit();
|
const bool has_submit = work->HasSubmit();
|
||||||
extracted_chunk->ExecuteAll(current_cmdbuf);
|
work->ExecuteAll(current_cmdbuf);
|
||||||
if (has_submit) {
|
if (has_submit) {
|
||||||
AllocateWorkerCommandBuffer();
|
AllocateWorkerCommandBuffer();
|
||||||
}
|
}
|
||||||
chunk_reserve.Push(std::move(extracted_chunk));
|
std::lock_guard reserve_lock{reserve_mutex};
|
||||||
}
|
chunk_reserve.push_back(std::move(work));
|
||||||
} while (!quit);
|
} while (!quit);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -269,12 +276,13 @@ void VKScheduler::EndRenderPass() {
|
||||||
}
|
}
|
||||||
|
|
||||||
void VKScheduler::AcquireNewChunk() {
|
void VKScheduler::AcquireNewChunk() {
|
||||||
if (chunk_reserve.Empty()) {
|
std::lock_guard lock{reserve_mutex};
|
||||||
|
if (chunk_reserve.empty()) {
|
||||||
chunk = std::make_unique<CommandChunk>();
|
chunk = std::make_unique<CommandChunk>();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
chunk = std::move(chunk_reserve.Front());
|
chunk = std::move(chunk_reserve.back());
|
||||||
chunk_reserve.Pop();
|
chunk_reserve.pop_back();
|
||||||
}
|
}
|
||||||
|
|
||||||
} // namespace Vulkan
|
} // namespace Vulkan
|
||||||
|
|
|
@ -6,14 +6,14 @@
|
||||||
|
|
||||||
#include <atomic>
|
#include <atomic>
|
||||||
#include <condition_variable>
|
#include <condition_variable>
|
||||||
|
#include <queue>
|
||||||
#include <cstddef>
|
#include <cstddef>
|
||||||
#include <memory>
|
#include <memory>
|
||||||
#include <stack>
|
|
||||||
#include <thread>
|
#include <thread>
|
||||||
#include <utility>
|
#include <utility>
|
||||||
|
|
||||||
#include "common/alignment.h"
|
#include "common/alignment.h"
|
||||||
#include "common/common_types.h"
|
#include "common/common_types.h"
|
||||||
#include "common/threadsafe_queue.h"
|
|
||||||
#include "video_core/renderer_vulkan/vk_master_semaphore.h"
|
#include "video_core/renderer_vulkan/vk_master_semaphore.h"
|
||||||
#include "video_core/vulkan_common/vulkan_wrapper.h"
|
#include "video_core/vulkan_common/vulkan_wrapper.h"
|
||||||
|
|
||||||
|
@ -220,11 +220,13 @@ private:
|
||||||
std::array<VkImage, 9> renderpass_images{};
|
std::array<VkImage, 9> renderpass_images{};
|
||||||
std::array<VkImageSubresourceRange, 9> renderpass_image_ranges{};
|
std::array<VkImageSubresourceRange, 9> renderpass_image_ranges{};
|
||||||
|
|
||||||
Common::SPSCQueue<std::unique_ptr<CommandChunk>> chunk_queue;
|
std::queue<std::unique_ptr<CommandChunk>> work_queue;
|
||||||
Common::SPSCQueue<std::unique_ptr<CommandChunk>> chunk_reserve;
|
std::vector<std::unique_ptr<CommandChunk>> chunk_reserve;
|
||||||
std::mutex mutex;
|
std::mutex reserve_mutex;
|
||||||
std::condition_variable cv;
|
std::mutex work_mutex;
|
||||||
bool quit = false;
|
std::condition_variable work_cv;
|
||||||
|
std::condition_variable wait_cv;
|
||||||
|
std::atomic_bool quit{};
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace Vulkan
|
} // namespace Vulkan
|
||||||
|
|
Reference in New Issue