 f0b00c1ae9
			
		
	
	
		f0b00c1ae9
		
			
		
	
	
	
	
		
			
			* Fix TXQ for 3D textures. Assumes the texture is 3D if the component mask contains Z. This fixes a bug in UE4 games where parts of the map had garbage pointers to lighting voxels, as the lookup 3D texture was not being initialized. Most notable game is THPS1+2. May need another PR to keep image store data alive and properly flush it in order using the AutoDeleteCache. * Get sampler type for TextureSize from bound textures.
		
			
				
	
	
		
			1772 lines
		
	
	
		
			No EOL
		
	
	
		
			54 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			1772 lines
		
	
	
		
			No EOL
		
	
	
		
			54 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| using Ryujinx.Graphics.Shader.Decoders;
 | |
| using Ryujinx.Graphics.Shader.IntermediateRepresentation;
 | |
| using Ryujinx.Graphics.Shader.Translation;
 | |
| using System;
 | |
| using System.Collections.Generic;
 | |
| 
 | |
| using static Ryujinx.Graphics.Shader.Instructions.InstEmitHelper;
 | |
| using static Ryujinx.Graphics.Shader.IntermediateRepresentation.OperandHelper;
 | |
| 
 | |
| namespace Ryujinx.Graphics.Shader.Instructions
 | |
