forked from Mirror/Ryujinx
123 lines
3.2 KiB
C#
123 lines
3.2 KiB
C#
|
using Ryujinx.Common.Logging;
|
||
|
using Silk.NET.Vulkan;
|
||
|
using System.Collections.Generic;
|
||
|
using System.Linq;
|
||
|
|
||
|
namespace Ryujinx.Graphics.Vulkan
|
||
|
{
|
||
|
class SyncManager
|
||
|
{
|
||
|
private class SyncHandle
|
||
|
{
|
||
|
public ulong ID;
|
||
|
public MultiFenceHolder Waitable;
|
||
|
}
|
||
|
|
||
|
private ulong _firstHandle = 0;
|
||
|
|
||
|
private readonly VulkanRenderer _gd;
|
||
|
private readonly Device _device;
|
||
|
private List<SyncHandle> _handles;
|
||
|
|
||
|
public SyncManager(VulkanRenderer gd, Device device)
|
||
|
{
|
||
|
_gd = gd;
|
||
|
_device = device;
|
||
|
_handles = new List<SyncHandle>();
|
||
|
}
|
||
|
|
||
|
public void Create(ulong id)
|
||
|
{
|
||
|
MultiFenceHolder waitable = new MultiFenceHolder();
|
||
|
|
||
|
_gd.FlushAllCommands();
|
||
|
_gd.CommandBufferPool.AddWaitable(waitable);
|
||
|
|
||
|
SyncHandle handle = new SyncHandle
|
||
|
{
|
||
|
ID = id,
|
||
|
Waitable = waitable
|
||
|
};
|
||
|
|
||
|
lock (_handles)
|
||
|
{
|
||
|
_handles.Add(handle);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
public void Wait(ulong id)
|
||
|
{
|
||
|
SyncHandle result = null;
|
||
|
|
||
|
lock (_handles)
|
||
|
{
|
||
|
if ((long)(_firstHandle - id) > 0)
|
||
|
{
|
||
|
return; // The handle has already been signalled or deleted.
|
||
|
}
|
||
|
|
||
|
foreach (SyncHandle handle in _handles)
|
||
|
{
|
||
|
if (handle.ID == id)
|
||
|
{
|
||
|
result = handle;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (result != null)
|
||
|
{
|
||
|
lock (result)
|
||
|
{
|
||
|
if (result.Waitable == null)
|
||
|
{
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
bool signaled = result.Waitable.WaitForFences(_gd.Api, _device, 1000000000);
|
||
|
if (!signaled)
|
||
|
{
|
||
|
Logger.Error?.PrintMsg(LogClass.Gpu, $"VK Sync Object {result.ID} failed to signal within 1000ms. Continuing...");
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
public void Cleanup()
|
||
|
{
|
||
|
// Iterate through handles and remove any that have already been signalled.
|
||
|
|
||
|
while (true)
|
||
|
{
|
||
|
SyncHandle first = null;
|
||
|
lock (_handles)
|
||
|
{
|
||
|
first = _handles.FirstOrDefault();
|
||
|
}
|
||
|
|
||
|
if (first == null) break;
|
||
|
|
||
|
bool signaled = first.Waitable.WaitForFences(_gd.Api, _device, 0);
|
||
|
if (signaled)
|
||
|
{
|
||
|
// Delete the sync object.
|
||
|
lock (_handles)
|
||
|
{
|
||
|
lock (first)
|
||
|
{
|
||
|
_firstHandle = first.ID + 1;
|
||
|
_handles.RemoveAt(0);
|
||
|
first.Waitable = null;
|
||
|
}
|
||
|
}
|
||
|
} else
|
||
|
{
|
||
|
// This sync handle and any following have not been reached yet.
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|