Ryujinx/src/Ryujinx.Graphics.Vulkan/FenceHolder.cs
riperiperi 1b28ecd63e
Vulkan: Simplify MultiFenceHolder and managing them (#4845)
* Vulkan: Simplify waitable add/remove

Removal of unnecessary hashset and dictionary

* Thread safety for GetBufferData in PersistentFlushBuffer

* Fix WaitForFencesImpl thread safety

* Proper methods for risky reference increments

* Wrong type of CB.

* Address feedback
2023-05-08 12:45:12 +02:00

98 lines
2.2 KiB
C#

using Silk.NET.Vulkan;
using System;
using System.Threading;
namespace Ryujinx.Graphics.Vulkan
{
class FenceHolder : IDisposable
{
private readonly Vk _api;
private readonly Device _device;
private Fence _fence;
private int _referenceCount;
private bool _disposed;
public unsafe FenceHolder(Vk api, Device device)
{
_api = api;
_device = device;
var fenceCreateInfo = new FenceCreateInfo()
{
SType = StructureType.FenceCreateInfo
};
api.CreateFence(device, in fenceCreateInfo, null, out _fence).ThrowOnError();
_referenceCount = 1;
}
public Fence GetUnsafe()
{
return _fence;
}
public bool TryGet(out Fence fence)
{
int lastValue;
do
{
lastValue = _referenceCount;
if (lastValue == 0)
{
fence = default;
return false;
}
}
while (Interlocked.CompareExchange(ref _referenceCount, lastValue + 1, lastValue) != lastValue);
fence = _fence;
return true;
}
public Fence Get()
{
Interlocked.Increment(ref _referenceCount);
return _fence;
}
public void Put()
{
if (Interlocked.Decrement(ref _referenceCount) == 0)
{
_api.DestroyFence(_device, _fence, Span<AllocationCallbacks>.Empty);
_fence = default;
}
}
public void Wait()
{
Span<Fence> fences = stackalloc Fence[]
{
_fence
};
FenceHelper.WaitAllIndefinitely(_api, _device, fences);
}
public bool IsSignaled()
{
Span<Fence> fences = stackalloc Fence[]
{
_fence
};
return FenceHelper.AllSignaled(_api, _device, fences);
}
public void Dispose()
{
if (!_disposed)
{
Put();
_disposed = true;
}
}
}
}