2019-10-13 02:02:07 -04:00
|
|
|
using System.Collections.Generic;
|
|
|
|
|
|
|
|
namespace Ryujinx.Graphics.Gpu.Image
|
|
|
|
{
|
2019-12-29 18:26:37 -05:00
|
|
|
/// <summary>
|
|
|
|
/// Texture pool cache.
|
|
|
|
/// This can keep multiple texture pools, and return the current one as needed.
|
|
|
|
/// It is useful for applications that uses multiple texture pools.
|
|
|
|
/// </summary>
|
2019-10-13 02:02:07 -04:00
|
|
|
class TexturePoolCache
|
|
|
|
{
|
|
|
|
private const int MaxCapacity = 4;
|
|
|
|
|
2019-10-17 22:41:18 -04:00
|
|
|
private GpuContext _context;
|
2019-10-13 02:02:07 -04:00
|
|
|
|
|
|
|
private LinkedList<TexturePool> _pools;
|
|
|
|
|
2019-12-29 18:26:37 -05:00
|
|
|
/// <summary>
|
|
|
|
/// Constructs a new instance of the texture pool.
|
|
|
|
/// </summary>
|
2020-01-01 18:14:18 -05:00
|
|
|
/// <param name="context">GPU context that the texture pool belongs to</param>
|
2019-10-17 22:41:18 -04:00
|
|
|
public TexturePoolCache(GpuContext context)
|
2019-10-13 02:02:07 -04:00
|
|
|
{
|
2019-10-17 22:41:18 -04:00
|
|
|
_context = context;
|
2019-10-13 02:02:07 -04:00
|
|
|
|
|
|
|
_pools = new LinkedList<TexturePool>();
|
|
|
|
}
|
|
|
|
|
2019-12-29 18:26:37 -05:00
|
|
|
/// <summary>
|
|
|
|
/// Finds a cache texture pool, or creates a new one if not found.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="address">Start address of the texture pool</param>
|
|
|
|
/// <param name="maximumId">Maximum ID of the texture pool</param>
|
|
|
|
/// <returns>The found or newly created texture pool</returns>
|
2019-10-13 02:02:07 -04:00
|
|
|
public TexturePool FindOrCreate(ulong address, int maximumId)
|
|
|
|
{
|
|
|
|
TexturePool pool;
|
|
|
|
|
|
|
|
// First we try to find the pool.
|
|
|
|
for (LinkedListNode<TexturePool> node = _pools.First; node != null; node = node.Next)
|
|
|
|
{
|
|
|
|
pool = node.Value;
|
|
|
|
|
|
|
|
if (pool.Address == address)
|
|
|
|
{
|
|
|
|
if (pool.CacheNode != _pools.Last)
|
|
|
|
{
|
|
|
|
_pools.Remove(pool.CacheNode);
|
|
|
|
|
|
|
|
pool.CacheNode = _pools.AddLast(pool);
|
|
|
|
}
|
|
|
|
|
|
|
|
return pool;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// If not found, create a new one.
|
2019-10-17 22:41:18 -04:00
|
|
|
pool = new TexturePool(_context, address, maximumId);
|
2019-10-13 02:02:07 -04:00
|
|
|
|
|
|
|
pool.CacheNode = _pools.AddLast(pool);
|
|
|
|
|
|
|
|
if (_pools.Count > MaxCapacity)
|
|
|
|
{
|
|
|
|
TexturePool oldestPool = _pools.First.Value;
|
|
|
|
|
|
|
|
_pools.RemoveFirst();
|
|
|
|
|
|
|
|
oldestPool.Dispose();
|
|
|
|
|
|
|
|
oldestPool.CacheNode = null;
|
|
|
|
}
|
|
|
|
|
|
|
|
return pool;
|
|
|
|
}
|
|
|
|
|
2019-12-29 18:26:37 -05:00
|
|
|
/// <summary>
|
|
|
|
/// Invalidates a memory range of all intersecting texture pools on the cache.
|
|
|
|
/// </summary>
|
|
|
|
/// <param name="address">Start address of the range to invalidate</param>
|
|
|
|
/// <param name="size">Size of the range to invalidate</param>
|
2019-10-13 02:02:07 -04:00
|
|
|
public void InvalidateRange(ulong address, ulong size)
|
|
|
|
{
|
|
|
|
for (LinkedListNode<TexturePool> node = _pools.First; node != null; node = node.Next)
|
|
|
|
{
|
|
|
|
TexturePool pool = node.Value;
|
|
|
|
|
|
|
|
pool.InvalidateRange(address, size);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|