mirror of
https://github.com/GreemDev/Ryujinx
synced 2025-01-12 05:51:25 +01:00
615 lines
25 KiB
C#
615 lines
25 KiB
C#
|
using Ryujinx.Common.Memory;
|
||
|
using Ryujinx.Graphics.Gpu.Shader.DiskCache;
|
||
|
using Ryujinx.Graphics.Shader;
|
||
|
using System;
|
||
|
using System.Collections.Generic;
|
||
|
using System.Numerics;
|
||
|
|
||
|
namespace Ryujinx.Graphics.Gpu.Shader
|
||
|
{
|
||
|
class ShaderSpecializationState
|
||
|
{
|
||
|
private const uint ComsMagic = (byte)'C' | ((byte)'O' << 8) | ((byte)'M' << 16) | ((byte)'S' << 24);
|
||
|
private const uint GfxsMagic = (byte)'G' | ((byte)'F' << 8) | ((byte)'X' << 16) | ((byte)'S' << 24);
|
||
|
private const uint TfbdMagic = (byte)'T' | ((byte)'F' << 8) | ((byte)'B' << 16) | ((byte)'D' << 24);
|
||
|
private const uint TexkMagic = (byte)'T' | ((byte)'E' << 8) | ((byte)'X' << 16) | ((byte)'K' << 24);
|
||
|
private const uint TexsMagic = (byte)'T' | ((byte)'E' << 8) | ((byte)'X' << 16) | ((byte)'S' << 24);
|
||
|
|
||
|
/// <summary>
|
||
|
/// Flags indicating GPU state that is used by the shader.
|
||
|
/// </summary>
|
||
|
[Flags]
|
||
|
private enum QueriedStateFlags
|
||
|
{
|
||
|
EarlyZForce = 1 << 0,
|
||
|
PrimitiveTopology = 1 << 1,
|
||
|
TessellationMode = 1 << 2,
|
||
|
TransformFeedback = 1 << 3
|
||
|
}
|
||
|
|
||
|
private QueriedStateFlags _queriedState;
|
||
|
private bool _compute;
|
||
|
private byte _constantBufferUsePerStage;
|
||
|
|
||
|
/// <summary>
|
||
|
/// Compute engine state.
|
||
|
/// </summary>
|
||
|
public GpuChannelComputeState ComputeState;
|
||
|
|
||
|
/// <summary>
|
||
|
/// 3D engine state.
|
||
|
/// </summary>
|
||
|
public GpuChannelGraphicsState GraphicsState;
|
||
|
|
||
|
/// <summary>
|
||
|
/// Contant buffers bound at the time the shader was compiled, per stage.
|
||
|
/// </summary>
|
||
|
public Array5<uint> ConstantBufferUse;
|
||
|
|
||
|
/// <summary>
|
||
|
/// Transform feedback buffers active at the time the shader was compiled.
|
||
|
/// </summary>
|
||
|
public TransformFeedbackDescriptor[] TransformFeedbackDescriptors;
|
||
|
|
||
|
/// <summary>
|
||
|
/// Flags indicating texture state that is used by the shader.
|
||
|
/// </summary>
|
||
|
[Flags]
|
||
|
private enum QueriedTextureStateFlags
|
||
|
{
|
||
|
TextureFormat = 1 << 0,
|
||
|
SamplerType = 1 << 1,
|
||
|
CoordNormalized = 1 << 2
|
||
|
}
|
||
|
|
||
|
/// <summary>
|
||
|
/// Reference type wrapping a value.
|
||
|
/// </summary>
|
||
|
private class Box<T>
|
||
|
{
|
||
|
/// <summary>
|
||
|
/// Wrapped value.
|
||
|
/// </summary>
|
||
|
public T Value;
|
||
|
}
|
||
|
|
||
|
/// <summary>
|
||
|
/// State of a texture or image that is accessed by the shader.
|
||
|
/// </summary>
|
||
|
private struct TextureSpecializationState
|
||
|
{
|
||
|
// New fields should be added to the end of the struct to keep disk shader cache compatibility.
|
||
|
|
||
|
/// <summary>
|
||
|
/// Flags indicating which state of the texture the shader depends on.
|
||
|
/// </summary>
|
||
|
public QueriedTextureStateFlags QueriedFlags;
|
||
|
|
||
|
/// <summary>
|
||
|
/// Encoded texture format value.
|
||
|
/// </summary>
|
||
|
public uint Format;
|
||
|
|
||
|
/// <summary>
|
||
|
/// True if the texture format is sRGB, false otherwise.
|
||
|
/// </summary>
|
||
|
public bool FormatSrgb;
|
||
|
|
||
|
/// <summary>
|
||
|
/// Texture target.
|
||
|
/// </summary>
|
||
|
public Image.TextureTarget TextureTarget;
|
||
|
|
||
|
/// <summary>
|
||
|
/// Indicates if the coordinates used to sample the texture are normalized or not (0.0..1.0 or 0..Width/Height).
|
||
|
/// </summary>
|
||
|
public bool CoordNormalized;
|
||
|
}
|
||
|
|
||
|
/// <summary>
|
||
|
/// Texture binding information, used to identify each texture accessed by the shader.
|
||
|
/// </summary>
|
||
|
private struct TextureKey : IEquatable<TextureKey>
|
||
|
{
|
||
|
// New fields should be added to the end of the struct to keep disk shader cache compatibility.
|
||
|
|
||
|
/// <summary>
|
||
|
/// Shader stage where the texture is used.
|
||
|
/// </summary>
|
||
|
public readonly int StageIndex;
|
||
|
|
||
|
/// <summary>
|
||
|
/// Texture handle offset in words on the texture buffer.
|
||
|
/// </summary>
|
||
|
public readonly int Handle;
|
||
|
|
||
|
/// <summary>
|
||
|
/// Constant buffer slot of the texture buffer (-1 to use the texture buffer index GPU register).
|
||
|
/// </summary>
|
||
|
public readonly int CbufSlot;
|
||
|
|
||
|
/// <summary>
|
||
|
/// Creates a new texture key.
|
||
|
/// </summary>
|
||
|
/// <param name="stageIndex">Shader stage where the texture is used</param>
|
||
|
/// <param name="handle">Texture handle offset in words on the texture buffer</param>
|
||
|
/// <param name="cbufSlot">Constant buffer slot of the texture buffer (-1 to use the texture buffer index GPU register)</param>
|
||
|
public TextureKey(int stageIndex, int handle, int cbufSlot)
|
||
|
{
|
||
|
StageIndex = stageIndex;
|
||
|
Handle = handle;
|
||
|
CbufSlot = cbufSlot;
|
||
|
}
|
||
|
|
||
|
public override bool Equals(object obj)
|
||
|
{
|
||
|
return obj is TextureKey textureKey && Equals(textureKey);
|
||
|
}
|
||
|
|
||
|
public bool Equals(TextureKey other)
|
||
|
{
|
||
|
return StageIndex == other.StageIndex && Handle == other.Handle && CbufSlot == other.CbufSlot;
|
||
|
}
|
||
|
|
||
|
public override int GetHashCode()
|
||
|
{
|
||
|
return HashCode.Combine(StageIndex, Handle, CbufSlot);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
private readonly Dictionary<TextureKey, Box<TextureSpecializationState>> _textureSpecialization;
|
||
|
|
||
|
/// <summary>
|
||
|
/// Creates a new instance of the shader specialization state.
|
||
|
/// </summary>
|
||
|
private ShaderSpecializationState()
|
||
|
{
|
||
|
_textureSpecialization = new Dictionary<TextureKey, Box<TextureSpecializationState>>();
|
||
|
}
|
||
|
|
||
|
/// <summary>
|
||
|
/// Creates a new instance of the shader specialization state.
|
||
|
/// </summary>
|
||
|
/// <param name="state">Current compute engine state</param>
|
||
|
public ShaderSpecializationState(GpuChannelComputeState state) : this()
|
||
|
{
|
||
|
ComputeState = state;
|
||
|
_compute = true;
|
||
|
}
|
||
|
|
||
|
/// <summary>
|
||
|
/// Creates a new instance of the shader specialization state.
|
||
|
/// </summary>
|
||
|
/// <param name="state">Current 3D engine state</param>
|
||
|
/// <param name="descriptors">Optional transform feedback buffers in use, if any</param>
|
||
|
public ShaderSpecializationState(GpuChannelGraphicsState state, TransformFeedbackDescriptor[] descriptors) : this()
|
||
|
{
|
||
|
GraphicsState = state;
|
||
|
_compute = false;
|
||
|
|
||
|
if (descriptors != null)
|
||
|
{
|
||
|
TransformFeedbackDescriptors = descriptors;
|
||
|
_queriedState |= QueriedStateFlags.TransformFeedback;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/// <summary>
|
||
|
/// Indicates that the shader accesses the early Z force state.
|
||
|
/// </summary>
|
||
|
public void RecordEarlyZForce()
|
||
|
{
|
||
|
_queriedState |= QueriedStateFlags.EarlyZForce;
|
||
|
}
|
||
|
|
||
|
/// <summary>
|
||
|
/// Indicates that the shader accesses the primitive topology state.
|
||
|
/// </summary>
|
||
|
public void RecordPrimitiveTopology()
|
||
|
{
|
||
|
_queriedState |= QueriedStateFlags.PrimitiveTopology;
|
||
|
}
|
||
|
|
||
|
/// <summary>
|
||
|
/// Indicates that the shader accesses the tessellation mode state.
|
||
|
/// </summary>
|
||
|
public void RecordTessellationMode()
|
||
|
{
|
||
|
_queriedState |= QueriedStateFlags.TessellationMode;
|
||
|
}
|
||
|
|
||
|
/// <summary>
|
||
|
/// Indicates that the shader accesses the constant buffer use state.
|
||
|
/// </summary>
|
||
|
/// <param name="stageIndex">Shader stage index</param>
|
||
|
/// <param name="useMask">Mask indicating the constant buffers bound at the time of the shader compilation</param>
|
||
|
public void RecordConstantBufferUse(int stageIndex, uint useMask)
|
||
|
{
|
||
|
ConstantBufferUse[stageIndex] = useMask;
|
||
|
_constantBufferUsePerStage |= (byte)(1 << stageIndex);
|
||
|
}
|
||
|
|
||
|
/// <summary>
|
||
|
/// Indicates that a given texture is accessed by the shader.
|
||
|
/// </summary>
|
||
|
/// <param name="stageIndex">Shader stage where the texture is used</param>
|
||
|
/// <param name="handle">Offset in words of the texture handle on the texture buffer</param>
|
||
|
/// <param name="cbufSlot">Slot of the texture buffer constant buffer</param>
|
||
|
/// <param name="descriptor">Descriptor of the texture</param>
|
||
|
public void RegisterTexture(int stageIndex, int handle, int cbufSlot, Image.TextureDescriptor descriptor)
|
||
|
{
|
||
|
Box<TextureSpecializationState> state = GetOrCreateTextureSpecState(stageIndex, handle, cbufSlot);
|
||
|
state.Value.Format = descriptor.UnpackFormat();
|
||
|
state.Value.FormatSrgb = descriptor.UnpackSrgb();
|
||
|
state.Value.TextureTarget = descriptor.UnpackTextureTarget();
|
||
|
state.Value.CoordNormalized = descriptor.UnpackTextureCoordNormalized();
|
||
|
}
|
||
|
|
||
|
/// <summary>
|
||
|
/// Indicates that a given texture is accessed by the shader.
|
||
|
/// </summary>
|
||
|
/// <param name="stageIndex">Shader stage where the texture is used</param>
|
||
|
/// <param name="handle">Offset in words of the texture handle on the texture buffer</param>
|
||
|
/// <param name="cbufSlot">Slot of the texture buffer constant buffer</param>
|
||
|
/// <param name="format">Maxwell texture format value</param>
|
||
|
/// <param name="formatSrgb">Whenever the texture format is a sRGB format</param>
|
||
|
/// <param name="target">Texture target type</param>
|
||
|
/// <param name="coordNormalized">Whenever the texture coordinates used on the shader are considered normalized</param>
|
||
|
public void RegisterTexture(
|
||
|
int stageIndex,
|
||
|
int handle,
|
||
|
int cbufSlot,
|
||
|
uint format,
|
||
|
bool formatSrgb,
|
||
|
Image.TextureTarget target,
|
||
|
bool coordNormalized)
|
||
|
{
|
||
|
Box<TextureSpecializationState> state = GetOrCreateTextureSpecState(stageIndex, handle, cbufSlot);
|
||
|
state.Value.Format = format;
|
||
|
state.Value.FormatSrgb = formatSrgb;
|
||
|
state.Value.TextureTarget = target;
|
||
|
state.Value.CoordNormalized = coordNormalized;
|
||
|
}
|
||
|
|
||
|
/// <summary>
|
||
|
/// Indicates that the format of a given texture was used during the shader translation process.
|
||
|
/// </summary>
|
||
|
/// <param name="stageIndex">Shader stage where the texture is used</param>
|
||
|
/// <param name="handle">Offset in words of the texture handle on the texture buffer</param>
|
||
|
/// <param name="cbufSlot">Slot of the texture buffer constant buffer</param>
|
||
|
public void RecordTextureFormat(int stageIndex, int handle, int cbufSlot)
|
||
|
{
|
||
|
Box<TextureSpecializationState> state = GetOrCreateTextureSpecState(stageIndex, handle, cbufSlot);
|
||
|
state.Value.QueriedFlags |= QueriedTextureStateFlags.TextureFormat;
|
||
|
}
|
||
|
|
||
|
/// <summary>
|
||
|
/// Indicates that the target of a given texture was used during the shader translation process.
|
||
|
/// </summary>
|
||
|
/// <param name="stageIndex">Shader stage where the texture is used</param>
|
||
|
/// <param name="handle">Offset in words of the texture handle on the texture buffer</param>
|
||
|
/// <param name="cbufSlot">Slot of the texture buffer constant buffer</param>
|
||
|
public void RecordTextureSamplerType(int stageIndex, int handle, int cbufSlot)
|
||
|
{
|
||
|
Box<TextureSpecializationState> state = GetOrCreateTextureSpecState(stageIndex, handle, cbufSlot);
|
||
|
state.Value.QueriedFlags |= QueriedTextureStateFlags.SamplerType;
|
||
|
}
|
||
|
|
||
|
/// <summary>
|
||
|
/// Indicates that the coordinate normalization state of a given texture was used during the shader translation process.
|
||
|
/// </summary>
|
||
|
/// <param name="stageIndex">Shader stage where the texture is used</param>
|
||
|
/// <param name="handle">Offset in words of the texture handle on the texture buffer</param>
|
||
|
/// <param name="cbufSlot">Slot of the texture buffer constant buffer</param>
|
||
|
public void RecordTextureCoordNormalized(int stageIndex, int handle, int cbufSlot)
|
||
|
{
|
||
|
Box<TextureSpecializationState> state = GetOrCreateTextureSpecState(stageIndex, handle, cbufSlot);
|
||
|
state.Value.QueriedFlags |= QueriedTextureStateFlags.CoordNormalized;
|
||
|
}
|
||
|
|
||
|
/// <summary>
|
||
|
/// Checks if a given texture was registerd on this specialization state.
|
||
|
/// </summary>
|
||
|
/// <param name="stageIndex">Shader stage where the texture is used</param>
|
||
|
/// <param name="handle">Offset in words of the texture handle on the texture buffer</param>
|
||
|
/// <param name="cbufSlot">Slot of the texture buffer constant buffer</param>
|
||
|
public bool TextureRegistered(int stageIndex, int handle, int cbufSlot)
|
||
|
{
|
||
|
return GetTextureSpecState(stageIndex, handle, cbufSlot) != null;
|
||
|
}
|
||
|
|
||
|
/// <summary>
|
||
|
/// Gets the recorded format of a given texture.
|
||
|
/// </summary>
|
||
|
/// <param name="stageIndex">Shader stage where the texture is used</param>
|
||
|
/// <param name="handle">Offset in words of the texture handle on the texture buffer</param>
|
||
|
/// <param name="cbufSlot">Slot of the texture buffer constant buffer</param>
|
||
|
public (uint, bool) GetFormat(int stageIndex, int handle, int cbufSlot)
|
||
|
{
|
||
|
TextureSpecializationState state = GetTextureSpecState(stageIndex, handle, cbufSlot).Value;
|
||
|
return (state.Format, state.FormatSrgb);
|
||
|
}
|
||
|
|
||
|
/// <summary>
|
||
|
/// Gets the recorded target of a given texture.
|
||
|
/// </summary>
|
||
|
/// <param name="stageIndex">Shader stage where the texture is used</param>
|
||
|
/// <param name="handle">Offset in words of the texture handle on the texture buffer</param>
|
||
|
/// <param name="cbufSlot">Slot of the texture buffer constant buffer</param>
|
||
|
public Image.TextureTarget GetTextureTarget(int stageIndex, int handle, int cbufSlot)
|
||
|
{
|
||
|
return GetTextureSpecState(stageIndex, handle, cbufSlot).Value.TextureTarget;
|
||
|
}
|
||
|
|
||
|
/// <summary>
|
||
|
/// Gets the recorded coordinate normalization state of a given texture.
|
||
|
/// </summary>
|
||
|
/// <param name="stageIndex">Shader stage where the texture is used</param>
|
||
|
/// <param name="handle">Offset in words of the texture handle on the texture buffer</param>
|
||
|
/// <param name="cbufSlot">Slot of the texture buffer constant buffer</param>
|
||
|
public bool GetCoordNormalized(int stageIndex, int handle, int cbufSlot)
|
||
|
{
|
||
|
return GetTextureSpecState(stageIndex, handle, cbufSlot).Value.CoordNormalized;
|
||
|
}
|
||
|
|
||
|
/// <summary>
|
||
|
/// Gets texture specialization state for a given texture, or create a new one if not present.
|
||
|
/// </summary>
|
||
|
/// <param name="stageIndex">Shader stage where the texture is used</param>
|
||
|
/// <param name="handle">Offset in words of the texture handle on the texture buffer</param>
|
||
|
/// <param name="cbufSlot">Slot of the texture buffer constant buffer</param>
|
||
|
/// <returns>Texture specialization state</returns>
|
||
|
private Box<TextureSpecializationState> GetOrCreateTextureSpecState(int stageIndex, int handle, int cbufSlot)
|
||
|
{
|
||
|
TextureKey key = new TextureKey(stageIndex, handle, cbufSlot);
|
||
|
|
||
|
if (!_textureSpecialization.TryGetValue(key, out Box<TextureSpecializationState> state))
|
||
|
{
|
||
|
_textureSpecialization.Add(key, state = new Box<TextureSpecializationState>());
|
||
|
}
|
||
|
|
||
|
return state;
|
||
|
}
|
||
|
|
||
|
/// <summary>
|
||
|
/// Gets texture specialization state for a given texture.
|
||
|
/// </summary>
|
||
|
/// <param name="stageIndex">Shader stage where the texture is used</param>
|
||
|
/// <param name="handle">Offset in words of the texture handle on the texture buffer</param>
|
||
|
/// <param name="cbufSlot">Slot of the texture buffer constant buffer</param>
|
||
|
/// <returns>Texture specialization state</returns>
|
||
|
private Box<TextureSpecializationState> GetTextureSpecState(int stageIndex, int handle, int cbufSlot)
|
||
|
{
|
||
|
TextureKey key = new TextureKey(stageIndex, handle, cbufSlot);
|
||
|
|
||
|
if (_textureSpecialization.TryGetValue(key, out Box<TextureSpecializationState> state))
|
||
|
{
|
||
|
return state;
|
||
|
}
|
||
|
|
||
|
return null;
|
||
|
}
|
||
|
|
||
|
/// <summary>
|
||
|
/// Checks if the recorded state matches the current GPU 3D engine state.
|
||
|
/// </summary>
|
||
|
/// <param name="channel">GPU channel</param>
|
||
|
/// <param name="poolState">Texture pool state</param>
|
||
|
/// <returns>True if the state matches, false otherwise</returns>
|
||
|
public bool MatchesGraphics(GpuChannel channel, GpuChannelPoolState poolState)
|
||
|
{
|
||
|
return Matches(channel, poolState, isCompute: false);
|
||
|
}
|
||
|
|
||
|
/// <summary>
|
||
|
/// Checks if the recorded state matches the current GPU compute engine state.
|
||
|
/// </summary>
|
||
|
/// <param name="channel">GPU channel</param>
|
||
|
/// <param name="poolState">Texture pool state</param>
|
||
|
/// <returns>True if the state matches, false otherwise</returns>
|
||
|
public bool MatchesCompute(GpuChannel channel, GpuChannelPoolState poolState)
|
||
|
{
|
||
|
return Matches(channel, poolState, isCompute: true);
|
||
|
}
|
||
|
|
||
|
/// <summary>
|
||
|
/// Checks if the recorded state matches the current GPU state.
|
||
|
/// </summary>
|
||
|
/// <param name="channel">GPU channel</param>
|
||
|
/// <param name="poolState">Texture pool state</param>
|
||
|
/// <param name="isCompute">Indicates whenever the check is requested by the 3D or compute engine</param>
|
||
|
/// <returns>True if the state matches, false otherwise</returns>
|
||
|
private bool Matches(GpuChannel channel, GpuChannelPoolState poolState, bool isCompute)
|
||
|
{
|
||
|
int constantBufferUsePerStageMask = _constantBufferUsePerStage;
|
||
|
|
||
|
while (constantBufferUsePerStageMask != 0)
|
||
|
{
|
||
|
int index = BitOperations.TrailingZeroCount(constantBufferUsePerStageMask);
|
||
|
|
||
|
uint useMask = isCompute
|
||
|
? channel.BufferManager.GetComputeUniformBufferUseMask()
|
||
|
: channel.BufferManager.GetGraphicsUniformBufferUseMask(index);
|
||
|
|
||
|
if (ConstantBufferUse[index] != useMask)
|
||
|
{
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
constantBufferUsePerStageMask &= ~(1 << index);
|
||
|
}
|
||
|
|
||
|
foreach (var kv in _textureSpecialization)
|
||
|
{
|
||
|
TextureKey textureKey = kv.Key;
|
||
|
|
||
|
(int textureBufferIndex, int samplerBufferIndex) = TextureHandle.UnpackSlots(textureKey.CbufSlot, poolState.TextureBufferIndex);
|
||
|
|
||
|
ulong textureCbAddress;
|
||
|
ulong samplerCbAddress;
|
||
|
|
||
|
if (isCompute)
|
||
|
{
|
||
|
textureCbAddress = channel.BufferManager.GetComputeUniformBufferAddress(textureBufferIndex);
|
||
|
samplerCbAddress = channel.BufferManager.GetComputeUniformBufferAddress(samplerBufferIndex);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
textureCbAddress = channel.BufferManager.GetGraphicsUniformBufferAddress(textureKey.StageIndex, textureBufferIndex);
|
||
|
samplerCbAddress = channel.BufferManager.GetGraphicsUniformBufferAddress(textureKey.StageIndex, samplerBufferIndex);
|
||
|
}
|
||
|
|
||
|
if (!channel.MemoryManager.Physical.IsMapped(textureCbAddress) || !channel.MemoryManager.Physical.IsMapped(samplerCbAddress))
|
||
|
{
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
Image.TextureDescriptor descriptor;
|
||
|
|
||
|
if (isCompute)
|
||
|
{
|
||
|
descriptor = channel.TextureManager.GetComputeTextureDescriptor(
|
||
|
poolState.TexturePoolGpuVa,
|
||
|
poolState.TextureBufferIndex,
|
||
|
poolState.TexturePoolMaximumId,
|
||
|
textureKey.Handle,
|
||
|
textureKey.CbufSlot);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
descriptor = channel.TextureManager.GetGraphicsTextureDescriptor(
|
||
|
poolState.TexturePoolGpuVa,
|
||
|
poolState.TextureBufferIndex,
|
||
|
poolState.TexturePoolMaximumId,
|
||
|
textureKey.StageIndex,
|
||
|
textureKey.Handle,
|
||
|
textureKey.CbufSlot);
|
||
|
}
|
||
|
|
||
|
Box<TextureSpecializationState> specializationState = kv.Value;
|
||
|
|
||
|
if (specializationState.Value.QueriedFlags.HasFlag(QueriedTextureStateFlags.CoordNormalized) &&
|
||
|
specializationState.Value.CoordNormalized != descriptor.UnpackTextureCoordNormalized())
|
||
|
{
|
||
|
return false;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
/// <summary>
|
||
|
/// Reads shader specialization state that has been serialized.
|
||
|
/// </summary>
|
||
|
/// <param name="dataReader">Data reader</param>
|
||
|
/// <returns>Shader specialization state</returns>
|
||
|
public static ShaderSpecializationState Read(ref BinarySerializer dataReader)
|
||
|
{
|
||
|
ShaderSpecializationState specState = new ShaderSpecializationState();
|
||
|
|
||
|
dataReader.Read(ref specState._queriedState);
|
||
|
dataReader.Read(ref specState._compute);
|
||
|
|
||
|
if (specState._compute)
|
||
|
{
|
||
|
dataReader.ReadWithMagicAndSize(ref specState.ComputeState, ComsMagic);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
dataReader.ReadWithMagicAndSize(ref specState.GraphicsState, GfxsMagic);
|
||
|
}
|
||
|
|
||
|
dataReader.Read(ref specState._constantBufferUsePerStage);
|
||
|
|
||
|
int constantBufferUsePerStageMask = specState._constantBufferUsePerStage;
|
||
|
|
||
|
while (constantBufferUsePerStageMask != 0)
|
||
|
{
|
||
|
int index = BitOperations.TrailingZeroCount(constantBufferUsePerStageMask);
|
||
|
dataReader.Read(ref specState.ConstantBufferUse[index]);
|
||
|
constantBufferUsePerStageMask &= ~(1 << index);
|
||
|
}
|
||
|
|
||
|
if (specState._queriedState.HasFlag(QueriedStateFlags.TransformFeedback))
|
||
|
{
|
||
|
ushort tfCount = 0;
|
||
|
dataReader.Read(ref tfCount);
|
||
|
specState.TransformFeedbackDescriptors = new TransformFeedbackDescriptor[tfCount];
|
||
|
|
||
|
for (int index = 0; index < tfCount; index++)
|
||
|
{
|
||
|
dataReader.ReadWithMagicAndSize(ref specState.TransformFeedbackDescriptors[index], TfbdMagic);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
ushort count = 0;
|
||
|
dataReader.Read(ref count);
|
||
|
|
||
|
for (int index = 0; index < count; index++)
|
||
|
{
|
||
|
TextureKey textureKey = default;
|
||
|
Box<TextureSpecializationState> textureState = new Box<TextureSpecializationState>();
|
||
|
|
||
|
dataReader.ReadWithMagicAndSize(ref textureKey, TexkMagic);
|
||
|
dataReader.ReadWithMagicAndSize(ref textureState.Value, TexsMagic);
|
||
|
|
||
|
specState._textureSpecialization[textureKey] = textureState;
|
||
|
}
|
||
|
|
||
|
return specState;
|
||
|
}
|
||
|
|
||
|
/// <summary>
|
||
|
/// Serializes the shader specialization state.
|
||
|
/// </summary>
|
||
|
/// <param name="dataWriter">Data writer</param>
|
||
|
public void Write(ref BinarySerializer dataWriter)
|
||
|
{
|
||
|
dataWriter.Write(ref _queriedState);
|
||
|
dataWriter.Write(ref _compute);
|
||
|
|
||
|
if (_compute)
|
||
|
{
|
||
|
dataWriter.WriteWithMagicAndSize(ref ComputeState, ComsMagic);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
dataWriter.WriteWithMagicAndSize(ref GraphicsState, GfxsMagic);
|
||
|
}
|
||
|
|
||
|
dataWriter.Write(ref _constantBufferUsePerStage);
|
||
|
|
||
|
int constantBufferUsePerStageMask = _constantBufferUsePerStage;
|
||
|
|
||
|
while (constantBufferUsePerStageMask != 0)
|
||
|
{
|
||
|
int index = BitOperations.TrailingZeroCount(constantBufferUsePerStageMask);
|
||
|
dataWriter.Write(ref ConstantBufferUse[index]);
|
||
|
constantBufferUsePerStageMask &= ~(1 << index);
|
||
|
}
|
||
|
|
||
|
if (_queriedState.HasFlag(QueriedStateFlags.TransformFeedback))
|
||
|
{
|
||
|
ushort tfCount = (ushort)TransformFeedbackDescriptors.Length;
|
||
|
dataWriter.Write(ref tfCount);
|
||
|
|
||
|
for (int index = 0; index < TransformFeedbackDescriptors.Length; index++)
|
||
|
{
|
||
|
dataWriter.WriteWithMagicAndSize(ref TransformFeedbackDescriptors[index], TfbdMagic);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
ushort count = (ushort)_textureSpecialization.Count;
|
||
|
dataWriter.Write(ref count);
|
||
|
|
||
|
foreach (var kv in _textureSpecialization)
|
||
|
{
|
||
|
var textureKey = kv.Key;
|
||
|
var textureState = kv.Value;
|
||
|
|
||
|
dataWriter.WriteWithMagicAndSize(ref textureKey, TexkMagic);
|
||
|
dataWriter.WriteWithMagicAndSize(ref textureState.Value, TexsMagic);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|