Genericise SwappableDisplaySurfaceSet

This commit is contained in:
YoshiRulz 2021-04-08 16:37:20 +10:00
parent 85eedb39fd
commit 0bfe3cb6ab
No known key found for this signature in database
GPG Key ID: C4DE31C245353FB7
2 changed files with 22 additions and 14 deletions

View File

@ -1,3 +1,4 @@
using System;
using System.Collections.Generic;
namespace BizHawk.Client.Common
@ -6,20 +7,25 @@ namespace BizHawk.Client.Common
/// encapsulates thread-safe concept of pending/current display surfaces, reusing buffers where matching
/// sizes are available and keeping them cleaned up when they don't seem like they'll need to be used anymore
/// </summary>
public class SwappableDisplaySurfaceSet
public class SwappableDisplaySurfaceSet<T>
where T : class, IDisplaySurface
{
private DisplaySurface _pending, _current;
private readonly Func<int, int, T> _createDispSurface;
private T _pending, _current;
private bool _isPending;
private readonly Queue<DisplaySurface> _releasedSurfaces = new Queue<DisplaySurface>();
private readonly Queue<T> _releasedSurfaces = new();
public SwappableDisplaySurfaceSet(Func<int, int, T> createDispSurface) => _createDispSurface = createDispSurface;
/// <summary>
/// retrieves a surface with the specified size, reusing an old buffer if available and clearing if requested
/// </summary>
public DisplaySurface AllocateSurface(int width, int height, bool needsClear = true)
public T AllocateSurface(int width, int height, bool needsClear = true)
{
for (; ; )
{
DisplaySurface trial;
T trial;
lock (this)
{
if (_releasedSurfaces.Count == 0) break;
@ -38,13 +44,13 @@ namespace BizHawk.Client.Common
trial.Dispose();
}
return new DisplaySurface(width, height);
return _createDispSurface(width, height);
}
/// <summary>
/// sets the provided buffer as pending. takes control of the supplied buffer
/// </summary>
public void SetPending(DisplaySurface newPending)
public void SetPending(T newPending)
{
lock (this)
{
@ -54,7 +60,7 @@ namespace BizHawk.Client.Common
}
}
public void ReleaseSurface(DisplaySurface surface)
public void ReleaseSurface(T surface)
{
lock (this) _releasedSurfaces.Enqueue(surface);
}
@ -62,7 +68,7 @@ namespace BizHawk.Client.Common
/// <summary>
/// returns the current buffer, making the most recent pending buffer (if there is such) as the new current first.
/// </summary>
public DisplaySurface GetCurrent()
public T GetCurrent()
{
lock (this)
{

View File

@ -31,6 +31,8 @@ namespace BizHawk.Client.EmuHawk
/// </summary>
public class DisplayManager : IDisplayManagerForApi, IWindowCoordsTransformer, IDisposable
{
private static DisplaySurface CreateDisplaySurface(int w, int h) => new(w, h);
private class DisplayManagerRenderTargetProvider : IRenderTargetProvider
{
private readonly Func<Size, RenderTarget> _callback;
@ -129,8 +131,8 @@ namespace BizHawk.Client.EmuHawk
}
}
_apiHawkSurfaceSets[DisplaySurfaceID.EmuCore] = new SwappableDisplaySurfaceSet();
_apiHawkSurfaceSets[DisplaySurfaceID.Client] = new SwappableDisplaySurfaceSet();
_apiHawkSurfaceSets[DisplaySurfaceID.EmuCore] = new(CreateDisplaySurface);
_apiHawkSurfaceSets[DisplaySurfaceID.Client] = new(CreateDisplaySurface);
_apiHawkSurfaceFrugalizers[DisplaySurfaceID.EmuCore] = new TextureFrugalizer(_gl);
_apiHawkSurfaceFrugalizers[DisplaySurfaceID.Client] = new TextureFrugalizer(_gl);
@ -1102,7 +1104,7 @@ namespace BizHawk.Client.EmuHawk
/// <remarks>Can't this just be a prop of <see cref="IDisplaySurface"/>? --yoshi</remarks>
private readonly Dictionary<IDisplaySurface, DisplaySurfaceID> _apiHawkSurfaceToID = new();
private readonly Dictionary<DisplaySurfaceID, SwappableDisplaySurfaceSet> _apiHawkSurfaceSets = new();
private readonly Dictionary<DisplaySurfaceID, SwappableDisplaySurfaceSet<DisplaySurface>> _apiHawkSurfaceSets = new();
/// <summary>
/// Peeks a locked lua surface, or returns null if it isn't locked
@ -1123,7 +1125,7 @@ namespace BizHawk.Client.EmuHawk
if (!_apiHawkSurfaceSets.TryGetValue(surfaceID, out var sdss))
{
sdss = new SwappableDisplaySurfaceSet();
sdss = new(CreateDisplaySurface);
_apiHawkSurfaceSets.Add(surfaceID, sdss);
}
@ -1171,7 +1173,7 @@ namespace BizHawk.Client.EmuHawk
/// <exception cref="InvalidOperationException">already unlocked</exception>
public void UnlockApiHawkSurface(IDisplaySurface surface)
{
if (surface is not DisplaySurface dispSurfaceImpl) throw new ArgumentException("o noes", nameof(surface));
if (surface is not DisplaySurface dispSurfaceImpl) throw new ArgumentException("don't mix " + nameof(IDisplaySurface) + " implementations!", nameof(surface));
if (!_apiHawkSurfaceToID.ContainsKey(dispSurfaceImpl))
{
throw new InvalidOperationException("Surface was not locked as a lua surface");