Replace BGRA and scale uniforms with a uniform block (#2496)
* Replace BGRA and scale uniforms with a uniform block * Setting the data again on program change is no longer needed * Optimize and resolve some warnings * Avoid redundant support buffer updates * Some optimizations to BindBuffers (now inlined) * Unify render scale arrays
This commit is contained in:
		
							parent
							
								
									b5b7e23fc4
								
							
						
					
					
						commit
						0f6ec446ea
					
				
					 13 changed files with 269 additions and 226 deletions
				
			
		|  | @ -1,9 +1,7 @@ | |||
| using Ryujinx.Common; | ||||
| using Ryujinx.Graphics.Shader.IntermediateRepresentation; | ||||
| using Ryujinx.Graphics.Shader.StructuredIr; | ||||
| using Ryujinx.Graphics.Shader.Translation; | ||||
| using System; | ||||
| using System.Collections.Generic; | ||||
| using System.Linq; | ||||
| 
 | ||||
| namespace Ryujinx.Graphics.Shader.CodeGen.Glsl | ||||
|  | @ -159,23 +157,38 @@ namespace Ryujinx.Graphics.Shader.CodeGen.Glsl | |||
|                 context.AppendLine(); | ||||
|             } | ||||
| 
 | ||||
|             if (context.Config.Stage == ShaderStage.Fragment || context.Config.Stage == ShaderStage.Compute) | ||||
|             { | ||||
|                 if (context.Config.Stage == ShaderStage.Fragment) | ||||
|                 { | ||||
|                     if (context.Config.GpuAccessor.QueryEarlyZForce()) | ||||
|                     { | ||||
|                         context.AppendLine("layout(early_fragment_tests) in;"); | ||||
|                         context.AppendLine(); | ||||
|                     } | ||||
|             bool isFragment = context.Config.Stage == ShaderStage.Fragment; | ||||
| 
 | ||||
|                     context.AppendLine($"uniform bool {DefaultNames.IsBgraName}[8];"); | ||||
|             if (isFragment || context.Config.Stage == ShaderStage.Compute) | ||||
|             { | ||||
|                 if (isFragment && context.Config.GpuAccessor.QueryEarlyZForce()) | ||||
|                 { | ||||
|                     context.AppendLine("layout(early_fragment_tests) in;"); | ||||
|                     context.AppendLine(); | ||||
|                 } | ||||
| 
 | ||||
|                 if (DeclareRenderScale(context)) | ||||
|                 if ((context.Config.UsedFeatures & (FeatureFlags.FragCoordXY | FeatureFlags.IntegerSampling)) != 0) | ||||
|                 { | ||||
|                     context.AppendLine(); | ||||
|                     string stage = OperandManager.GetShaderStagePrefix(context.Config.Stage); | ||||
| 
 | ||||
|                     int scaleElements = context.Config.GetTextureDescriptors().Length + context.Config.GetImageDescriptors().Length; | ||||
| 
 | ||||
|                     if (isFragment) | ||||
|                     { | ||||
|                         scaleElements++; // Also includes render target scale, for gl_FragCoord. | ||||
|                     } | ||||
| 
 | ||||
|                     DeclareSupportUniformBlock(context, isFragment, scaleElements); | ||||
| 
 | ||||
|                     if (context.Config.UsedFeatures.HasFlag(FeatureFlags.IntegerSampling)) | ||||
|                     { | ||||
|                         AppendHelperFunction(context, $"Ryujinx.Graphics.Shader/CodeGen/Glsl/HelperFunctions/TexelFetchScale_{stage}.glsl"); | ||||
|                         context.AppendLine(); | ||||
|                     } | ||||
|                 } | ||||
|                 else if (isFragment) | ||||
|                 { | ||||
|                     DeclareSupportUniformBlock(context, true, 0); | ||||
|                 } | ||||
|             } | ||||
| 
 | ||||
|  | @ -498,31 +511,33 @@ namespace Ryujinx.Graphics.Shader.CodeGen.Glsl | |||
|             } | ||||
|         } | ||||
| 
 | ||||
