diff --git a/Ryujinx.Graphics.Gpu/Shader/ShaderCache.cs b/Ryujinx.Graphics.Gpu/Shader/ShaderCache.cs
index ae128ed4..e6d46884 100644
--- a/Ryujinx.Graphics.Gpu/Shader/ShaderCache.cs
+++ b/Ryujinx.Graphics.Gpu/Shader/ShaderCache.cs
@@ -40,7 +40,7 @@ namespace Ryujinx.Graphics.Gpu.Shader
///
/// Version of the codegen (to be changed when codegen or guest format change).
///
- private const ulong ShaderCodeGenVersion = 3132;
+ private const ulong ShaderCodeGenVersion = 3054;
// Progress reporting helpers
private volatile int _shaderCount;
diff --git a/Ryujinx.Graphics.Shader/Decoders/Decoder.cs b/Ryujinx.Graphics.Shader/Decoders/Decoder.cs
index 8820527f..6fa4055a 100644
--- a/Ryujinx.Graphics.Shader/Decoders/Decoder.cs
+++ b/Ryujinx.Graphics.Shader/Decoders/Decoder.cs
@@ -308,7 +308,8 @@ namespace Ryujinx.Graphics.Shader.Decoders
int attr = offset + elemIndex * 4;
if (attr >= AttributeConsts.UserAttributeBase && attr < AttributeConsts.UserAttributeEnd)
{
- int index = (attr - AttributeConsts.UserAttributeBase) / 16;
+ int userAttr = attr - AttributeConsts.UserAttributeBase;
+ int index = userAttr / 16;
if (isStore)
{
@@ -316,7 +317,7 @@ namespace Ryujinx.Graphics.Shader.Decoders
}
else
{
- config.SetInputUserAttribute(index, perPatch);
+ config.SetInputUserAttribute(index, (userAttr >> 2) & 3, perPatch);
}
}
diff --git a/Ryujinx.Graphics.Shader/Translation/ShaderConfig.cs b/Ryujinx.Graphics.Shader/Translation/ShaderConfig.cs
index 3b10ab21..8be7ceae 100644
--- a/Ryujinx.Graphics.Shader/Translation/ShaderConfig.cs
+++ b/Ryujinx.Graphics.Shader/Translation/ShaderConfig.cs
@@ -54,6 +54,11 @@ namespace Ryujinx.Graphics.Shader.Translation
private int _nextUsedInputAttributes;
private int _thisUsedInputAttributes;
+ public UInt128 NextInputAttributesComponents { get; private set; }
+ public UInt128 ThisInputAttributesComponents { get; private set; }
+ public UInt128 NextInputAttributesPerPatchComponents { get; private set; }
+ public UInt128 ThisInputAttributesPerPatchComponents { get; private set; }
+
private int _usedConstantBuffers;
private int _usedStorageBuffers;
private int _usedStorageBuffersWrite;
@@ -227,11 +232,12 @@ namespace Ryujinx.Graphics.Shader.Translation
UsedOutputAttributes |= 1 << index;
}
- public void SetInputUserAttribute(int index, bool perPatch)
+ public void SetInputUserAttribute(int index, int component, bool perPatch)
{
if (perPatch)
{
UsedInputAttributesPerPatch |= 1 << index;
+ ThisInputAttributesPerPatchComponents |= UInt128.Pow2(index * 4 + component);
}
else
{
@@ -239,6 +245,7 @@ namespace Ryujinx.Graphics.Shader.Translation
UsedInputAttributes |= mask;
_thisUsedInputAttributes |= mask;
+ ThisInputAttributesComponents |= UInt128.Pow2(index * 4 + component);
}
}
@@ -256,6 +263,8 @@ namespace Ryujinx.Graphics.Shader.Translation
public void MergeFromtNextStage(ShaderConfig config)
{
+ NextInputAttributesComponents = config.ThisInputAttributesComponents;
+ NextInputAttributesPerPatchComponents = config.ThisInputAttributesPerPatchComponents;
NextUsesFixedFuncAttributes = config.UsedFeatures.HasFlag(FeatureFlags.FixedFuncAttr);
MergeOutputUserAttributes(config.UsedInputAttributes, config.UsedInputAttributesPerPatch);
}
diff --git a/Ryujinx.Graphics.Shader/Translation/Translator.cs b/Ryujinx.Graphics.Shader/Translation/Translator.cs
index 603b20d6..e594c818 100644
--- a/Ryujinx.Graphics.Shader/Translation/Translator.cs
+++ b/Ryujinx.Graphics.Shader/Translation/Translator.cs
@@ -214,24 +214,24 @@ namespace Ryujinx.Graphics.Shader.Translation
InitializeOutput(context, AttributeConsts.PositionX, perPatch: false);
}
- int usedAttributes = context.Config.UsedOutputAttributes;
- while (usedAttributes != 0)
+ UInt128 usedAttributes = context.Config.NextInputAttributesComponents;
+ while (usedAttributes != UInt128.Zero)
{
- int index = BitOperations.TrailingZeroCount(usedAttributes);
+ int index = usedAttributes.TrailingZeroCount();
- InitializeOutput(context, AttributeConsts.UserAttributeBase + index * 16, perPatch: false);
+ InitializeOutputComponent(context, AttributeConsts.UserAttributeBase + index * 4, perPatch: false);
- usedAttributes &= ~(1 << index);
+ usedAttributes &= ~UInt128.Pow2(index);
}
- int usedAttributesPerPatch = context.Config.UsedOutputAttributesPerPatch;
- while (usedAttributesPerPatch != 0)
+ UInt128 usedAttributesPerPatch = context.Config.NextInputAttributesPerPatchComponents;
+ while (usedAttributesPerPatch != UInt128.Zero)
{
- int index = BitOperations.TrailingZeroCount(usedAttributesPerPatch);
+ int index = usedAttributesPerPatch.TrailingZeroCount();
- InitializeOutput(context, AttributeConsts.UserAttributeBase + index * 16, perPatch: true);
+ InitializeOutputComponent(context, AttributeConsts.UserAttributeBase + index * 4, perPatch: true);
- usedAttributesPerPatch &= ~(1 << index);
+ usedAttributesPerPatch &= ~UInt128.Pow2(index);
}
if (config.NextUsesFixedFuncAttributes)
@@ -260,6 +260,12 @@ namespace Ryujinx.Graphics.Shader.Translation
}
}
+ private static void InitializeOutputComponent(EmitterContext context, int attrOffset, bool perPatch)
+ {
+ int c = (attrOffset >> 2) & 3;
+ context.Copy(perPatch ? AttributePerPatch(attrOffset) : Attribute(attrOffset), ConstF(c == 3 ? 1f : 0f));
+ }
+
private static void EmitOps(EmitterContext context, Block block)
{
for (int opIndex = 0; opIndex < block.OpCodes.Count; opIndex++)
diff --git a/Ryujinx.Graphics.Shader/Translation/UInt128.cs b/Ryujinx.Graphics.Shader/Translation/UInt128.cs
new file mode 100644
index 00000000..590f7690
--- /dev/null
+++ b/Ryujinx.Graphics.Shader/Translation/UInt128.cs
@@ -0,0 +1,74 @@
+using System;
+using System.Numerics;
+
+namespace Ryujinx.Graphics.Shader.Translation
+{
+ struct UInt128 : IEquatable
+ {
+ public static UInt128 Zero => new UInt128() { _v0 = 0, _v1 = 0 };
+
+ private ulong _v0;
+ private ulong _v1;
+
+ public int TrailingZeroCount()
+ {
+ int count = BitOperations.TrailingZeroCount(_v0);
+ if (count == 64)
+ {
+ count += BitOperations.TrailingZeroCount(_v1);
+ }
+
+ return count;
+ }
+
+ public static UInt128 Pow2(int x)
+ {
+ if (x >= 64)
+ {
+ return new UInt128() { _v0 = 0, _v1 = 1UL << (x - 64 ) };
+ }
+
+ return new UInt128() { _v0 = 1UL << x, _v1 = 0 };
+ }
+
+ public static UInt128 operator ~(UInt128 x)
+ {
+ return new UInt128() { _v0 = ~x._v0, _v1 = ~x._v1 };
+ }
+
+ public static UInt128 operator &(UInt128 x, UInt128 y)
+ {
+ return new UInt128() { _v0 = x._v0 & y._v0, _v1 = x._v1 & y._v1 };
+ }
+
+ public static UInt128 operator |(UInt128 x, UInt128 y)
+ {
+ return new UInt128() { _v0 = x._v0 | y._v0, _v1 = x._v1 | y._v1 };
+ }
+
+ public static bool operator ==(UInt128 x, UInt128 y)
+ {
+ return x.Equals(y);
+ }
+
+ public static bool operator !=(UInt128 x, UInt128 y)
+ {
+ return !x.Equals(y);
+ }
+
+ public override bool Equals(object obj)
+ {
+ return obj is UInt128 other && Equals(other);
+ }
+
+ public bool Equals(UInt128 other)
+ {
+ return _v0 == other._v0 && _v1 == other._v1;
+ }
+
+ public override int GetHashCode()
+ {
+ return HashCode.Combine(_v0, _v1);
+ }
+ }
+}
\ No newline at end of file