| {
 | |
|     static partial class InstEmit
 | |
|     {
 | |
|         private const bool Sample1DAs2D = true;
 | |
| 
 | |
|         public static void Suld(EmitterContext context)
 | |
|         {
 | |
|             context.Config.SetUsedFeature(FeatureFlags.IntegerSampling);
 | |
| 
 | |
|             OpCodeImage op = (OpCodeImage)context.CurrOp;
 | |
| 
 | |
|             SamplerType type = ConvertSamplerType(op.Dimensions);
 | |
| 
 | |
|             if (type == SamplerType.None)
 | |
|             {
 | |
|                 context.Config.GpuAccessor.Log("Invalid image store sampler type.");
 | |
| 
 | |
|                 return;
 | |
|             }
 | |
| 
 | |
|             // Rb is Rd on the SULD instruction.
 | |
|             int rdIndex = op.Rb.Index;
 | |
|             int raIndex = op.Ra.Index;
 | |
| 
 | |
|             Operand Ra()
 | |
|             {
 | |
|                 if (raIndex > RegisterConsts.RegisterZeroIndex)
 | |
|                 {
 | |
|                     return Const(0);
 | |
|                 }
 | |
| 
 | |
|                 return context.Copy(Register(raIndex++, RegisterType.Gpr));
 | |
|             }
 | |
| 
 | |
|             List<Operand> sourcesList = new List<Operand>();
 | |
| 
 | |
|             if (op.IsBindless)
 | |
|             {
 | |
|                 sourcesList.Add(context.Copy(Register(op.Rc)));
 | |
|             }
 | |
| 
 | |
|             int coordsCount = type.GetDimensions();
 | |
| 
 | |
|             for (int index = 0; index < coordsCount; index++)
 | |
|             {
 | |
|                 sourcesList.Add(Ra());
 | |
|             }
 | |
| 
 | |
|             if (Sample1DAs2D && (type & SamplerType.Mask) == SamplerType.Texture1D)
 | |
|             {
 | |
|                 sourcesList.Add(Const(0));
 | |
| 
 | |
|                 type &= ~SamplerType.Mask;
 | |
|                 type |= SamplerType.Texture2D;
 | |
|             }
 | |
| 
 | |
|             if (type.HasFlag(SamplerType.Array))
 | |
|             {
 | |
|                 sourcesList.Add(Ra());
 | |
| 
 | |
|                 type |= SamplerType.Array;
 | |
|             }
 | |
| 
 | |
|             Operand[] sources = sourcesList.ToArray();
 | |
| 
 | |
|             int handle = !op.IsBindless ? op.HandleOffset : 0;
 | |
| 
 | |
|             TextureFlags flags = op.IsBindless ? TextureFlags.Bindless : TextureFlags.None;
 | |
| 
 | |
|             if (op.UseComponents)
 | |
|             {
 | |
|                 int componentMask = (int)op.Components;
 | |
| 
 | |
|                 for (int compMask = componentMask, compIndex = 0; compMask != 0; compMask >>= 1, compIndex++)
 | |
|                 {
 | |
|                     if ((compMask & 1) == 0)
 | |
|                     {
 | |
|                         continue;
 | |
|                     }
 | |
| 
 | |
|                     if (rdIndex == RegisterConsts.RegisterZeroIndex)
 | |
|                     {
 | |
|                         break;
 | |
|                     }
 | |
| 
 | |
|                     Operand rd = Register(rdIndex++, RegisterType.Gpr);
 | |
| 
 | |
|                     TextureOperation operation = context.CreateTextureOperation(
 | |
|                         Instruction.ImageLoad,
 | |
|                         type,
 | |
|                         flags,
 | |
|                         handle,
 | |
|                         compIndex,
 | |
|                         rd,
 | |
|                         sources);
 | |
| 
 | |
|                     if (!op.IsBindless)
 | |
|                     {
 | |
|                         operation.Format = context.Config.GetTextureFormat(handle);
 | |
|                     }
 | |
| 
 | |
|                     context.Add(operation);
 | |
|                 }
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 if (op.ByteAddress)
 | |
|                 {
 | |
|                     int xIndex = op.IsBindless ? 1 : 0;
 | |
| 
 | |
|                     sources[xIndex] = context.ShiftRightS32(sources[xIndex], Const(GetComponentSizeInBytesLog2(op.Size)));
 | |
|                 }
 | |
| 
 | |
|                 int components = GetComponents(op.Size);
 | |
| 
 | |
|                 for (int compIndex = 0; compIndex < components; compIndex++)
 | |
|                 {
 | |
|                     if (rdIndex == RegisterConsts.RegisterZeroIndex)
 | |
|                     {
 | |
|                         break;
 | |
|                     }
 | |
| 
 | |
|                     Operand rd = Register(rdIndex++, RegisterType.Gpr);
 | |
| 
 | |
|                     TextureOperation operation = context.CreateTextureOperation(
 | |
|                         Instruction.ImageLoad,
 | |
|                         type,
 | |
|                         GetTextureFormat(op.Size),
 | |
|                         flags,
 | |
|                         handle,
 | |
|                         compIndex,
 | |
|                         rd,
 | |
|                         sources);
 | |
| 
 | |
|                     context.Add(operation);
 | |
| 
 | |
|                     switch (op.Size)
 | |
|                     {
 | |
|                         case IntegerSize.U8:  context.Copy(rd, ZeroExtendTo32(context, rd, 8));  break;
 | |
|                         case IntegerSize.U16: context.Copy(rd, ZeroExtendTo32(context, rd, 16)); break;
 | |
|                         case IntegerSize.S8:  context.Copy(rd, SignExtendTo32(context, rd, 8));  break;
 | |
|                         case IntegerSize.S16: context.Copy(rd, SignExtendTo32(context, rd, 16)); break;
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public static void Sust(EmitterContext context)
 | |
|         {
 | |
|             OpCodeImage op = (OpCodeImage)context.CurrOp;
 | |
| 
 | |
|             SamplerType type = ConvertSamplerType(op.Dimensions);
 | |
| 
 | |
|             if (type == SamplerType.None)
 | |
|             {
 | |
|                 context.Config.GpuAccessor.Log("Invalid image store sampler type.");
 | |
| 
 | |
|                 return;
 | |
|             }
 | |
| 
 | |
|             int raIndex = op.Ra.Index;
 | |
|             int rbIndex = op.Rb.Index;
 | |
| 
 | |
|             Operand Ra()
 | |
|             {
 | |
|                 if (raIndex > RegisterConsts.RegisterZeroIndex)
 | |
|                 {
 | |
|                     return Const(0);
 | |
|                 }
 | |
| 
 | |
|                 return context.Copy(Register(raIndex++, RegisterType.Gpr));
 | |
|             }
 | |
| 
 | |
|             Operand Rb()
 | |
|             {
 | |
|                 if (rbIndex > RegisterConsts.RegisterZeroIndex)
 | |
|                 {
 | |
|                     return Const(0);
 | |
|                 }
 | |
| 
 | |
|                 return context.Copy(Register(rbIndex++, RegisterType.Gpr));
 | |
|             }
 | |
| 
 | |
|             List<Operand> sourcesList = new List<Operand>();
 | |
| 
 | |
|             if (op.IsBindless)
 | |
|             {
 | |
|                 sourcesList.Add(context.Copy(Register(op.Rc)));
 | |
|             }
 | |
| 
 | |
|             int coordsCount = type.GetDimensions();
 | |
| 
 | |
|             for (int index = 0; index < coordsCount; index++)
 | |
|             {
 | |
|                 sourcesList.Add(Ra());
 | |
|             }
 | |
| 
 | |
|             if (Sample1DAs2D && (type & SamplerType.Mask) == SamplerType.Texture1D)
 | |
|             {
 | |
|                 sourcesList.Add(Const(0));
 | |
| 
 | |
|                 type &= ~SamplerType.Mask;
 | |
|                 type |= SamplerType.Texture2D;
 | |
|             }
 | |
| 
 | |
|             if (type.HasFlag(SamplerType.Array))
 | |
|             {
 | |
|                 sourcesList.Add(Ra());
 | |
| 
 | |
|                 type |= SamplerType.Array;
 | |
|             }
 | |
| 
 | |
|             TextureFormat format = TextureFormat.Unknown;
 | |
| 
 | |
|             if (op.UseComponents)
 | |
|             {
 | |
|                 int componentMask = (int)op.Components;
 | |
| 
 | |
|                 for (int compMask = componentMask, compIndex = 0; compMask != 0; compMask >>= 1, compIndex++)
 | |
|                 {
 | |
|                     if ((compMask & 1) != 0)
 | |
|                     {
 | |
|                         sourcesList.Add(Rb());
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|                 if (!op.IsBindless)
 | |
|                 {
 | |
|                     format = context.Config.GetTextureFormat(op.HandleOffset);
 | |
|                 }
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 if (op.ByteAddress)
 | |
|                 {
 | |
|                     int xIndex = op.IsBindless ? 1 : 0;
 | |
| 
 | |
|                     sourcesList[xIndex] = context.ShiftRightS32(sourcesList[xIndex], Const(GetComponentSizeInBytesLog2(op.Size)));
 | |
|                 }
 | |
| 
 | |
|                 int components = GetComponents(op.Size);
 | |
| 
 | |
|                 for (int compIndex = 0; compIndex < components; compIndex++)
 | |
|                 {
 | |
|                     sourcesList.Add(Rb());
 | |
|                 }
 | |
| 
 | |
|                 format = GetTextureFormat(op.Size);
 | |
|             }
 | |
| 
 | |
|             Operand[] sources = sourcesList.ToArray();
 | |
| 
 | |
|             int handle = !op.IsBindless ? op.HandleOffset : 0;
 | |
| 
 | |
|             TextureFlags flags = op.IsBindless ? TextureFlags.Bindless : TextureFlags.None;
 | |
| 
 | |
|             TextureOperation operation = context.CreateTextureOperation(
 | |
|                 Instruction.ImageStore,
 | |
|                 type,
 | |
|                 format,
 | |
|                 flags,
 | |
|                 handle,
 | |
|                 0,
 | |
|                 null,
 | |
|                 sources);
 | |
| 
 | |
|             context.Add(operation);
 | |
|         }
 | |
| 
 | |
|         public static void Sured(EmitterContext context)
 | |
|         {
 | |
|             OpCodeSured op = (OpCodeSured)context.CurrOp;
 | |
| 
 | |
|             SamplerType type = ConvertSamplerType(op.Dimensions);
 | |
| 
 | |
|             if (type == SamplerType.None)
 | |
|             {
 | |
|                 context.Config.GpuAccessor.Log("Invalid image reduction sampler type.");
 | |
| 
 | |
|                 return;
 | |
|             }
 | |
| 
 | |
|             int raIndex = op.Ra.Index;
 | |
|             int rbIndex = op.Rb.Index;
 | |
| 
 | |
|             Operand Ra()
 | |
|             {
 | |
|                 if (raIndex > RegisterConsts.RegisterZeroIndex)
 | |
|                 {
 | |
|                     return Const(0);
 | |
|                 }
 | |
| 
 | |
|                 return context.Copy(Register(raIndex++, RegisterType.Gpr));
 | |
|             }
 | |
| 
 | |
|             Operand Rb()
 | |
|             {
 | |
|                 if (rbIndex > RegisterConsts.RegisterZeroIndex)
 | |
|                 {
 | |
|                     return Const(0);
 | |
|                 }
 | |
| 
 | |
|                 return context.Copy(Register(rbIndex++, RegisterType.Gpr));
 | |
|             }
 | |
| 
 | |
|             List<Operand> sourcesList = new List<Operand>();
 | |
| 
 | |
|             if (op.IsBindless)
 | |
|             {
 | |
|                 sourcesList.Add(context.Copy(Register(op.Rc)));
 | |
|             }
 | |
| 
 | |
|             int coordsCount = type.GetDimensions();
 | |
| 
 | |
|             for (int index = 0; index < coordsCount; index++)
 | |
|             {
 | |
|                 sourcesList.Add(Ra());
 | |
|             }
 | |
| 
 | |
|             if (Sample1DAs2D && (type & SamplerType.Mask) == SamplerType.Texture1D)
 | |
|             {
 | |
|                 sourcesList.Add(Const(0));
 | |
| 
 | |
|                 type &= ~SamplerType.Mask;
 | |
|                 type |= SamplerType.Texture2D;
 | |
|             }
 | |
| 
 | |
|             if (type.HasFlag(SamplerType.Array))
 | |
|             {
 | |
|                 sourcesList.Add(Ra());
 | |
| 
 | |
|                 type |= SamplerType.Array;
 | |
|             }
 | |
| 
 | |
|             TextureFormat format = TextureFormat.R32Sint;
 | |
| 
 | |
|             if (op.UseType)
 | |
|             {
 | |
|                 if (op.ByteAddress)
 | |
|                 {
 | |
|                     int xIndex = op.IsBindless ? 1 : 0;
 | |
| 
 | |
|                     sourcesList[xIndex] = context.ShiftRightS32(sourcesList[xIndex], Const(GetComponentSizeInBytesLog2(op.Type)));
 | |
|                 }
 | |
| 
 | |
|                 // TODO: FP and 64-bit formats.
 | |
|                 format = (op.Type == ReductionType.SD32 || op.Type == ReductionType.SD64) ?
 | |
|                     context.Config.GetTextureFormatAtomic(op.HandleOffset) :
 | |
|                     GetTextureFormat(op.Type);
 | |
|             }
 | |
|             else if (!op.IsBindless)
 | |
|             {
 | |
|                 format = context.Config.GetTextureFormatAtomic(op.HandleOffset);
 | |
|             }
 | |
| 
 | |
|             sourcesList.Add(Rb());
 | |
| 
 | |
|             Operand[] sources = sourcesList.ToArray();
 | |
| 
 | |
|             int handle = op.HandleOffset;
 | |
| 
 | |
|             TextureFlags flags = GetAtomicOpFlags(op.AtomicOp);
 | |
| 
 | |
|             if (op.IsBindless)
 | |
|             {
 | |
|                 handle = 0;
 | |
|                 flags |= TextureFlags.Bindless;
 | |
|             }
 | |
| 
 | |
|             TextureOperation operation = context.CreateTextureOperation(
 | |
|                 Instruction.ImageAtomic,
 | |
|                 type,
 | |
|                 format,
 | |
|                 flags,
 | |
|                 handle,
 | |
|                 0,
 | |
|                 null,
 | |
|                 sources);
 | |
| 
 | |
|             context.Add(operation);
 | |
|         }
 | |
| 
 | |
|         public static void Suatom(EmitterContext context)
 | |
|         {
 | |
|             OpCodeSuatom op = (OpCodeSuatom)context.CurrOp;
 | |
| 
 | |
|             SamplerType type = ConvertSamplerType(op.Dimensions);
 | |
| 
 | |
|             if (type == SamplerType.None)
 | |
|             {
 | |
|                 context.Config.GpuAccessor.Log("Invalid image atomic sampler type.");
 | |
| 
 | |
|                 return;
 | |
|             }
 | |
| 
 | |
|             int raIndex = op.Ra.Index;
 | |
|             int rbIndex = op.Rb.Index;
 | |
| 
 | |
|             Operand Ra()
 | |
|             {
 | |
|                 if (raIndex > RegisterConsts.RegisterZeroIndex)
 | |
|                 {
 | |
|                     return Const(0);
 | |
|                 }
 | |
| 
 | |
|                 return context.Copy(Register(raIndex++, RegisterType.Gpr));
 | |
|             }
 | |
| 
 | |
|             Operand Rb()
 | |
|             {
 | |
|                 if (rbIndex > RegisterConsts.RegisterZeroIndex)
 | |
|                 {
 | |
|                     return Const(0);
 | |
|                 }
 | |
| 
 | |
|                 return context.Copy(Register(rbIndex++, RegisterType.Gpr));
 | |
|             }
 | |
| 
 | |
|             int rdIndex = op.Rd.Index;
 | |
| 
 | |
|             Operand GetDest()
 | |
|             {
 | |
|                 if (rdIndex > RegisterConsts.RegisterZeroIndex)
 | |
|                 {
 | |
|                     return Const(0);
 | |
|                 }
 | |
| 
 | |
|                 return Register(rdIndex++, RegisterType.Gpr);
 | |
|             }
 | |
| 
 | |
|             List<Operand> sourcesList = new List<Operand>();
 | |
| 
 | |
|             if (op.IsBindless)
 | |
|             {
 | |
|                 sourcesList.Add(context.Copy(Register(op.Rc)));
 | |
|             }
 | |
| 
 | |
|             int coordsCount = type.GetDimensions();
 | |
| 
 | |
|             for (int index = 0; index < coordsCount; index++)
 | |
|             {
 | |
|                 sourcesList.Add(Ra());
 | |
|             }
 | |
| 
 | |
|             if (Sample1DAs2D && (type & SamplerType.Mask) == SamplerType.Texture1D)
 | |
|             {
 | |
|                 sourcesList.Add(Const(0));
 | |
| 
 | |
|                 type &= ~SamplerType.Mask;
 | |
|                 type |= SamplerType.Texture2D;
 | |
|             }
 | |
| 
 | |
|             if (type.HasFlag(SamplerType.Array))
 | |
|             {
 | |
|                 sourcesList.Add(Ra());
 | |
| 
 | |
|                 type |= SamplerType.Array;
 | |
|             }
 | |
| 
 | |
|             TextureFormat format = TextureFormat.R32Sint;
 | |
| 
 | |
|             if (op.UseType)
 | |
|             {
 | |
|                 if (op.ByteAddress)
 | |
|                 {
 | |
|                     int xIndex = op.IsBindless ? 1 : 0;
 | |
| 
 | |
|                     sourcesList[xIndex] = context.ShiftRightS32(sourcesList[xIndex], Const(GetComponentSizeInBytesLog2(op.Type)));
 | |
|                 }
 | |
| 
 | |
|                 // TODO: FP and 64-bit formats.
 | |
|                 format = (op.Type == ReductionType.SD32 || op.Type == ReductionType.SD64) ?
 | |
|                     context.Config.GetTextureFormatAtomic(op.HandleOffset) :
 | |
|                     GetTextureFormat(op.Type);
 | |
|             }
 | |
|             else if (!op.IsBindless)
 | |
|             {
 | |
|                 format = context.Config.GetTextureFormatAtomic(op.HandleOffset);
 | |
|             }
 | |
| 
 | |
|             if (op.CompareAndSwap)
 | |
|             {
 | |
|                 sourcesList.Add(Rb());
 | |
|             }
 | |
| 
 | |
|             sourcesList.Add(Rb());
 | |
| 
 | |
|             Operand[] sources = sourcesList.ToArray();
 | |
| 
 | |
|             int handle = op.HandleOffset;
 | |
| 
 | |
|             TextureFlags flags = op.CompareAndSwap ? TextureFlags.CAS : GetAtomicOpFlags(op.AtomicOp);
 | |
| 
 | |
|             if (op.IsBindless)
 | |
|             {
 | |
|                 handle = 0;
 | |
|                 flags |= TextureFlags.Bindless;
 | |
|             }
 | |
| 
 | |
|             TextureOperation operation = context.CreateTextureOperation(
 | |
|                 Instruction.ImageAtomic,
 | |
|                 type,
 | |
|                 format,
 | |
|                 flags,
 | |
|                 handle,
 | |
|                 0,
 | |
|                 GetDest(),
 | |
|                 sources);
 | |
| 
 | |
|             context.Add(operation);
 | |
|         }
 | |
| 
 | |
|         public static void Tex(EmitterContext context)
 | |
|         {
 | |
|             EmitTextureSample(context, TextureFlags.None);
 | |
|         }
 | |
| 
 | |
|         public static void TexB(EmitterContext context)
 | |
|         {
 | |
|             EmitTextureSample(context, TextureFlags.Bindless);
 | |
|         }
 | |
| 
 | |
|         public static void Tld(EmitterContext context)
 | |
|         {
 | |
|             context.Config.SetUsedFeature(FeatureFlags.IntegerSampling);
 | |
| 
 | |
|             EmitTextureSample(context, TextureFlags.IntCoords);
 | |
|         }
 | |
| 
 | |
|         public static void TldB(EmitterContext context)
 | |
|         {
 | |
|             context.Config.SetUsedFeature(FeatureFlags.IntegerSampling);
 | |
| 
 | |
|             EmitTextureSample(context, TextureFlags.IntCoords | TextureFlags.Bindless);
 | |
|         }
 | |
| 
 | |
|         public static void Texs(EmitterContext context)
 | |
|         {
 | |
|             OpCodeTextureScalar op = (OpCodeTextureScalar)context.CurrOp;
 | |
| 
 | |
|             if (op.Rd0.IsRZ && op.Rd1.IsRZ)
 | |
|             {
 | |
|                 return;
 | |
|             }
 | |
| 
 | |
|             List<Operand> sourcesList = new List<Operand>();
 | |
| 
 | |
|             int raIndex = op.Ra.Index;
 | |
|             int rbIndex = op.Rb.Index;
 | |
| 
 | |
|             Operand Ra()
 | |
|             {
 | |
|                 if (raIndex > RegisterConsts.RegisterZeroIndex)
 | |
|                 {
 | |
|                     return Const(0);
 | |
|                 }
 | |
| 
 | |
|                 return context.Copy(Register(raIndex++, RegisterType.Gpr));
 | |
|             }
 | |
| 
 | |
|             Operand Rb()
 | |
|             {
 | |
|                 if (rbIndex > RegisterConsts.RegisterZeroIndex)
 | |
|                 {
 | |
|                     return Const(0);
 | |
|                 }
 | |
| 
 | |
|                 return context.Copy(Register(rbIndex++, RegisterType.Gpr));
 | |
|             }
 | |
| 
 | |
|             void AddTextureOffset(int coordsCount, int stride, int size)
 | |
|             {
 | |
|                 Operand packedOffs = Rb();
 | |
| 
 | |
|                 for (int index = 0; index < coordsCount; index++)
 | |
|                 {
 | |
|                     sourcesList.Add(context.BitfieldExtractS32(packedOffs, Const(index * stride), Const(size)));
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             SamplerType  type;
 | |
|             TextureFlags flags;
 | |
| 
 | |
|             if (op is OpCodeTexs texsOp)
 | |
|             {
 | |
|                 type = ConvertSamplerType(texsOp.Target);
 | |
| 
 | |
|                 if (type == SamplerType.None)
 | |
|                 {
 | |
|                     context.Config.GpuAccessor.Log("Invalid texture sampler type.");
 | |
| 
 | |
|                     return;
 | |
|                 }
 | |
| 
 | |
|                 flags = ConvertTextureFlags(texsOp.Target);
 | |
| 
 | |
|                 // We don't need to handle 1D -> Buffer conversions here as
 | |
|                 // only texture sample with integer coordinates can ever use buffer targets.
 | |
| 
 | |
|                 if ((type & SamplerType.Array) != 0)
 | |
|                 {
 | |
|                     Operand arrayIndex = Ra();
 | |
| 
 | |
|                     sourcesList.Add(Ra());
 | |
|                     sourcesList.Add(Rb());
 | |
| 
 | |
|                     sourcesList.Add(arrayIndex);
 | |
| 
 | |
|                     if ((type & SamplerType.Shadow) != 0)
 | |
|                     {
 | |
|                         sourcesList.Add(Rb());
 | |
|                     }
 | |
| 
 | |
|                     if ((flags & TextureFlags.LodLevel) != 0)
 | |
|                     {
 | |
|                         sourcesList.Add(ConstF(0));
 | |
|                     }
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     switch (texsOp.Target)
 | |
|                     {
 | |
|                         case TextureTarget.Texture1DLodZero:
 | |
|                             sourcesList.Add(Ra());
 | |
| 
 | |
|                             if (Sample1DAs2D)
 | |
|                             {
 | |
|                                 sourcesList.Add(ConstF(0));
 | |
| 
 | |
|                                 type &= ~SamplerType.Mask;
 | |
|                                 type |= SamplerType.Texture2D;
 | |
|                             }
 | |
| 
 | |
|                             sourcesList.Add(ConstF(0));
 | |
|                             break;
 | |
| 
 | |
|                         case TextureTarget.Texture2D:
 | |
|                             sourcesList.Add(Ra());
 | |
|                             sourcesList.Add(Rb());
 | |
|                             break;
 | |
| 
 | |
|                         case TextureTarget.Texture2DLodZero:
 | |
|                             sourcesList.Add(Ra());
 | |
|                             sourcesList.Add(Rb());
 | |
|                             sourcesList.Add(ConstF(0));
 | |
|                             break;
 | |
| 
 | |
|                         case TextureTarget.Texture2DLodLevel:
 | |
|                         case TextureTarget.Texture2DDepthCompare:
 | |
|                         case TextureTarget.Texture3D:
 | |
|                         case TextureTarget.TextureCube:
 | |
|                             sourcesList.Add(Ra());
 | |
|                             sourcesList.Add(Ra());
 | |
|                             sourcesList.Add(Rb());
 | |
|                             break;
 | |
| 
 | |
|                         case TextureTarget.Texture2DLodZeroDepthCompare:
 | |
|                         case TextureTarget.Texture3DLodZero:
 | |
|                             sourcesList.Add(Ra());
 | |
|                             sourcesList.Add(Ra());
 | |
|                             sourcesList.Add(Rb());
 | |
|                             sourcesList.Add(ConstF(0));
 | |
|                             break;
 | |
| 
 | |
|                         case TextureTarget.Texture2DLodLevelDepthCompare:
 | |
|                         case TextureTarget.TextureCubeLodLevel:
 | |
|                             sourcesList.Add(Ra());
 | |
|                             sourcesList.Add(Ra());
 | |
|                             sourcesList.Add(Rb());
 | |
|                             sourcesList.Add(Rb());
 | |
|                             break;
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|             else if (op is OpCodeTlds tldsOp)
 | |
|             {
 | |
|                 type = ConvertSamplerType(tldsOp.Target);
 | |
| 
 | |
|                 if (type == SamplerType.None)
 | |
|                 {
 | |
|                     context.Config.GpuAccessor.Log("Invalid texel fetch sampler type.");
 | |
| 
 | |
|                     return;
 | |
|                 }
 | |
| 
 | |
|                 context.Config.SetUsedFeature(FeatureFlags.IntegerSampling);
 | |
| 
 | |
|                 flags = ConvertTextureFlags(tldsOp.Target) | TextureFlags.IntCoords;
 | |
| 
 | |
|                 if (tldsOp.Target == TexelLoadTarget.Texture1DLodZero && context.Config.GpuAccessor.QuerySamplerType(tldsOp.HandleOffset) == SamplerType.TextureBuffer)
 | |
|                 {
 | |
|                     type   = SamplerType.TextureBuffer;
 | |
|                     flags &= ~TextureFlags.LodLevel;
 | |
|                 }
 | |
| 
 | |
|                 switch (tldsOp.Target)
 | |
|                 {
 | |
|                     case TexelLoadTarget.Texture1DLodZero:
 | |
|                         sourcesList.Add(Ra());
 | |
| 
 | |
|                         if (type != SamplerType.TextureBuffer)
 | |
|                         {
 | |
|                             if (Sample1DAs2D)
 | |
|                             {
 | |
|                                 sourcesList.Add(ConstF(0));
 | |
| 
 | |
|                                 type &= ~SamplerType.Mask;
 | |
|                                 type |= SamplerType.Texture2D;
 | |
|                             }
 | |
| 
 | |
|                             sourcesList.Add(ConstF(0));
 | |
|                         }
 | |
|                         break;
 | |
| 
 | |
|                     case TexelLoadTarget.Texture1DLodLevel:
 | |
|                         sourcesList.Add(Ra());
 | |
| 
 | |
|                         if (Sample1DAs2D)
 | |
|                         {
 | |
|                             sourcesList.Add(ConstF(0));
 | |
| 
 | |
|                             type &= ~SamplerType.Mask;
 | |
|                             type |= SamplerType.Texture2D;
 | |
|                         }
 | |
| 
 | |
|                         sourcesList.Add(Rb());
 | |
|                         break;
 | |
| 
 | |
|                     case TexelLoadTarget.Texture2DLodZero:
 | |
|                         sourcesList.Add(Ra());
 | |
|                         sourcesList.Add(Rb());
 | |
|                         sourcesList.Add(Const(0));
 | |
|                         break;
 | |
| 
 | |
|                     case TexelLoadTarget.Texture2DLodZeroOffset:
 | |
|                         sourcesList.Add(Ra());
 | |
|                         sourcesList.Add(Ra());
 | |
|                         sourcesList.Add(Const(0));
 | |
|                         break;
 | |
| 
 | |
|                     case TexelLoadTarget.Texture2DLodZeroMultisample:
 | |
|                     case TexelLoadTarget.Texture2DLodLevel:
 | |
|                     case TexelLoadTarget.Texture2DLodLevelOffset:
 | |
|                         sourcesList.Add(Ra());
 | |
|                         sourcesList.Add(Ra());
 | |
|                         sourcesList.Add(Rb());
 | |
|                         break;
 | |
| 
 | |
|                     case TexelLoadTarget.Texture3DLodZero:
 | |
|                         sourcesList.Add(Ra());
 | |
|                         sourcesList.Add(Ra());
 | |
|                         sourcesList.Add(Rb());
 | |
|                         sourcesList.Add(Const(0));
 | |
|                         break;
 | |
| 
 | |
|                     case TexelLoadTarget.Texture2DArrayLodZero:
 | |
|                         sourcesList.Add(Rb());
 | |
|                         sourcesList.Add(Rb());
 | |
|                         sourcesList.Add(Ra());
 | |
|                         sourcesList.Add(Const(0));
 | |
|                         break;
 | |
|                 }
 | |
| 
 | |
|                 if ((flags & TextureFlags.Offset) != 0)
 | |
|                 {
 | |
|                     AddTextureOffset(type.GetDimensions(), 4, 4);
 | |
|                 }
 | |
|             }
 | |
|             else if (op is OpCodeTld4s tld4sOp)
 | |
|             {
 | |
|                 if (!(tld4sOp.HasDepthCompare || tld4sOp.HasOffset))
 | |
|                 {
 | |
|                     sourcesList.Add(Ra());
 | |
|                     sourcesList.Add(Rb());
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     sourcesList.Add(Ra());
 | |
|                     sourcesList.Add(Ra());
 | |
|                 }
 | |
| 
 | |
|                 type  = SamplerType.Texture2D;
 | |
|                 flags = TextureFlags.Gather;
 | |
| 
 | |
|                 if (tld4sOp.HasDepthCompare)
 | |
|                 {
 | |
|                     sourcesList.Add(Rb());
 | |
| 
 | |
|                     type |= SamplerType.Shadow;
 | |
|                 }
 | |
| 
 | |
|                 if (tld4sOp.HasOffset)
 | |
|                 {
 | |
|                     AddTextureOffset(type.GetDimensions(), 8, 6);
 | |
| 
 | |
|                     flags |= TextureFlags.Offset;
 | |
|                 }
 | |
| 
 | |
|                 sourcesList.Add(Const(tld4sOp.GatherCompIndex));
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 throw new InvalidOperationException($"Invalid opcode type \"{op.GetType().Name}\".");
 | |
|             }
 | |
| 
 | |
|             Operand[] sources = sourcesList.ToArray();
 | |
| 
 | |
|             Operand[] rd0 = new Operand[2] { ConstF(0), ConstF(0) };
 | |
|             Operand[] rd1 = new Operand[2] { ConstF(0), ConstF(0) };
 | |
| 
 | |
|             int destIncrement = 0;
 | |
| 
 | |
|             Operand GetDest()
 | |
|             {
 | |
|                 int high = destIncrement >> 1;
 | |
|                 int low  = destIncrement &  1;
 | |
| 
 | |
|                 destIncrement++;
 | |
| 
 | |
|                 if (op.IsFp16)
 | |
|                 {
 | |
|                     return high != 0
 | |
|                         ? (rd1[low] = Local())
 | |
|                         : (rd0[low] = Local());
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     int rdIndex = high != 0 ? op.Rd1.Index : op.Rd0.Index;
 | |
| 
 | |
|                     if (rdIndex < RegisterConsts.RegisterZeroIndex)
 | |
|                     {
 | |
|                         rdIndex += low;
 | |
|                     }
 | |
| 
 | |
|                     return Register(rdIndex, RegisterType.Gpr);
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             int handle = op.HandleOffset;
 | |
| 
 | |
|             for (int compMask = op.ComponentMask, compIndex = 0; compMask != 0; compMask >>= 1, compIndex++)
 | |
|             {
 | |
|                 if ((compMask & 1) != 0)
 | |
|                 {
 | |
|                     Operand dest = GetDest();
 | |
| 
 | |
|                     TextureOperation operation = context.CreateTextureOperation(
 | |
|                         Instruction.TextureSample,
 | |
|                         type,
 | |
|                         flags,
 | |
|                         handle,
 | |
|                         compIndex,
 | |
|                         dest,
 | |
|                         sources);
 | |
| 
 | |
|                     context.Add(operation);
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             if (op.IsFp16)
 | |
|             {
 | |
|                 context.Copy(Register(op.Rd0), context.PackHalf2x16(rd0[0], rd0[1]));
 | |
|                 context.Copy(Register(op.Rd1), context.PackHalf2x16(rd1[0], rd1[1]));
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public static void Tld4(EmitterContext context)
 | |
|         {
 | |
|             IOpCodeTld4 op = (IOpCodeTld4)context.CurrOp;
 | |
| 
 | |
|             if (op.Rd.IsRZ)
 | |
|             {
 | |
|                 return;
 | |
|             }
 | |
| 
 | |
|             int raIndex = op.Ra.Index;
 | |
|             int rbIndex = op.Rb.Index;
 | |
| 
 | |
|             Operand Ra()
 | |
|             {
 | |
|                 if (raIndex > RegisterConsts.RegisterZeroIndex)
 | |
|                 {
 | |
|                     return Const(0);
 | |
|                 }
 | |
| 
 | |
|                 return context.Copy(Register(raIndex++, RegisterType.Gpr));
 | |
|             }
 | |
| 
 | |
|             Operand Rb()
 | |
|             {
 | |
|                 if (rbIndex > RegisterConsts.RegisterZeroIndex)
 | |
|                 {
 | |
|                     return Const(0);
 | |
|                 }
 | |
| 
 | |
|                 return context.Copy(Register(rbIndex++, RegisterType.Gpr));
 | |
|             }
 | |
| 
 | |
|             Operand arrayIndex = op.IsArray ? Ra() : null;
 | |
| 
 | |
|             List<Operand> sourcesList = new List<Operand>();
 | |
| 
 | |
|             SamplerType  type  = ConvertSamplerType(op.Dimensions);
 | |
|             TextureFlags flags = TextureFlags.Gather;
 | |
| 
 | |
|             if (op.Bindless)
 | |
|             {
 | |
|                 sourcesList.Add(Rb());
 | |
| 
 | |
|                 flags |= TextureFlags.Bindless;
 | |
|             }
 | |
| 
 | |
|             int coordsCount = type.GetDimensions();
 | |
| 
 | |
|             for (int index = 0; index < coordsCount; index++)
 | |
|             {
 | |
|                 sourcesList.Add(Ra());
 | |
|             }
 | |
| 
 | |
|             bool is1DTo2D = Sample1DAs2D && type == SamplerType.Texture1D;
 | |
| 
 | |
|             if (is1DTo2D)
 | |
|             {
 | |
|                 sourcesList.Add(ConstF(0));
 | |
| 
 | |
|                 type = SamplerType.Texture2D;
 | |
|             }
 | |
| 
 | |
|             if (op.IsArray)
 | |
|             {
 | |
|                 sourcesList.Add(arrayIndex);
 | |
| 
 | |
|                 type |= SamplerType.Array;
 | |
|             }
 | |
| 
 | |
|             Operand[] packedOffs = new Operand[2];
 | |
| 
 | |
|             packedOffs[0] = op.Offset != TextureGatherOffset.None    ? Rb() : null;
 | |
|             packedOffs[1] = op.Offset == TextureGatherOffset.Offsets ? Rb() : null;
 | |
| 
 | |
|             if (op.HasDepthCompare)
 | |
|             {
 | |
|                 sourcesList.Add(Rb());
 | |
| 
 | |
|                 type |= SamplerType.Shadow;
 | |
|             }
 | |
| 
 | |
|             if (op.Offset != TextureGatherOffset.None)
 | |
|             {
 | |
|                 int offsetTexelsCount = op.Offset == TextureGatherOffset.Offsets ? 4 : 1;
 | |
| 
 | |
|                 for (int index = 0; index < coordsCount * offsetTexelsCount; index++)
 | |
|                 {
 | |
|                     Operand packed = packedOffs[(index >> 2) & 1];
 | |
| 
 | |
|                     sourcesList.Add(context.BitfieldExtractS32(packed, Const((index & 3) * 8), Const(6)));
 | |
|                 }
 | |
| 
 | |
|                 if (is1DTo2D)
 | |
|                 {
 | |
|                     for (int index = 0; index < offsetTexelsCount; index++)
 | |
|                     {
 | |
|                         sourcesList.Add(Const(0));
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|                 flags |= op.Offset == TextureGatherOffset.Offsets
 | |
|                     ? TextureFlags.Offsets
 | |
|                     : TextureFlags.Offset;
 | |
|             }
 | |
| 
 | |
|             sourcesList.Add(Const(op.GatherCompIndex));
 | |
| 
 | |
|             Operand[] sources = sourcesList.ToArray();
 | |
| 
 | |
|             int rdIndex = op.Rd.Index;
 | |
| 
 | |
|             Operand GetDest()
 | |
|             {
 | |
|                 if (rdIndex > RegisterConsts.RegisterZeroIndex)
 | |
|                 {
 | |
|                     return Const(0);
 | |
|                 }
 | |
| 
 | |
|                 return Register(rdIndex++, RegisterType.Gpr);
 | |
|             }
 | |
| 
 | |
|             int handle = op.HandleOffset;
 | |
| 
 | |
|             for (int compMask = op.ComponentMask, compIndex = 0; compMask != 0; compMask >>= 1, compIndex++)
 | |
|             {
 | |
|                 if ((compMask & 1) != 0)
 | |
|                 {
 | |
|                     Operand dest = GetDest();
 | |
| 
 | |
|                     TextureOperation operation = context.CreateTextureOperation(
 | |
|                         Instruction.TextureSample,
 | |
|                         type,
 | |
|                         flags,
 | |
|                         handle,
 | |
|                         compIndex,
 | |
|                         dest,
 | |
|                         sources);
 | |
| 
 | |
|                     context.Add(operation);
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public static void TmmlB(EmitterContext context)
 | |
|         {
 | |
|             EmitTextureMipMapLevel(context, true);
 | |
|         }
 | |
| 
 | |
|         public static void Tmml(EmitterContext context)
 | |
|         {
 | |
|             EmitTextureMipMapLevel(context, false);
 | |
|         }
 | |
| 
 | |
|         private static void EmitTextureMipMapLevel(EmitterContext context, bool isBindless)
 | |
|         {
 | |
|             OpCodeTexture op = (OpCodeTexture)context.CurrOp;
 | |
| 
 | |
|             if (op.Rd.IsRZ)
 | |
|             {
 | |
|                 return;
 | |
|             }
 | |
| 
 | |
|             int raIndex = op.Ra.Index;
 | |
|             int rbIndex = op.Rb.Index;
 | |
| 
 | |
|             Operand Ra()
 | |
|             {
 | |
|                 if (raIndex > RegisterConsts.RegisterZeroIndex)
 | |
|                 {
 | |
|                     return Const(0);
 | |
|                 }
 | |
| 
 | |
|                 return context.Copy(Register(raIndex++, RegisterType.Gpr));
 | |
|             }
 | |
| 
 | |
|             Operand Rb()
 | |
|             {
 | |
|                 if (rbIndex > RegisterConsts.RegisterZeroIndex)
 | |
|                 {
 | |
|                     return Const(0);
 | |
|                 }
 | |
| 
 | |
|                 return context.Copy(Register(rbIndex++, RegisterType.Gpr));
 | |
|             }
 | |
| 
 | |
|             TextureFlags flags = TextureFlags.None;
 | |
| 
 | |
|             List<Operand> sourcesList = new List<Operand>();
 | |
| 
 | |
|             if (isBindless)
 | |
|             {
 | |
|                 sourcesList.Add(Rb());
 | |
| 
 | |
|                 flags |= TextureFlags.Bindless;
 | |
|             }
 | |
| 
 | |
|             SamplerType type = ConvertSamplerType(op.Dimensions);
 | |
| 
 | |
|             int coordsCount = type.GetDimensions();
 | |
| 
 | |
|             Operand arrayIndex = op.IsArray ? Ra() : null;
 | |
| 
 | |
|             for (int index = 0; index < coordsCount; index++)
 | |
|             {
 | |
|                 sourcesList.Add(Ra());
 | |
|             }
 | |
| 
 | |
|             if (Sample1DAs2D && type == SamplerType.Texture1D)
 | |
|             {
 | |
|                 sourcesList.Add(ConstF(0));
 | |
| 
 | |
|                 type = SamplerType.Texture2D;
 | |
|             }
 | |
| 
 | |
|             if (op.IsArray)
 | |
|             {
 | |
|                 sourcesList.Add(arrayIndex);
 | |
| 
 | |
|                 type |= SamplerType.Array;
 | |
|             }
 | |
| 
 | |
|             Operand[] sources = sourcesList.ToArray();
 | |
| 
 | |
|             int rdIndex = op.Rd.Index;
 | |
| 
 | |
|             Operand GetDest()
 | |
|             {
 | |
|                 if (rdIndex > RegisterConsts.RegisterZeroIndex)
 | |
|                 {
 | |
|                     return Const(0);
 | |
|                 }
 | |
| 
 | |
|                 return Register(rdIndex++, RegisterType.Gpr);
 | |
|             }
 | |
| 
 | |
|             int handle = !isBindless ? op.HandleOffset : 0;
 | |
| 
 | |
|             for (int compMask = op.ComponentMask, compIndex = 0; compMask != 0; compMask >>= 1, compIndex++)
 | |
|             {
 | |
|                 if ((compMask & 1) != 0)
 | |
|                 {
 | |
|                     Operand dest = GetDest();
 | |
| 
 | |
|                     // Components z and w aren't standard, we return 0 in this case and add a comment.
 | |
|                     if (compIndex >= 2)
 | |
|                     {
 | |
|                         context.Add(new CommentNode("Unsupported component z or w found"));
 | |
|                         context.Copy(dest, Const(0));
 | |
|                     }
 | |
|                     else
 | |
|                     {
 | |
|                         Operand tempDest = Local();
 | |
| 
 | |
|                         TextureOperation operation = context.CreateTextureOperation(
 | |
|                             Instruction.Lod,
 | |
|                             type,
 | |
|                             flags,
 | |
|                             handle,
 | |
|                             compIndex ^ 1, // The instruction component order is the inverse of GLSL's.
 | |
|                             tempDest,
 | |
|                             sources);
 | |
| 
 | |
|                         context.Add(operation);
 | |
| 
 | |
|                         tempDest = context.FPMultiply(tempDest, ConstF(256.0f));
 | |
| 
 | |
|                         Operand fixedPointValue = context.FPConvertToS32(tempDest);
 | |
| 
 | |
|                         context.Copy(dest, fixedPointValue);
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public static void Txd(EmitterContext context)
 | |
|         {
 | |
|             OpCodeTxd op = (OpCodeTxd)context.CurrOp;
 | |
| 
 | |
|             if (op.Rd.IsRZ)
 | |
|             {
 | |
|                 return;
 | |
|             }
 | |
| 
 | |
|             int raIndex = op.Ra.Index;
 | |
|             int rbIndex = op.Rb.Index;
 | |
| 
 | |
|             Operand Ra()
 | |
|             {
 | |
|                 if (raIndex > RegisterConsts.RegisterZeroIndex)
 | |
|                 {
 | |
|                     return Const(0);
 | |
|                 }
 | |
| 
 | |
|                 return context.Copy(Register(raIndex++, RegisterType.Gpr));
 | |
|             }
 | |
| 
 | |
|             Operand Rb()
 | |
|             {
 | |
|                 if (rbIndex > RegisterConsts.RegisterZeroIndex)
 | |
|                 {
 | |
|                     return Const(0);
 | |
|                 }
 | |
| 
 | |
|                 return context.Copy(Register(rbIndex++, RegisterType.Gpr));
 | |
|             }
 | |
| 
 | |
|             TextureFlags flags = TextureFlags.Derivatives;
 | |
| 
 | |
|             List<Operand> sourcesList = new List<Operand>();
 | |
| 
 | |
|             if (op.IsBindless)
 | |
|             {
 | |
|                 sourcesList.Add(Ra());
 | |
| 
 | |
|                 flags |= TextureFlags.Bindless;
 | |
|             }
 | |
| 
 | |
|             SamplerType type = ConvertSamplerType(op.Dimensions);
 | |
| 
 | |
|             int coordsCount = type.GetDimensions();
 | |
| 
 | |
|             for (int index = 0; index < coordsCount; index++)
 | |
|             {
 | |
|                 sourcesList.Add(Ra());
 | |
|             }
 | |
| 
 | |
|             bool is1DTo2D = Sample1DAs2D && type == SamplerType.Texture1D;
 | |
| 
 | |
|             if (is1DTo2D)
 | |
|             {
 | |
|                 sourcesList.Add(ConstF(0));
 | |
| 
 | |
|                 type = SamplerType.Texture2D;
 | |
|             }
 | |
| 
 | |
|             Operand packedParams = Ra();
 | |
| 
 | |
|             if (op.IsArray)
 | |
|             {
 | |
|                 sourcesList.Add(context.BitwiseAnd(packedParams, Const(0xffff)));
 | |
| 
 | |
|                 type |= SamplerType.Array;
 | |
|             }
 | |
| 
 | |
|             // Derivatives (X and Y).
 | |
|             for (int dIndex = 0; dIndex < 2 * coordsCount; dIndex++)
 | |
|             {
 | |
|                 sourcesList.Add(Rb());
 | |
| 
 | |
|                 if (is1DTo2D)
 | |
|                 {
 | |
|                     sourcesList.Add(ConstF(0));
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             if (op.HasOffset)
 | |
|             {
 | |
|                 for (int index = 0; index < coordsCount; index++)
 | |
|                 {
 | |
|                     sourcesList.Add(context.BitfieldExtractS32(packedParams, Const(16 + index * 4), Const(4)));
 | |
|                 }
 | |
| 
 | |
|                 if (is1DTo2D)
 | |
|                 {
 | |
|                     sourcesList.Add(Const(0));
 | |
|                 }
 | |
| 
 | |
|                 flags |= TextureFlags.Offset;
 | |
|             }
 | |
| 
 | |
|             Operand[] sources = sourcesList.ToArray();
 | |
| 
 | |
|             int rdIndex = op.Rd.Index;
 | |
| 
 | |
|             Operand GetDest()
 | |
|             {
 | |
|                 if (rdIndex > RegisterConsts.RegisterZeroIndex)
 | |
|                 {
 | |
|                     return Const(0);
 | |
|                 }
 | |
| 
 | |
|                 return Register(rdIndex++, RegisterType.Gpr);
 | |
|             }
 | |
| 
 | |
|             int handle = !op.IsBindless ? op.HandleOffset : 0;
 | |
| 
 | |
|             for (int compMask = op.ComponentMask, compIndex = 0; compMask != 0; compMask >>= 1, compIndex++)
 | |
|             {
 | |
|                 if ((compMask & 1) != 0)
 | |
|                 {
 | |
|                     Operand dest = GetDest();
 | |
| 
 | |
|                     TextureOperation operation = context.CreateTextureOperation(
 | |
|                         Instruction.TextureSample,
 | |
|                         type,
 | |
|                         flags,
 | |
|                         handle,
 | |
|                         compIndex,
 | |
|                         dest,
 | |
|                         sources);
 | |
| 
 | |
|                     context.Add(operation);
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public static void Txq(EmitterContext context)
 | |
|         {
 | |
|             EmitTextureQuery(context, bindless: false);
 | |
|         }
 | |
| 
 | |
|         public static void TxqB(EmitterContext context)
 | |
|         {
 | |
|             EmitTextureQuery(context, bindless: true);
 | |
|         }
 | |
| 
 | |
|         private static void EmitTextureQuery(EmitterContext context, bool bindless)
 | |
|         {
 | |
|             OpCodeTex op = (OpCodeTex)context.CurrOp;
 | |
| 
 | |
|             if (op.Rd.IsRZ)
 | |
|             {
 | |
|                 return;
 | |
|             }
 | |
| 
 | |
|             context.Config.SetUsedFeature(FeatureFlags.IntegerSampling);
 | |
| 
 | |
|             TextureProperty property = (TextureProperty)op.RawOpCode.Extract(22, 6);
 | |
| 
 | |
|             // TODO: Validate and use property.
 | |
|             Instruction inst = Instruction.TextureSize;
 | |
| 
 | |
|             TextureFlags flags = bindless ? TextureFlags.Bindless : TextureFlags.None;
 | |
| 
 | |
|             int raIndex = op.Ra.Index;
 | |
| 
 | |
|             Operand Ra()
 | |
|             {
 | |
|                 if (raIndex > RegisterConsts.RegisterZeroIndex)
 | |
|                 {
 | |
|                     return Const(0);
 | |
|                 }
 | |
| 
 | |
|                 return context.Copy(Register(raIndex++, RegisterType.Gpr));
 | |
|             }
 | |
| 
 | |
|             List<Operand> sourcesList = new List<Operand>();
 | |
| 
 | |
|             if (bindless)
 | |
|             {
 | |
|                 sourcesList.Add(Ra());
 | |
|             }
 | |
| 
 | |
|             sourcesList.Add(Ra());
 | |
| 
 | |
|             Operand[] sources = sourcesList.ToArray();
 | |
| 
 | |
|             int rdIndex = op.Rd.Index;
 | |
| 
 | |
|             Operand GetDest()
 | |
|             {
 | |
|                 if (rdIndex > RegisterConsts.RegisterZeroIndex)
 | |
|                 {
 | |
|                     return Const(0);
 | |
|                 }
 | |
| 
 | |
|                 return Register(rdIndex++, RegisterType.Gpr);
 | |
|             }
 | |
| 
 | |
|             int handle = !bindless ? op.HandleOffset : 0;
 | |
| 
 | |
|             SamplerType type;
 | |
| 
 | |
|             if (bindless)
 | |
|             {
 | |
|                 type = (op.ComponentMask & 4) != 0 ? SamplerType.Texture3D : SamplerType.Texture2D;
 | |
|             } 
 | |
|             else
 | |
|             {
 | |
|                 type = context.Config.GpuAccessor.QuerySamplerType(handle);
 | |
|             }
 | |
| 
 | |
|             for (int compMask = op.ComponentMask, compIndex = 0; compMask != 0; compMask >>= 1, compIndex++)
 | |
|             {
 | |
|                 if ((compMask & 1) != 0)
 | |
|                 {
 | |
|                     Operand dest = GetDest();
 | |
| 
 | |
|                     TextureOperation operation = context.CreateTextureOperation(
 | |
|                         inst,
 | |
|                         type,
 | |
|                         flags,
 | |
|                         handle,
 | |
|                         compIndex,
 | |
|                         dest,
 | |
|                         sources);
 | |
| 
 | |
|                     context.Add(operation);
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         private static void EmitTextureSample(EmitterContext context, TextureFlags flags)
 | |
|         {
 | |
|             OpCodeTexture op = (OpCodeTexture)context.CurrOp;
 | |
| 
 | |
|             bool isBindless = (flags & TextureFlags.Bindless) != 0;
 | |
| 
 | |
|             if (op.Rd.IsRZ)
 | |
|             {
 | |
|                 return;
 | |
|             }
 | |
| 
 | |
|             int raIndex = op.Ra.Index;
 | |
|             int rbIndex = op.Rb.Index;
 | |
| 
 | |
|             Operand Ra()
 | |
|             {
 | |
|                 if (raIndex > RegisterConsts.RegisterZeroIndex)
 | |
|                 {
 | |
|                     return Const(0);
 | |
|                 }
 | |
| 
 | |
|                 return context.Copy(Register(raIndex++, RegisterType.Gpr));
 | |
|             }
 | |
| 
 | |
|             Operand Rb()
 | |
|             {
 | |
|                 if (rbIndex > RegisterConsts.RegisterZeroIndex)
 | |
|                 {
 | |
|                     return Const(0);
 | |
|                 }
 | |
| 
 | |
|                 return context.Copy(Register(rbIndex++, RegisterType.Gpr));
 | |
|             }
 | |
| 
 | |
|             Operand arrayIndex = op.IsArray ? Ra() : null;
 | |
| 
 | |
|             List<Operand> sourcesList = new List<Operand>();
 | |
| 
 | |
|             if (isBindless)
 | |
|             {
 | |
|                 sourcesList.Add(Rb());
 | |
|             }
 | |
| 
 | |
|             SamplerType type = ConvertSamplerType(op.Dimensions);
 | |
| 
 | |
|             bool hasLod = op.LodMode > TextureLodMode.LodZero;
 | |
| 
 | |
|             if (type == SamplerType.Texture1D && (flags & ~TextureFlags.Bindless) == TextureFlags.IntCoords && !(hasLod ||
 | |
|                 op.HasDepthCompare ||
 | |
|                 op.HasOffset ||
 | |
|                 op.IsArray ||
 | |
|                 op.IsMultisample))
 | |
|             {
 | |
|                 // For bindless, we don't have any way to know the texture type,
 | |
|                 // so we assume it's texture buffer when the sampler type is 1D, since that's more common.
 | |
|                 bool isTypeBuffer = isBindless || context.Config.GpuAccessor.QuerySamplerType(op.HandleOffset) == SamplerType.TextureBuffer;
 | |
| 
 | |
|                 if (isTypeBuffer)
 | |
|                 {
 | |
|                     type = SamplerType.TextureBuffer;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             int coordsCount = type.GetDimensions();
 | |
| 
 | |
|             for (int index = 0; index < coordsCount; index++)
 | |
|             {
 | |
|                 sourcesList.Add(Ra());
 | |
|             }
 | |
| 
 | |
|             if (Sample1DAs2D && type == SamplerType.Texture1D)
 | |
|             {
 | |
|                 sourcesList.Add(ConstF(0));
 | |
| 
 | |
|                 type = SamplerType.Texture2D;
 | |
|             }
 | |
| 
 | |
|             if (op.IsArray)
 | |
|             {
 | |
|                 sourcesList.Add(arrayIndex);
 | |
| 
 | |
|                 type |= SamplerType.Array;
 | |
|             }
 | |
| 
 | |
|             Operand lodValue = hasLod ? Rb() : ConstF(0);
 | |
| 
 | |
|             Operand packedOffs = op.HasOffset ? Rb() : null;
 | |
| 
 | |
|             if (op.HasDepthCompare)
 | |
|             {
 | |
|                 sourcesList.Add(Rb());
 | |
| 
 | |
|                 type |= SamplerType.Shadow;
 | |
|             }
 | |
| 
 | |
|             if ((op.LodMode == TextureLodMode.LodZero  ||
 | |
|                  op.LodMode == TextureLodMode.LodLevel ||
 | |
|                  op.LodMode == TextureLodMode.LodLevelA) && !op.IsMultisample && type != SamplerType.TextureBuffer)
 | |
|             {
 | |
|                 sourcesList.Add(lodValue);
 | |
| 
 | |
|                 flags |= TextureFlags.LodLevel;
 | |
|             }
 | |
| 
 | |
|             if (op.HasOffset)
 | |
|             {
 | |
|                 for (int index = 0; index < coordsCount; index++)
 | |
|                 {
 | |
|                     sourcesList.Add(context.BitfieldExtractS32(packedOffs, Const(index * 4), Const(4)));
 | |
|                 }
 | |
| 
 | |
|                 flags |= TextureFlags.Offset;
 | |
|             }
 | |
| 
 | |
|             if (op.LodMode == TextureLodMode.LodBias ||
 | |
|                 op.LodMode == TextureLodMode.LodBiasA)
 | |
|             {
 | |
|                 sourcesList.Add(lodValue);
 | |
| 
 | |
|                 flags |= TextureFlags.LodBias;
 | |
|             }
 | |
| 
 | |
|             if (op.IsMultisample)
 | |
|             {
 | |
|                 sourcesList.Add(Rb());
 | |
| 
 | |
|                 type |= SamplerType.Multisample;
 | |
|             }
 | |
| 
 | |
|             Operand[] sources = sourcesList.ToArray();
 | |
| 
 | |
|             int rdIndex = op.Rd.Index;
 | |
| 
 | |
|             Operand GetDest()
 | |
|             {
 | |
|                 if (rdIndex > RegisterConsts.RegisterZeroIndex)
 | |
|                 {
 | |
|                     return Const(0);
 | |
|                 }
 | |
| 
 | |
|                 return Register(rdIndex++, RegisterType.Gpr);
 | |
|             }
 | |
| 
 | |
|             int handle = !isBindless ? op.HandleOffset : 0;
 | |
| 
 | |
|             for (int compMask = op.ComponentMask, compIndex = 0; compMask != 0; compMask >>= 1, compIndex++)
 | |
|             {
 | |
|                 if ((compMask & 1) != 0)
 | |
|                 {
 | |
|                     Operand dest = GetDest();
 | |
| 
 | |
|                     TextureOperation operation = context.CreateTextureOperation(
 | |
|                         Instruction.TextureSample,
 | |
|                         type,
 | |
|                         flags,
 | |
|                         handle,
 | |
|                         compIndex,
 | |
|                         dest,
 | |
|                         sources);
 | |
| 
 | |
|                     context.Add(operation);
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         private static int GetComponents(IntegerSize size)
 | |
|         {
 | |
|             return size switch
 | |
|             {
 | |
|                 IntegerSize.B64   => 2,
 | |
|                 IntegerSize.B128  => 4,
 | |
|                 IntegerSize.UB128 => 4,
 | |
|                 _                 => 1
 | |
|             };
 | |
|         }
 | |
| 
 | |
|         private static int GetComponentSizeInBytesLog2(IntegerSize size)
 | |
|         {
 | |
|             return size switch
 | |
|             {
 | |
|                 IntegerSize.U8    => 0,
 | |
|                 IntegerSize.S8    => 0,
 | |
|                 IntegerSize.U16   => 1,
 | |
|                 IntegerSize.S16   => 1,
 | |
|                 IntegerSize.B32   => 2,
 | |
|                 IntegerSize.B64   => 3,
 | |
|                 IntegerSize.B128  => 4,
 | |
|                 IntegerSize.UB128 => 4,
 | |
|                 _                 => 2
 | |
|             };
 | |
|         }
 | |
| 
 | |
|         private static TextureFormat GetTextureFormat(IntegerSize size)
 | |
|         {
 | |
|             return size switch
 | |
|             {
 | |
|                 IntegerSize.U8    => TextureFormat.R8Uint,
 | |
|                 IntegerSize.S8    => TextureFormat.R8Sint,
 | |
|                 IntegerSize.U16   => TextureFormat.R16Uint,
 | |
|                 IntegerSize.S16   => TextureFormat.R16Sint,
 | |
|                 IntegerSize.B32   => TextureFormat.R32Uint,
 | |
|                 IntegerSize.B64   => TextureFormat.R32G32Uint,
 | |
|                 IntegerSize.B128  => TextureFormat.R32G32B32A32Uint,
 | |
|                 IntegerSize.UB128 => TextureFormat.R32G32B32A32Uint,
 | |
|                 _                 => TextureFormat.R32Uint
 | |
|             };
 | |
|         }
 | |
| 
 | |
|         private static int GetComponentSizeInBytesLog2(ReductionType type)
 | |
|         {
 | |
|             return type switch
 | |
|             {
 | |
|                 ReductionType.U32         => 2,
 | |
|                 ReductionType.S32         => 2,
 | |
|                 ReductionType.U64         => 3,
 | |
|                 ReductionType.FP32FtzRn   => 2,
 | |
|                 ReductionType.FP16x2FtzRn => 2,
 | |
|                 ReductionType.S64         => 3,
 | |
|                 ReductionType.SD32        => 2,
 | |
|                 ReductionType.SD64        => 3,
 | |
|                 _                         => 2
 | |
|             };
 | |
|         }
 | |
| 
 | |
|         private static TextureFormat GetTextureFormat(ReductionType type)
 | |
|         {
 | |
|             return type switch
 | |
|             {
 | |
|                 ReductionType.U32         => TextureFormat.R32Uint,
 | |
|                 ReductionType.S32         => TextureFormat.R32Sint,
 | |
|                 ReductionType.U64         => TextureFormat.R32G32Uint,
 | |
|                 ReductionType.FP32FtzRn   => TextureFormat.R32Float,
 | |
|                 ReductionType.FP16x2FtzRn => TextureFormat.R16G16Float,
 | |
|                 ReductionType.S64         => TextureFormat.R32G32Uint,
 | |
|                 ReductionType.SD32        => TextureFormat.R32Uint,
 | |
|                 ReductionType.SD64        => TextureFormat.R32G32Uint,
 | |
|                 _                         => TextureFormat.R32Uint
 | |
|             };
 | |
|         }
 | |
| 
 | |
|         private static TextureFlags GetAtomicOpFlags(AtomicOp op)
 | |
|         {
 | |
|             return op switch
 | |
|             {
 | |
|                 AtomicOp.Add                => TextureFlags.Add,
 | |
|                 AtomicOp.Minimum            => TextureFlags.Minimum,
 | |
|                 AtomicOp.Maximum            => TextureFlags.Maximum,
 | |
|                 AtomicOp.Increment          => TextureFlags.Increment,
 | |
|                 AtomicOp.Decrement          => TextureFlags.Decrement,
 | |
|                 AtomicOp.BitwiseAnd         => TextureFlags.BitwiseAnd,
 | |
|                 AtomicOp.BitwiseOr          => TextureFlags.BitwiseOr,
 | |
|                 AtomicOp.BitwiseExclusiveOr => TextureFlags.BitwiseXor,
 | |
|                 AtomicOp.Swap               => TextureFlags.Swap,
 | |
|                 _                           => TextureFlags.Add
 | |
|             };
 | |
|         }
 | |
| 
 | |
|         private static SamplerType ConvertSamplerType(ImageDimensions target)
 | |
|         {
 | |
|             return target switch
 | |
|             {
 | |
|                 ImageDimensions.Image1D      => SamplerType.Texture1D,
 | |
|                 ImageDimensions.ImageBuffer  => SamplerType.TextureBuffer,
 | |
|                 ImageDimensions.Image1DArray => SamplerType.Texture1D | SamplerType.Array,
 | |
|                 ImageDimensions.Image2D      => SamplerType.Texture2D,
 | |
|                 ImageDimensions.Image2DArray => SamplerType.Texture2D | SamplerType.Array,
 | |
|                 ImageDimensions.Image3D      => SamplerType.Texture3D,
 | |
|                 _                            => SamplerType.None
 | |
|             };
 | |
|         }
 | |
| 
 | |
|         private static SamplerType ConvertSamplerType(TextureDimensions dimensions)
 | |
|         {
 | |
|             return dimensions switch
 | |
|             {
 | |
|                 TextureDimensions.Texture1D   => SamplerType.Texture1D,
 | |
|                 TextureDimensions.Texture2D   => SamplerType.Texture2D,
 | |
|                 TextureDimensions.Texture3D   => SamplerType.Texture3D,
 | |
|                 TextureDimensions.TextureCube => SamplerType.TextureCube,
 | |
|                 _ => throw new ArgumentException($"Invalid texture dimensions \"{dimensions}\".")
 | |
|             };
 | |
|         }
 | |
| 
 | |
|         private static SamplerType ConvertSamplerType(TextureTarget type)
 | |
|         {
 | |
|             switch (type)
 | |
|             {
 | |
|                 case TextureTarget.Texture1DLodZero:
 | |
|                     return SamplerType.Texture1D;
 | |
| 
 | |
|                 case TextureTarget.Texture2D:
 | |
|                 case TextureTarget.Texture2DLodZero:
 | |
|                 case TextureTarget.Texture2DLodLevel:
 | |
|                     return SamplerType.Texture2D;
 | |
| 
 | |
|                 case TextureTarget.Texture2DDepthCompare:
 | |
|                 case TextureTarget.Texture2DLodLevelDepthCompare:
 | |
|                 case TextureTarget.Texture2DLodZeroDepthCompare:
 | |
|                     return SamplerType.Texture2D | SamplerType.Shadow;
 | |
| 
 | |
|                 case TextureTarget.Texture2DArray:
 | |
|                 case TextureTarget.Texture2DArrayLodZero:
 | |
|                     return SamplerType.Texture2D | SamplerType.Array;
 | |
| 
 | |
|                 case TextureTarget.Texture2DArrayLodZeroDepthCompare:
 | |
|                     return SamplerType.Texture2D | SamplerType.Array | SamplerType.Shadow;
 | |
| 
 | |
|                 case TextureTarget.Texture3D:
 | |
|                 case TextureTarget.Texture3DLodZero:
 | |
|                     return SamplerType.Texture3D;
 | |
| 
 | |
|                 case TextureTarget.TextureCube:
 | |
|                 case TextureTarget.TextureCubeLodLevel:
 | |
|                     return SamplerType.TextureCube;
 | |
|             }
 | |
| 
 | |
|             return SamplerType.None;
 | |
|         }
 | |
| 
 | |
|         private static SamplerType ConvertSamplerType(TexelLoadTarget type)
 | |
|         {
 | |
|             switch (type)
 | |
|             {
 | |
|                 case TexelLoadTarget.Texture1DLodZero:
 | |
|                 case TexelLoadTarget.Texture1DLodLevel:
 | |
|                     return SamplerType.Texture1D;
 | |
| 
 | |
|                 case TexelLoadTarget.Texture2DLodZero:
 | |
|                 case TexelLoadTarget.Texture2DLodZeroOffset:
 | |
|                 case TexelLoadTarget.Texture2DLodLevel:
 | |
|                 case TexelLoadTarget.Texture2DLodLevelOffset:
 | |
|                     return SamplerType.Texture2D;
 | |
| 
 | |
|                 case TexelLoadTarget.Texture2DLodZeroMultisample:
 | |
|                     return SamplerType.Texture2D | SamplerType.Multisample;
 | |
| 
 | |
|                 case TexelLoadTarget.Texture3DLodZero:
 | |
|                     return SamplerType.Texture3D;
 | |
| 
 | |
|                 case TexelLoadTarget.Texture2DArrayLodZero:
 | |
|                     return SamplerType.Texture2D | SamplerType.Array;
 | |
|             }
 | |
| 
 | |
|             return SamplerType.None;
 | |
|         }
 | |
| 
 | |
|         private static TextureFlags ConvertTextureFlags(TextureTarget type)
 | |
|         {
 | |
|             switch (type)
 | |
|             {
 | |
|                 case TextureTarget.Texture1DLodZero:
 | |
|                 case TextureTarget.Texture2DLodZero:
 | |
|                 case TextureTarget.Texture2DLodLevel:
 | |
|                 case TextureTarget.Texture2DLodLevelDepthCompare:
 | |
|                 case TextureTarget.Texture2DLodZeroDepthCompare:
 | |
|                 case TextureTarget.Texture2DArrayLodZero:
 | |
|                 case TextureTarget.Texture2DArrayLodZeroDepthCompare:
 | |
|                 case TextureTarget.Texture3DLodZero:
 | |
|                 case TextureTarget.TextureCubeLodLevel:
 | |
|                     return TextureFlags.LodLevel;
 | |
| 
 | |
|                 case TextureTarget.Texture2D:
 | |
|                 case TextureTarget.Texture2DDepthCompare:
 | |
|                 case TextureTarget.Texture2DArray:
 | |
|                 case TextureTarget.Texture3D:
 | |
|                 case TextureTarget.TextureCube:
 | |
|                     return TextureFlags.None;
 | |
|             }
 | |
| 
 | |
|             return TextureFlags.None;
 | |
|         }
 | |
| 
 | |
|         private static TextureFlags ConvertTextureFlags(TexelLoadTarget type)
 | |
|         {
 | |
|             switch (type)
 | |
|             {
 | |
|                 case TexelLoadTarget.Texture1DLodZero:
 | |
|                 case TexelLoadTarget.Texture1DLodLevel:
 | |
|                 case TexelLoadTarget.Texture2DLodZero:
 | |
|                 case TexelLoadTarget.Texture2DLodLevel:
 | |
|                 case TexelLoadTarget.Texture2DLodZeroMultisample:
 | |
|                 case TexelLoadTarget.Texture3DLodZero:
 | |
|                 case TexelLoadTarget.Texture2DArrayLodZero:
 | |
|                     return TextureFlags.LodLevel;
 | |
| 
 | |
|                 case TexelLoadTarget.Texture2DLodZeroOffset:
 | |
|                 case TexelLoadTarget.Texture2DLodLevelOffset:
 | |
|                     return TextureFlags.LodLevel | TextureFlags.Offset;
 | |
|             }
 | |
| 
 | |
|             return TextureFlags.None;
 | |
|         }
 | |
|     }
 | |
| } |