|         private static bool DeclareRenderScale(CodeGenContext context) | ||||
|         private static void DeclareSupportUniformBlock(CodeGenContext context, bool isFragment, int scaleElements) | ||||
|         { | ||||
|             if ((context.Config.UsedFeatures & (FeatureFlags.FragCoordXY | FeatureFlags.IntegerSampling)) != 0) | ||||
|             if (!isFragment && scaleElements == 0) | ||||
|             { | ||||
|                 string stage = OperandManager.GetShaderStagePrefix(context.Config.Stage); | ||||
| 
 | ||||
|                 int scaleElements = context.Config.GetTextureDescriptors().Length + context.Config.GetImageDescriptors().Length; | ||||
| 
 | ||||
|                 if (context.Config.Stage == ShaderStage.Fragment) | ||||
|                 { | ||||
|                     scaleElements++; // Also includes render target scale, for gl_FragCoord. | ||||
|                 } | ||||
| 
 | ||||
|                 context.AppendLine($"uniform float {stage}_renderScale[{scaleElements}];"); | ||||
| 
 | ||||
|                 if (context.Config.UsedFeatures.HasFlag(FeatureFlags.IntegerSampling)) | ||||
|                 { | ||||
|                     context.AppendLine(); | ||||
|                     AppendHelperFunction(context, $"Ryujinx.Graphics.Shader/CodeGen/Glsl/HelperFunctions/TexelFetchScale_{stage}.glsl"); | ||||
|                 } | ||||
| 
 | ||||
|                 return true; | ||||
|                 return; | ||||
|             } | ||||
| 
 | ||||
|             return false; | ||||
|             context.AppendLine($"layout (binding = 0, std140) uniform {DefaultNames.SupportBlockName}"); | ||||
|             context.EnterScope(); | ||||
| 
 | ||||
|             if (isFragment) | ||||
|             { | ||||
|                 context.AppendLine($"uint {DefaultNames.SupportBlockAlphaTestName};"); | ||||
|                 context.AppendLine($"bool {DefaultNames.SupportBlockIsBgraName}[{SupportBuffer.FragmentIsBgraCount}];"); | ||||
|             } | ||||
|             else | ||||
|             { | ||||
|                 context.AppendLine($"uint s_reserved[{SupportBuffer.ComputeRenderScaleOffset / SupportBuffer.FieldSize}];"); | ||||
|             } | ||||
| 
 | ||||
|             if (scaleElements != 0) | ||||
|             { | ||||
|                 context.AppendLine($"float {DefaultNames.SupportBlockRenderScaleName}[{scaleElements}];"); | ||||
|             } | ||||
| 
 | ||||
|             context.LeaveScope(";"); | ||||
|             context.AppendLine(); | ||||
|         } | ||||
| 
 | ||||
|         private static void AppendHelperFunction(CodeGenContext context, string filename) | ||||
|  |  | |||
|  | @ -14,6 +14,11 @@ namespace Ryujinx.Graphics.Shader.CodeGen.Glsl | |||
| 
 | ||||
|         public const string DataName = "data"; | ||||
| 
 | ||||
|         public const string SupportBlockName = "support_block"; | ||||
|         public const string SupportBlockAlphaTestName = "s_alpha_test"; | ||||
|         public const string SupportBlockIsBgraName = "s_is_bgra"; | ||||
|         public const string SupportBlockRenderScaleName = "s_render_scale"; | ||||
| 
 | ||||
|         public const string BlockSuffix = "block"; | ||||
| 
 | ||||
|         public const string UniformNamePrefix = "c"; | ||||
|  | @ -25,7 +30,5 @@ namespace Ryujinx.Graphics.Shader.CodeGen.Glsl | |||
|         public const string ArgumentNamePrefix = "a"; | ||||
| 
 | ||||
|         public const string UndefinedName = "undef"; | ||||
| 
 | ||||
