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:
ReinUsesLisp 2021-05-07 06:26:12 -03:00 committed by ameerj
parent 56c47951c5
commit 36f1586267
2 changed files with 42 additions and 32 deletions

View file

@ -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

View file

@ -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