Check FenceCycle signalled flag immediately before waiting

The lock release within the wait for submission means that another thread could end up signalling the cycle and then the VK wait still happen after when the lock has been reacquired.
This commit is contained in:
Billy Laws 2022-10-22 13:25:51 +01:00
parent 1a23b929a7
commit 95d849e1f6

View File

@ -132,14 +132,15 @@ namespace skyline::gpu {
}); });
std::unique_lock lock{mutex}; std::unique_lock lock{mutex};
submitCondition.wait(lock, [&] { return submitted; });
if (signalled.test(std::memory_order_relaxed)) { if (signalled.test(std::memory_order_relaxed)) {
if (shouldDestroy) if (shouldDestroy)
DestroyDependencies(); DestroyDependencies();
return; return;
} }
submitCondition.wait(lock, [&] { return submitted; });
vk::Result waitResult; vk::Result waitResult;
while ((waitResult = (*device).waitForFences(1, &fence, false, std::numeric_limits<u64>::max(), *device.getDispatcher())) != vk::Result::eSuccess) { while ((waitResult = (*device).waitForFences(1, &fence, false, std::numeric_limits<u64>::max(), *device.getDispatcher())) != vk::Result::eSuccess) {
if (waitResult == vk::Result::eTimeout) if (waitResult == vk::Result::eTimeout)
@ -227,7 +228,7 @@ namespace skyline::gpu {
* @param cycle The cycle to chain to this one, this is nullable and this function will be a no-op if this is nullptr * @param cycle The cycle to chain to this one, this is nullable and this function will be a no-op if this is nullptr
*/ */
void ChainCycle(const std::shared_ptr<FenceCycle> &cycle) { void ChainCycle(const std::shared_ptr<FenceCycle> &cycle) {
if (cycle && !signalled.test(std::memory_order_consume) && cycle.get() != this) if (cycle && !signalled.test(std::memory_order_consume) && cycle.get() != this && !cycle->Poll())
chainedCycles.Append(cycle); // If the cycle isn't the current cycle or already signalled, we need to chain it chainedCycles.Append(cycle); // If the cycle isn't the current cycle or already signalled, we need to chain it
} }