mirror of
https://github.com/Ryujinx/Ryujinx.git
synced 2025-01-01 15:55:59 +00:00
4965681e06
* GPU: Swap bindings array instead of copying Reduces work on UpdateShaderState. Now the cost is a few reference moves for arrays, rather than copying data. Downside: bindings arrays are no longer readonly. * Micro optimisation * Add missing docs * Address Feedback
515 lines
20 KiB
C#
515 lines
20 KiB
C#
using Ryujinx.Graphics.GAL;
|
|
using Ryujinx.Graphics.Gpu.Engine.Types;
|
|
using Ryujinx.Graphics.Gpu.Shader;
|
|
using System;
|
|
|
|
namespace Ryujinx.Graphics.Gpu.Image
|
|
{
|
|
/// <summary>
|
|
/// Texture manager.
|
|
/// </summary>
|
|
class TextureManager : IDisposable
|
|
{
|
|
private readonly GpuContext _context;
|
|
private readonly GpuChannel _channel;
|
|
|
|
private readonly TextureBindingsManager _cpBindingsManager;
|
|
private readonly TextureBindingsManager _gpBindingsManager;
|
|
private readonly TexturePoolCache _texturePoolCache;
|
|
private readonly SamplerPoolCache _samplerPoolCache;
|
|
|
|
private readonly Texture[] _rtColors;
|
|
private readonly ITexture[] _rtHostColors;
|
|
private Texture _rtDepthStencil;
|
|
private ITexture _rtHostDs;
|
|
|
|
public int ClipRegionWidth { get; private set; }
|
|
public int ClipRegionHeight { get; private set; }
|
|
|
|
/// <summary>
|
|
/// The scaling factor applied to all currently bound render targets.
|
|
/// </summary>
|
|
public float RenderTargetScale { get; private set; } = 1f;
|
|
|
|
/// <summary>
|
|
/// Creates a new instance of the texture manager.
|
|
/// </summary>
|
|
/// <param name="context">GPU context that the texture manager belongs to</param>
|
|
/// <param name="channel">GPU channel that the texture manager belongs to</param>
|
|
public TextureManager(GpuContext context, GpuChannel channel)
|
|
{
|
|
_context = context;
|
|
_channel = channel;
|
|
|
|
TexturePoolCache texturePoolCache = new TexturePoolCache(context);
|
|
SamplerPoolCache samplerPoolCache = new SamplerPoolCache(context);
|
|
|
|
float[] scales = new float[64];
|
|
new Span<float>(scales).Fill(1f);
|
|
|
|
_cpBindingsManager = new TextureBindingsManager(context, channel, texturePoolCache, samplerPoolCache, scales, isCompute: true);
|
|
_gpBindingsManager = new TextureBindingsManager(context, channel, texturePoolCache, samplerPoolCache, scales, isCompute: false);
|
|
_texturePoolCache = texturePoolCache;
|
|
_samplerPoolCache = samplerPoolCache;
|
|
|
|
_rtColors = new Texture[Constants.TotalRenderTargets];
|
|
_rtHostColors = new ITexture[Constants.TotalRenderTargets];
|
|
}
|
|
|
|
/// <summary>
|
|
/// Sets the texture and image bindings for the compute pipeline.
|
|
/// </summary>
|
|
/// <param name="bindings">Bindings for the active shader</param>
|
|
public void SetComputeBindings(CachedShaderBindings bindings)
|
|
{
|
|
_cpBindingsManager.SetBindings(bindings);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Sets the texture and image bindings for the graphics pipeline.
|
|
/// </summary>
|
|
/// <param name="bindings">Bindings for the active shader</param>
|
|
public void SetGraphicsBindings(CachedShaderBindings bindings)
|
|
{
|
|
_gpBindingsManager.SetBindings(bindings);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Sets the texture constant buffer index on the compute pipeline.
|
|
/// </summary>
|
|
/// <param name="index">The texture constant buffer index</param>
|
|
public void SetComputeTextureBufferIndex(int index)
|
|
{
|
|
_cpBindingsManager.SetTextureBufferIndex(index);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Sets the texture constant buffer index on the graphics pipeline.
|
|
/// </summary>
|
|
/// <param name="index">The texture constant buffer index</param>
|
|
public void SetGraphicsTextureBufferIndex(int index)
|
|
{
|
|
_gpBindingsManager.SetTextureBufferIndex(index);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Sets the current sampler pool on the compute pipeline.
|
|
/// </summary>
|
|
/// <param name="gpuVa">The start GPU virtual address of the sampler pool</param>
|
|
/// <param name="maximumId">The maximum ID of the sampler pool</param>
|
|
/// <param name="samplerIndex">The indexing type of the sampler pool</param>
|
|
public void SetComputeSamplerPool(ulong gpuVa, int maximumId, SamplerIndex samplerIndex)
|
|
{
|
|
_cpBindingsManager.SetSamplerPool(gpuVa, maximumId, samplerIndex);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Sets the current sampler pool on the graphics pipeline.
|
|
/// </summary>
|
|
/// <param name="gpuVa">The start GPU virtual address of the sampler pool</param>
|
|
/// <param name="maximumId">The maximum ID of the sampler pool</param>
|
|
/// <param name="samplerIndex">The indexing type of the sampler pool</param>
|
|
public void SetGraphicsSamplerPool(ulong gpuVa, int maximumId, SamplerIndex samplerIndex)
|
|
{
|
|
_gpBindingsManager.SetSamplerPool(gpuVa, maximumId, samplerIndex);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Sets the current texture pool on the compute pipeline.
|
|
/// </summary>
|
|
/// <param name="gpuVa">The start GPU virtual address of the texture pool</param>
|
|
/// <param name="maximumId">The maximum ID of the texture pool</param>
|
|
public void SetComputeTexturePool(ulong gpuVa, int maximumId)
|
|
{
|
|
_cpBindingsManager.SetTexturePool(gpuVa, maximumId);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Sets the current texture pool on the graphics pipeline.
|
|
/// </summary>
|
|
/// <param name="gpuVa">The start GPU virtual address of the texture pool</param>
|
|
/// <param name="maximumId">The maximum ID of the texture pool</param>
|
|
public void SetGraphicsTexturePool(ulong gpuVa, int maximumId)
|
|
{
|
|
_gpBindingsManager.SetTexturePool(gpuVa, maximumId);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Check if a texture's scale must be updated to match the configured resolution scale.
|
|
/// </summary>
|
|
/// <param name="texture">The texture to check</param>
|
|
/// <returns>True if the scale needs updating, false if the scale is up to date</returns>
|
|
private bool ScaleNeedsUpdated(Texture texture)
|
|
{
|
|
return texture != null && !(texture.ScaleMode == TextureScaleMode.Blacklisted || texture.ScaleMode == TextureScaleMode.Undesired) && texture.ScaleFactor != GraphicsConfig.ResScale;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Sets the render target color buffer.
|
|
/// </summary>
|
|
/// <param name="index">The index of the color buffer to set (up to 8)</param>
|
|
/// <param name="color">The color buffer texture</param>
|
|
/// <returns>True if render target scale must be updated.</returns>
|
|
public bool SetRenderTargetColor(int index, Texture color)
|
|
{
|
|
bool hasValue = color != null;
|
|
bool changesScale = (hasValue != (_rtColors[index] != null)) || (hasValue && RenderTargetScale != color.ScaleFactor);
|
|
|
|
if (_rtColors[index] != color)
|
|
{
|
|
_rtColors[index]?.SignalModifying(false);
|
|
|
|
if (color != null)
|
|
{
|
|
color.SynchronizeMemory();
|
|
color.SignalModifying(true);
|
|
}
|
|
|
|
_rtColors[index] = color;
|
|
}
|
|
|
|
return changesScale || ScaleNeedsUpdated(color);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Sets the render target depth-stencil buffer.
|
|
/// </summary>
|
|
/// <param name="depthStencil">The depth-stencil buffer texture</param>
|
|
/// <returns>True if render target scale must be updated.</returns>
|
|
public bool SetRenderTargetDepthStencil(Texture depthStencil)
|
|
{
|
|
bool hasValue = depthStencil != null;
|
|
bool changesScale = (hasValue != (_rtDepthStencil != null)) || (hasValue && RenderTargetScale != depthStencil.ScaleFactor);
|
|
|
|
if (_rtDepthStencil != depthStencil)
|
|
{
|
|
_rtDepthStencil?.SignalModifying(false);
|
|
|
|
if (depthStencil != null)
|
|
{
|
|
depthStencil.SynchronizeMemory();
|
|
depthStencil.SignalModifying(true);
|
|
}
|
|
|
|
_rtDepthStencil = depthStencil;
|
|
}
|
|
|
|
return changesScale || ScaleNeedsUpdated(depthStencil);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Sets the host clip region, which should be the intersection of all render target texture sizes.
|
|
/// </summary>
|
|
/// <param name="width">Width of the clip region, defined as the minimum width across all bound textures</param>
|
|
/// <param name="height">Height of the clip region, defined as the minimum height across all bound textures</param>
|
|
public void SetClipRegion(int width, int height)
|
|
{
|
|
ClipRegionWidth = width;
|
|
ClipRegionHeight = height;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the first available bound colour target, or the depth stencil target if not present.
|
|
/// </summary>
|
|
/// <returns>The first bound colour target, otherwise the depth stencil target</returns>
|
|
public Texture GetAnyRenderTarget()
|
|
{
|
|
return _rtColors[0] ?? _rtDepthStencil;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Updates the Render Target scale, given the currently bound render targets.
|
|
/// This will update scale to match the configured scale, scale textures that are eligible but not scaled,
|
|
/// and propagate blacklisted status from one texture to the ones bound with it.
|
|
/// </summary>
|
|
/// <param name="singleUse">If this is not -1, it indicates that only the given indexed target will be used.</param>
|
|
public void UpdateRenderTargetScale(int singleUse)
|
|
{
|
|
// Make sure all scales for render targets are at the highest they should be. Blacklisted targets should propagate their scale to the other targets.
|
|
bool mismatch = false;
|
|
bool blacklisted = false;
|
|
bool hasUpscaled = false;
|
|
bool hasUndesired = false;
|
|
float targetScale = GraphicsConfig.ResScale;
|
|
|
|
void ConsiderTarget(Texture target)
|
|
{
|
|
if (target == null) return;
|
|
float scale = target.ScaleFactor;
|
|
|
|
switch (target.ScaleMode)
|
|
{
|
|
case TextureScaleMode.Blacklisted:
|
|
mismatch |= scale != 1f;
|
|
blacklisted = true;
|
|
break;
|
|
case TextureScaleMode.Eligible:
|
|
mismatch = true; // We must make a decision.
|
|
break;
|
|
case TextureScaleMode.Undesired:
|
|
hasUndesired = true;
|
|
mismatch |= scale != 1f || hasUpscaled; // If another target is upscaled, scale this one up too.
|
|
break;
|
|
case TextureScaleMode.Scaled:
|
|
hasUpscaled = true;
|
|
mismatch |= hasUndesired || scale != targetScale; // If the target scale has changed, reset the scale for all targets.
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (singleUse != -1)
|
|
{
|
|
// If only one target is in use (by a clear, for example) the others do not need to be checked for mismatching scale.
|
|
ConsiderTarget(_rtColors[singleUse]);
|
|
}
|
|
else
|
|
{
|
|
foreach (Texture color in _rtColors)
|
|
{
|
|
ConsiderTarget(color);
|
|
}
|
|
}
|
|
|
|
ConsiderTarget(_rtDepthStencil);
|
|
|
|
mismatch |= blacklisted && hasUpscaled;
|
|
|
|
if (blacklisted || (hasUndesired && !hasUpscaled))
|
|
{
|
|
targetScale = 1f;
|
|
}
|
|
|
|
if (mismatch)
|
|
{
|
|
if (blacklisted)
|
|
{
|
|
// Propagate the blacklisted state to the other textures.
|
|
foreach (Texture color in _rtColors)
|
|
{
|
|
color?.BlacklistScale();
|
|
}
|
|
|
|
_rtDepthStencil?.BlacklistScale();
|
|
}
|
|
else
|
|
{
|
|
// Set the scale of the other textures.
|
|
foreach (Texture color in _rtColors)
|
|
{
|
|
color?.SetScale(targetScale);
|
|
}
|
|
|
|
_rtDepthStencil?.SetScale(targetScale);
|
|
}
|
|
}
|
|
|
|
RenderTargetScale = targetScale;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets a texture and a sampler from their respective pools from a texture ID and a sampler ID.
|
|
/// </summary>
|
|
/// <param name="textureId">ID of the texture</param>
|
|
/// <param name="samplerId">ID of the sampler</param>
|
|
public (Texture, Sampler) GetGraphicsTextureAndSampler(int textureId, int samplerId)
|
|
{
|
|
return _gpBindingsManager.GetTextureAndSampler(textureId, samplerId);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Commits bindings on the compute pipeline.
|
|
/// </summary>
|
|
/// <param name="specState">Specialization state for the bound shader</param>
|
|
/// <returns>True if all bound textures match the current shader specialization state, false otherwise</returns>
|
|
public bool CommitComputeBindings(ShaderSpecializationState specState)
|
|
{
|
|
// Every time we switch between graphics and compute work,
|
|
// we must rebind everything.
|
|
// Since compute work happens less often, we always do that
|
|
// before and after the compute dispatch.
|
|
|
|
_texturePoolCache.Tick();
|
|
_samplerPoolCache.Tick();
|
|
|
|
_cpBindingsManager.Rebind();
|
|
bool result = _cpBindingsManager.CommitBindings(specState);
|
|
_gpBindingsManager.Rebind();
|
|
|
|
return result;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Commits bindings on the graphics pipeline.
|
|
/// </summary>
|
|
/// <param name="specState">Specialization state for the bound shader</param>
|
|
/// <returns>True if all bound textures match the current shader specialization state, false otherwise</returns>
|
|
public bool CommitGraphicsBindings(ShaderSpecializationState specState)
|
|
{
|
|
_texturePoolCache.Tick();
|
|
_samplerPoolCache.Tick();
|
|
|
|
bool result = _gpBindingsManager.CommitBindings(specState);
|
|
|
|
UpdateRenderTargets();
|
|
|
|
return result;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Returns a texture pool from the cache, with the given address and maximum id.
|
|
/// </summary>
|
|
/// <param name="poolGpuVa">GPU virtual address of the texture pool</param>
|
|
/// <param name="maximumId">Maximum ID of the texture pool</param>
|
|
/// <returns>The texture pool</returns>
|
|
public TexturePool GetTexturePool(ulong poolGpuVa, int maximumId)
|
|
{
|
|
ulong poolAddress = _channel.MemoryManager.Translate(poolGpuVa);
|
|
|
|
TexturePool texturePool = _texturePoolCache.FindOrCreate(_channel, poolAddress, maximumId);
|
|
|
|
return texturePool;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets a texture descriptor used on the compute pipeline.
|
|
/// </summary>
|
|
/// <param name="poolGpuVa">GPU virtual address of the texture pool</param>
|
|
/// <param name="bufferIndex">Index of the constant buffer with texture handles</param>
|
|
/// <param name="maximumId">Maximum ID of the texture pool</param>
|
|
/// <param name="handle">Shader "fake" handle of the texture</param>
|
|
/// <param name="cbufSlot">Shader constant buffer slot of the texture</param>
|
|
/// <returns>The texture descriptor</returns>
|
|
public TextureDescriptor GetComputeTextureDescriptor(ulong poolGpuVa, int bufferIndex, int maximumId, int handle, int cbufSlot)
|
|
{
|
|
return _cpBindingsManager.GetTextureDescriptor(poolGpuVa, bufferIndex, maximumId, 0, handle, cbufSlot);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets a texture descriptor used on the graphics pipeline.
|
|
/// </summary>
|
|
/// <param name="poolGpuVa">GPU virtual address of the texture pool</param>
|
|
/// <param name="bufferIndex">Index of the constant buffer with texture handles</param>
|
|
/// <param name="maximumId">Maximum ID of the texture pool</param>
|
|
/// <param name="stageIndex">Index of the shader stage where the texture is bound</param>
|
|
/// <param name="handle">Shader "fake" handle of the texture</param>
|
|
/// <param name="cbufSlot">Shader constant buffer slot of the texture</param>
|
|
/// <returns>The texture descriptor</returns>
|
|
public TextureDescriptor GetGraphicsTextureDescriptor(
|
|
ulong poolGpuVa,
|
|
int bufferIndex,
|
|
int maximumId,
|
|
int stageIndex,
|
|
int handle,
|
|
int cbufSlot)
|
|
{
|
|
return _gpBindingsManager.GetTextureDescriptor(poolGpuVa, bufferIndex, maximumId, stageIndex, handle, cbufSlot);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Update host framebuffer attachments based on currently bound render target buffers.
|
|
/// </summary>
|
|
public void UpdateRenderTargets()
|
|
{
|
|
bool anyChanged = false;
|
|
|
|
if (_rtHostDs != _rtDepthStencil?.HostTexture)
|
|
{
|
|
_rtHostDs = _rtDepthStencil?.HostTexture;
|
|
|
|
anyChanged = true;
|
|
}
|
|
|
|
for (int index = 0; index < _rtColors.Length; index++)
|
|
{
|
|
ITexture hostTexture = _rtColors[index]?.HostTexture;
|
|
|
|
if (_rtHostColors[index] != hostTexture)
|
|
{
|
|
_rtHostColors[index] = hostTexture;
|
|
|
|
anyChanged = true;
|
|
}
|
|
}
|
|
|
|
if (anyChanged)
|
|
{
|
|
_context.Renderer.Pipeline.SetRenderTargets(_rtHostColors, _rtHostDs);
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Update host framebuffer attachments based on currently bound render target buffers.
|
|
/// </summary>
|
|
/// <remarks>
|
|
/// All attachments other than <paramref name="index"/> will be unbound.
|
|
/// </remarks>
|
|
/// <param name="index">Index of the render target color to be updated</param>
|
|
public void UpdateRenderTarget(int index)
|
|
{
|
|
new Span<ITexture>(_rtHostColors).Fill(null);
|
|
_rtHostColors[index] = _rtColors[index]?.HostTexture;
|
|
_rtHostDs = null;
|
|
|
|
_context.Renderer.Pipeline.SetRenderTargets(_rtHostColors, null);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Update host framebuffer attachments based on currently bound render target buffers.
|
|
/// </summary>
|
|
/// <remarks>
|
|
/// All color attachments will be unbound.
|
|
/// </remarks>
|
|
public void UpdateRenderTargetDepthStencil()
|
|
{
|
|
new Span<ITexture>(_rtHostColors).Fill(null);
|
|
_rtHostDs = _rtDepthStencil?.HostTexture;
|
|
|
|
_context.Renderer.Pipeline.SetRenderTargets(_rtHostColors, _rtHostDs);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Forces the texture and sampler pools to be re-loaded from the cache on next use.
|
|
/// </summary>
|
|
public void ReloadPools()
|
|
{
|
|
_cpBindingsManager.ReloadPools();
|
|
_gpBindingsManager.ReloadPools();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Forces all textures, samplers, images and render targets to be rebound the next time
|
|
/// CommitGraphicsBindings is called.
|
|
/// </summary>
|
|
public void Rebind()
|
|
{
|
|
_gpBindingsManager.Rebind();
|
|
|
|
for (int index = 0; index < _rtHostColors.Length; index++)
|
|
{
|
|
_rtHostColors[index] = null;
|
|
}
|
|
|
|
_rtHostDs = null;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Disposes the texture manager.
|
|
/// It's an error to use the texture manager after disposal.
|
|
/// </summary>
|
|
public void Dispose()
|
|
{
|
|
// Textures are owned by the texture cache, so we shouldn't dispose the texture pool cache.
|
|
_samplerPoolCache.Dispose();
|
|
|
|
for (int i = 0; i < _rtColors.Length; i++)
|
|
{
|
|
_rtColors[i]?.DecrementReferenceCount();
|
|
_rtColors[i] = null;
|
|
}
|
|
|
|
_rtDepthStencil?.DecrementReferenceCount();
|
|
_rtDepthStencil = null;
|
|
}
|
|
}
|
|
}
|