|         public const string IsBgraName = "is_bgra"; | ||||
|     } | ||||
| } | ||||
|  | @ -1,6 +1,6 @@ | |||
| ivec2 Helper_TexelFetchScale(ivec2 inputVec, int samplerIndex) | ||||
| { | ||||
|     float scale = cp_renderScale[samplerIndex]; | ||||
|     float scale = s_render_scale[samplerIndex]; | ||||
|     if (scale == 1.0) | ||||
|     { | ||||
|         return inputVec; | ||||
|  | @ -10,7 +10,7 @@ | |||
| 
 | ||||
| int Helper_TextureSizeUnscale(int size, int samplerIndex) | ||||
| { | ||||
|     float scale = cp_renderScale[samplerIndex]; | ||||
|     float scale = s_render_scale[samplerIndex]; | ||||
|     if (scale == 1.0) | ||||
|     { | ||||
|         return size; | ||||
|  |  | |||
|  | @ -1,6 +1,6 @@ | |||
| ivec2 Helper_TexelFetchScale(ivec2 inputVec, int samplerIndex) | ||||
| { | ||||
|     float scale = fp_renderScale[1 + samplerIndex]; | ||||
|     float scale = s_render_scale[1 + samplerIndex]; | ||||
|     if (scale == 1.0) | ||||
|     { | ||||
|         return inputVec; | ||||
|  | @ -17,7 +17,7 @@ | |||
| 
 | ||||
| int Helper_TextureSizeUnscale(int size, int samplerIndex) | ||||
| { | ||||
|     float scale = abs(fp_renderScale[1 + samplerIndex]); | ||||
|     float scale = abs(s_render_scale[1 + samplerIndex]); | ||||
|     if (scale == 1.0) | ||||
|     { | ||||
|         return size; | ||||
|  |  | |||
|  | @ -68,14 +68,14 @@ namespace Ryujinx.Graphics.Shader.CodeGen.Glsl | |||
|             { AttributeConsts.LtMask,              new BuiltInAttribute("unpackUint2x32(gl_SubGroupLtMaskARB).x", VariableType.U32)  }, | ||||
| 
 | ||||
|             // Support uniforms. | ||||
|             { AttributeConsts.FragmentOutputIsBgraBase + 0,  new BuiltInAttribute($"{DefaultNames.IsBgraName}[0]",  VariableType.Bool) }, | ||||
|             { AttributeConsts.FragmentOutputIsBgraBase + 4,  new BuiltInAttribute($"{DefaultNames.IsBgraName}[1]",  VariableType.Bool) }, | ||||
|             { AttributeConsts.FragmentOutputIsBgraBase + 8,  new BuiltInAttribute($"{DefaultNames.IsBgraName}[2]",  VariableType.Bool) }, | ||||
|             { AttributeConsts.FragmentOutputIsBgraBase + 12, new BuiltInAttribute($"{DefaultNames.IsBgraName}[3]",  VariableType.Bool) }, | ||||
|             { AttributeConsts.FragmentOutputIsBgraBase + 16, new BuiltInAttribute($"{DefaultNames.IsBgraName}[4]",  VariableType.Bool) }, | ||||
|             { AttributeConsts.FragmentOutputIsBgraBase + 20, new BuiltInAttribute($"{DefaultNames.IsBgraName}[5]",  VariableType.Bool) }, | ||||
|             { AttributeConsts.FragmentOutputIsBgraBase + 24, new BuiltInAttribute($"{DefaultNames.IsBgraName}[6]",  VariableType.Bool) }, | ||||
|             { AttributeConsts.FragmentOutputIsBgraBase + 28, new BuiltInAttribute($"{DefaultNames.IsBgraName}[7]",  VariableType.Bool) } | ||||
|             { AttributeConsts.FragmentOutputIsBgraBase + 0,  new BuiltInAttribute($"{DefaultNames.SupportBlockIsBgraName}[0]",  VariableType.Bool) }, | ||||
|             { AttributeConsts.FragmentOutputIsBgraBase + 4,  new BuiltInAttribute($"{DefaultNames.SupportBlockIsBgraName}[1]",  VariableType.Bool) }, | ||||
|             { AttributeConsts.FragmentOutputIsBgraBase + 8,  new BuiltInAttribute($"{DefaultNames.SupportBlockIsBgraName}[2]",  VariableType.Bool) }, | ||||
|             { AttributeConsts.FragmentOutputIsBgraBase + 12, new BuiltInAttribute($"{DefaultNames.SupportBlockIsBgraName}[3]",  VariableType.Bool) }, | ||||
|             { AttributeConsts.FragmentOutputIsBgraBase + 16, new BuiltInAttribute($"{DefaultNames.SupportBlockIsBgraName}[4]",  VariableType.Bool) }, | ||||
|             { AttributeConsts.FragmentOutputIsBgraBase + 20, new BuiltInAttribute($"{DefaultNames.SupportBlockIsBgraName}[5]",  VariableType.Bool) }, | ||||
|             { AttributeConsts.FragmentOutputIsBgraBase + 24, new BuiltInAttribute($"{DefaultNames.SupportBlockIsBgraName}[6]",  VariableType.Bool) }, | ||||
|             { AttributeConsts.FragmentOutputIsBgraBase + 28, new BuiltInAttribute($"{DefaultNames.SupportBlockIsBgraName}[7]",  VariableType.Bool) } | ||||
|         }; | ||||
| 
 | ||||
|         private Dictionary<AstOperand, string> _locals; | ||||
|  | @ -194,8 +194,8 @@ namespace Ryujinx.Graphics.Shader.CodeGen.Glsl | |||
|                     { | ||||
|                         switch (value & ~3) | ||||
|                         { | ||||
|                             case AttributeConsts.PositionX: return "(gl_FragCoord.x / fp_renderScale[0])"; | ||||
|                             case AttributeConsts.PositionY: return "(gl_FragCoord.y / fp_renderScale[0])"; | ||||
|                             case AttributeConsts.PositionX: return $"(gl_FragCoord.x / {DefaultNames.SupportBlockRenderScaleName}[0])"; | ||||
|                             case AttributeConsts.PositionY: return $"(gl_FragCoord.y / {DefaultNames.SupportBlockRenderScaleName}[0])"; | ||||
|                             case AttributeConsts.PositionZ: return "gl_FragCoord.z"; | ||||
|                             case AttributeConsts.PositionW: return "gl_FragCoord.w"; | ||||
|                         } | ||||
|  |  | |||
							
								
								
									
										18
									
								
								Ryujinx.Graphics.Shader/SupportBuffer.cs
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								Ryujinx.Graphics.Shader/SupportBuffer.cs
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,18 @@ | |||
| namespace Ryujinx.Graphics.Shader | ||||
| { | ||||
|     public static class SupportBuffer | ||||
|     { | ||||
|         public const int FieldSize = 16; // Each field takes 16 bytes on default layout, even bool. | ||||
| 
 | ||||
|         public const int FragmentAlphaTestOffset = 0; | ||||
|         public const int FragmentIsBgraOffset = FieldSize; | ||||
|         public const int FragmentIsBgraCount = 8; | ||||
|         public const int FragmentRenderScaleOffset = FragmentIsBgraOffset + FragmentIsBgraCount * FieldSize; | ||||
|         public const int ComputeRenderScaleOffset = FragmentRenderScaleOffset + FieldSize; // Skip first scale that is used for the render target | ||||
| 
 | ||||
|         // One for the render target, 32 for the textures, and 8 for the images. | ||||
|         private const int RenderScaleMaxCount = 1 + 32 + 8; | ||||
| 
 | ||||
|         public const int RequiredSize = FragmentRenderScaleOffset + RenderScaleMaxCount * FieldSize; | ||||
|     } | ||||
| } | ||||
|  | @ -7,6 +7,12 @@ namespace Ryujinx.Graphics.Shader.Translation | |||
|         public int TexturesCount { get; private set; } | ||||
|         public int ImagesCount { get; private set; } | ||||
| 
 | ||||
|         public TranslationCounts() | ||||
|         { | ||||
|             // The first binding is reserved for the support buffer. | ||||
|             UniformBuffersCount = 1; | ||||
|         } | ||||
| 
 | ||||
|         internal int IncrementUniformBuffersCount() | ||||
|         { | ||||
|             return UniformBuffersCount++; | ||||
|  |  | |||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 gdkchan
						gdkchan