Ryujinx/Ryujinx.Graphics/Gal/Shader/SpirvDecompiler.cs

1146 lines
37 KiB
C#
Raw Normal View History

2018-06-13 00:25:01 -03:00
using System;
using System.IO;
using System.Collections.Generic;
using Ryujinx.Graphics.Gal.Shader.SPIRV;
namespace Ryujinx.Graphics.Gal.Shader
{
public class SpirvDecompiler
{
private class SpirvVariable
{
public Instruction Id;
public StorageClass Storage;
public string Name;
public int Location = -1;
}
private enum OperType
{
Bool,
F32,
I32
}
private delegate int LocationAllocator();
private delegate Instruction GetInstExpr(ShaderIrOp Op);
private Dictionary<ShaderIrInst, GetInstExpr> InstsExpr;
private GlslDecl Decl;
private ShaderIrBlock[] Blocks;
private Assembler Assembler;
private int UniformCount = 0;
private int InAttributeCount = 0;
private int OutAttributeCount = 0;
private Instruction TypeVoid,
TypeBool, TypeBool_Private,
TypeInt, TypeUInt,
TypeFloat,
TypeImage2D, TypeSampler2D, TypeSampler2D_Uniform;
private Instruction[] TypeFloats, TypeFloats_In, TypeFloats_Out, TypeFloats_Uniform, TypeFloats_Private;
private Instruction TrueConstant, FalseConstant;
//Holds debug info, they are not needed but do not hurt to add
private List<Instruction> Names;
//Types and constants have to be defined sequentially
//because some types require constants and most constants require types
private List<Instruction> TypesConstants;
private List<Instruction> Decorates;
//Variables declarations. They are different to "Variables" declared below
//These holds instructions
private List<Instruction> VarsDeclaration;
private List<Instruction> Code;
private List<SpirvVariable> Variables;
private Instruction Main;
private Instruction PerVertexVar = null;
private Instruction GlslExtension;
private GLSLstd450Builder Glsl450;
//
private Instruction GetOperExpr(ShaderIrOp Op, ShaderIrNode Oper)
{
return GetExprWithCast(Op, Oper, GetSrcExpr(Oper));
}
private Instruction GetFcltExpr(ShaderIrOp Op)
{
return new OpFOrdLessThan(
TypeBool,
GetOperExpr(Op, Op.OperandA),
GetOperExpr(Op, Op.OperandB));
}
private Instruction GetSrcNodeTypeId(ShaderIrOp Op)
{
switch (GetSrcNodeType(Op))
{
case OperType.Bool: return TypeBool;
case OperType.F32: return TypeFloat;
case OperType.I32: return TypeInt;
default:
throw new InvalidOperationException();
}
}
private Instruction GetFcgtExpr(ShaderIrOp Op)
{
return new OpFOrdGreaterThan(
TypeBool,
GetOperExpr(Op, Op.OperandA),
GetOperExpr(Op, Op.OperandB));
}
private Instruction GetFnegExpr(ShaderIrOp Op)
{
return new OpFNegate(TypeFloat, GetOperExpr(Op, Op.OperandA));
}
private Instruction GetFaddExpr(ShaderIrOp Op)
{
return new OpFAdd(
TypeFloat,
GetOperExpr(Op, Op.OperandA),
GetOperExpr(Op, Op.OperandB));
}
private Instruction GetFmulExpr(ShaderIrOp Op)
{
return new OpFMul(
TypeFloat,
GetOperExpr(Op, Op.OperandA),
GetOperExpr(Op, Op.OperandB));
}
private Instruction GetFfmaExpr(ShaderIrOp Op)
{
return Glsl450.Fma(
TypeFloat,
GetOperExpr(Op, Op.OperandA),
GetOperExpr(Op, Op.OperandB),
GetOperExpr(Op, Op.OperandC));
}
//
public SpirvDecompiler()
{
InstsExpr = new Dictionary<ShaderIrInst, GetInstExpr>()
{
{ ShaderIrInst.Fadd, GetFaddExpr },
{ ShaderIrInst.Fmul, GetFmulExpr },
{ ShaderIrInst.Fclt, GetFcltExpr },
{ ShaderIrInst.Fcgt, GetFcgtExpr },
{ ShaderIrInst.Fneg, GetFnegExpr },
{ ShaderIrInst.Ffma, GetFfmaExpr }
};
Assembler = new Assembler();
GlslExtension = new OpExtInstImport("GLSL.std.450");
Glsl450 = new GLSLstd450Builder(GlslExtension);
TypesConstants = new List<Instruction>();
Names = new List<Instruction>();
Decorates = new List<Instruction>();
VarsDeclaration = new List<Instruction>();
Code = new List<Instruction>();
Variables = new List<SpirvVariable>();
//Declare types types
TypeVoid = new OpTypeVoid();
TypeBool = new OpTypeBool();
TypeBool_Private = new OpTypePointer(StorageClass.Private, TypeBool);
TrueConstant = new OpConstantTrue(TypeBool);
FalseConstant = new OpConstantFalse(TypeBool);
TypeInt = new OpTypeInt(32, true);
TypeUInt = new OpTypeInt(32, false);
TypeFloats = new Instruction[4];
TypeFloats_In = new Instruction[4];
TypeFloats_Out = new Instruction[4];
TypeFloats_Uniform = new Instruction[4];
TypeFloats_Private = new Instruction[4];
TypeFloat = new OpTypeFloat(32);
TypeFloats[0] = TypeFloat;
for (int i = 0; i < 4; i++)
{
if (i > 0)
{
TypeFloats[i] = new OpTypeVector(TypeFloat, i+1);
}
TypeFloats_In[i] = new OpTypePointer(StorageClass.Input, TypeFloats[i]);
TypeFloats_Out[i] = new OpTypePointer(StorageClass.Output, TypeFloats[i]);
TypeFloats_Uniform[i] = new OpTypePointer(StorageClass.UniformConstant, TypeFloats[i]);
TypeFloats_Private[i] = new OpTypePointer(StorageClass.Private, TypeFloats[i]);
}
TypeImage2D = new OpTypeImage(TypeFloat, Dim.Dim2D, 0, 0, 0, 0, ImageFormat.Unknown);
TypeSampler2D = new OpTypeSampledImage(TypeImage2D);
TypeSampler2D_Uniform = new OpTypePointer(StorageClass.UniformConstant, TypeSampler2D);
//Add them (these do not need to be added safely)
TypesConstants.Add(TypeVoid);
TypesConstants.Add(TypeBool);
TypesConstants.Add(TypeBool_Private);
TypesConstants.Add(TrueConstant);
TypesConstants.Add(FalseConstant);
TypesConstants.Add(TypeInt);
TypesConstants.Add(TypeUInt);
for (int i = 0; i < 4; i++)
{
TypesConstants.Add(TypeFloats[i]);
TypesConstants.Add(TypeFloats_In[i]);
TypesConstants.Add(TypeFloats_Out[i]);
TypesConstants.Add(TypeFloats_Uniform[i]);
TypesConstants.Add(TypeFloats_Private[i]);
}
TypesConstants.Add(TypeImage2D);
TypesConstants.Add(TypeSampler2D);
TypesConstants.Add(TypeSampler2D_Uniform);
}
public byte[] Decompile(IGalMemory Memory, long Position, GalShaderType ShaderType)
{
Blocks = ShaderDecoder.Decode(Memory, Position);
Decl = new GlslDecl(Blocks, ShaderType);
BuildBuiltIns();
BuildTextures();
BuildUniforms();
BuildInAttributes();
BuildOutAttributes();
BuildGprs();
BuildDeclPreds();
BuildMain();
BuildCode();
PrintHeader();
PrintEntryPoint();
Assembler.Add(Names);
Assembler.Add(Decorates);
Assembler.Add(TypesConstants);
Assembler.Add(VarsDeclaration);
Assembler.Add(Code);
//Temp code ahead
var Stream = new MemoryStream();
Assembler.Write(Stream);
byte[] Bytecode = Stream.ToArray();
Stream.Close();
return Bytecode;
}
private void BuildBuiltIns()
{
switch (Decl.ShaderType)
{
case GalShaderType.Vertex:
Instruction One = AllocConstant(TypeUInt, new LiteralNumber((int)1));
Instruction ArrFloatUInt1 = new OpTypeArray(TypeFloat, One);
Instruction PerVertex = new OpTypeStruct(TypeFloats[3], TypeFloat, ArrFloatUInt1, ArrFloatUInt1);
Instruction OutputPerVertex = new OpTypePointer(StorageClass.Output, PerVertex);
Decorates.Add(new OpMemberDecorate(PerVertex, 0, Decoration.BuiltIn, BuiltIn.Position));
Decorates.Add(new OpMemberDecorate(PerVertex, 1, Decoration.BuiltIn, BuiltIn.PointSize));
Decorates.Add(new OpMemberDecorate(PerVertex, 2, Decoration.BuiltIn, BuiltIn.ClipDistance));
Decorates.Add(new OpMemberDecorate(PerVertex, 3, Decoration.BuiltIn, BuiltIn.CullDistance));
Decorates.Add(new OpDecorate(PerVertex, Decoration.Block));
TypesConstants.Add(ArrFloatUInt1);
TypesConstants.Add(PerVertex);
TypesConstants.Add(OutputPerVertex);
PerVertexVar = new OpVariable(OutputPerVertex, StorageClass.Output);
VarsDeclaration.Add(PerVertexVar);
Names.Add(new OpName(PerVertex, "gl_PerVertex"));
break;
}
}
private void BuildTextures()
{
foreach (ShaderDeclInfo DeclInfo in Decl.Textures.Values)
{
Instruction Variable = AllocLocatedVariable(
TypeSampler2D_Uniform,
StorageClass.UniformConstant,
DeclInfo.Name,
AllocUniformLocation);
//TODO What is a "DescriptorSet"? It sounds like something from Vulkan
Decorates.Add(new OpDecorate(Variable, Decoration.DescriptorSet, new LiteralNumber(0)));
}
}
private void BuildUniforms()
{
if (Decl.ShaderType == GalShaderType.Vertex)
{
Instruction Float2 = TypeFloats_Uniform[1];
AllocLocatedVariable(
Float2,
StorageClass.UniformConstant,
GalConsts.FlipUniformName,
AllocUniformLocation);
}
foreach (ShaderDeclInfo DeclInfo in Decl.Uniforms.Values)
{
Instruction ElemType = TypeFloats[DeclInfo.Size - 1];
Operand Size = new LiteralNumber(DeclInfo.Index + 1);
Instruction Constant = AllocConstant(TypeInt, Size);
Instruction ArrayType = AllocType(new OpTypeArray(ElemType, Constant));
Instruction ArrayTypeUniform = AllocType(new OpTypePointer(StorageClass.UniformConstant, ArrayType));
AllocLocatedVariable(
ArrayTypeUniform,
StorageClass.UniformConstant,
DeclInfo.Name,
AllocUniformLocation);
}
}
private void BuildInAttributes()
{
if (Decl.ShaderType == GalShaderType.Fragment)
{
AllocLocatedVariable(
TypeFloats_In[3],
StorageClass.Input,
GlslDecl.PositionOutAttrName,
AllocInAttributeLocation);
}
BuildAttributes(
Decl.InAttributes.Values,
TypeFloats_In,
StorageClass.Input,
AllocInAttributeLocation);
}
private void BuildOutAttributes()
{
if (Decl.ShaderType == GalShaderType.Vertex)
{
AllocLocatedVariable(
TypeFloats_In[3],
StorageClass.Output,
GlslDecl.PositionOutAttrName,
AllocOutAttributeLocation);
}
BuildAttributes(
Decl.OutAttributes.Values,
TypeFloats_Out,
StorageClass.Output,
AllocOutAttributeLocation);
}
private void BuildGprs()
{
foreach (ShaderDeclInfo DeclInfo in Decl.Gprs.Values)
{
Instruction Type = TypeFloats_Private[DeclInfo.Size - 1];
AllocVariable(Type, StorageClass.Private, DeclInfo.Name);
}
}
private void BuildDeclPreds()
{
foreach (ShaderDeclInfo DeclInfo in Decl.Preds.Values)
{
AllocVariable(TypeBool_Private, StorageClass.Private, DeclInfo.Name);
}
}
private void BuildAttributes(
IEnumerable<ShaderDeclInfo> Decls,
Instruction[] TypeFloats_InOut,
StorageClass StorageClass,
LocationAllocator Allocator)
{
foreach (ShaderDeclInfo DeclInfo in Decls)
{
if (DeclInfo.Index >= 0)
{
Instruction Type = TypeFloats_InOut[DeclInfo.Size - 1];
AllocLocatedVariable(Type, StorageClass, DeclInfo.Name, Allocator);
}
}
}
private void BuildMain()
{
Instruction TypeFunction = AllocType(new OpTypeFunction(TypeVoid));
Main = new OpFunction(TypeVoid, FunctionControl.None, TypeFunction);
}
private void BuildCode()
{
Code.Add(Main);
// First label is implicit when building first block
Dictionary<ShaderIrBlock, Instruction> Labels = new Dictionary<ShaderIrBlock, Instruction>();
foreach (ShaderIrBlock Block in Blocks)
{
Labels[Block] = new OpLabel();
}
Instruction EndLabel = new OpLabel();
for (int BlockIndex = 0; BlockIndex < Blocks.Length; BlockIndex++)
{
BuildBlock(BlockIndex, Labels, EndLabel);
}
Code.Add(EndLabel);
//TODO
//SB.AppendLine(Identation + "gl_Position.xy *= flip;");
//SB.AppendLine(Identation + GlslDecl.PositionOutAttrName + " = gl_Position;");
//SB.AppendLine(Identation + GlslDecl.PositionOutAttrName + ".w = 1;");
Code.Add(new OpReturn());
Code.Add(new OpFunctionEnd());
}
private void BuildBlock(
int BlockIndex,
Dictionary<ShaderIrBlock, Instruction> Labels,
Instruction EndLabel)
{
ShaderIrBlock Block = Blocks[BlockIndex];
Code.Add(Labels[Block]);
bool HasBranchTail = BuildNodes(Block, Block.Nodes, Labels, EndLabel);
// No unconditional branch instruction found. Branch to next block
if (!HasBranchTail)
{
Instruction Label;
if (Block.Next != null)
{
Label = Labels[Block.Next];
}
else if (BlockIndex + 1 < Blocks.Length)
{
ShaderIrBlock NextBlock = Blocks[BlockIndex + 1];
Label = Labels[NextBlock];
}
else
{
Label = EndLabel;
}
Code.Add(new OpBranch(Label));
}
}
private bool BuildNodes(
ShaderIrBlock Block,
List<ShaderIrNode> Nodes,
Dictionary<ShaderIrBlock, Instruction> Labels,
Instruction EndLabel)
{
foreach (ShaderIrNode Node in Nodes)
{
if (Node is ShaderIrCond Cond)
{
Instruction CondExpr = GetSrcExpr(Cond.Pred, true);
if (Cond.Not)
{
CondExpr = new OpLogicalNot(TypeBool, CondExpr);
Code.Add(CondExpr);
}
if (Cond.Child is ShaderIrOp Op && Op.Inst == ShaderIrInst.Bra)
{
Instruction BranchLabel = Labels[Block.Branch];
Instruction SkipLabel = new OpLabel();
Instruction Branch = new OpBranchConditional(CondExpr, BranchLabel, SkipLabel);
Code.Add(Branch);
Code.Add(SkipLabel);
}
else
{
Instruction ExecuteLabel = new OpLabel();
Instruction SkipLabel = new OpLabel();
Instruction Branch = new OpBranchConditional(CondExpr, ExecuteLabel, SkipLabel);
Code.Add(Branch);
Code.Add(ExecuteLabel);
List<ShaderIrNode> ChildList = new List<ShaderIrNode>();
ChildList.Add(Cond.Child);
bool HasBranchTail = BuildNodes(Block, ChildList, Labels, EndLabel);
if (!HasBranchTail)
{
Code.Add(new OpBranch(SkipLabel));
}
Code.Add(SkipLabel);
}
}
else if (Node is ShaderIrAsg Asg)
{
if (IsValidOutOper(Asg.Dst))
{
Instruction SrcExpr = GetSrcExpr(Asg.Src, true);
Instruction Expr = GetExprWithCast(Asg.Dst, Asg.Src, SrcExpr);
Instruction Target = GetDstOperName(Asg.Dst);
Code.Add(new OpStore(Target, Expr));
}
}
else if (Node is ShaderIrOp Op)
{
if (Op.Inst == ShaderIrInst.Bra)
{
Instruction BranchLabel = Labels[Block.Branch];
Code.Add(new OpBranch(BranchLabel));
//Unconditional branch found, ignore following nodes in this hierarchy
return true;
}
else if (Op.Inst == ShaderIrInst.Exit)
{
Code.Add(new OpBranch(EndLabel));
//Ignore following nodes, same as ^
return true;
}
else
{
Instruction Operation = GetSrcExpr(Op, true);
}
}
else if (Node is ShaderIrCmnt Comment)
{
// Couldn't find a commentary OpCode in Spirv, for now just ignore it
}
else
{
throw new InvalidOperationException();
}
}
return false;
}
private Instruction GetExprWithCast(ShaderIrNode Dst, ShaderIrNode Src, Instruction Expr)
{
//Note: The "DstType" (of the cast) is the type that the operation
//uses on the source operands, while the "SrcType" is the destination
//type of the operand result (if it is a operation) or just the type
//of the variable for registers/uniforms/attributes.
OperType DstType = GetSrcNodeType(Dst);
OperType SrcType = GetDstNodeType(Src);
if (DstType != SrcType)
{
//Check for invalid casts
//(like bool to int/float and others).
if (SrcType != OperType.F32 &&
SrcType != OperType.I32)
{
throw new InvalidOperationException();
}
switch (Src)
{
case ShaderIrOperGpr Gpr:
{
//When the Gpr is ZR, just return the 0 value directly,
//since the float encoding for 0 is 0.
if (Gpr.IsConst)
{
return AllocConstant(TypeFloat, new LiteralNumber(0f));
}
break;
}
case ShaderIrOperImm Imm:
{
if (DstType == OperType.F32)
{
float Value = BitConverter.Int32BitsToSingle(Imm.Value);
if (!float.IsNaN(Value) && !float.IsInfinity(Value))
{
return AllocConstant(TypeFloat, new LiteralNumber(Value));
}
}
break;
}
}
switch (DstType)
{
case OperType.F32:
return new OpBitcast(TypeFloat, Expr);
case OperType.I32:
return new OpBitcast(TypeInt, Expr);
}
}
return Expr;
}
private static OperType GetDstNodeType(ShaderIrNode Node)
{
//Special case instructions with the result type different
//from the input types (like integer <-> float conversion) here.
if (Node is ShaderIrOp Op)
{
switch (Op.Inst)
{
case ShaderIrInst.Stof:
case ShaderIrInst.Txlf:
case ShaderIrInst.Utof:
return OperType.F32;
case ShaderIrInst.Ftos:
case ShaderIrInst.Ftou:
return OperType.I32;
}
}
return GetSrcNodeType(Node);
}
private static OperType GetSrcNodeType(ShaderIrNode Node)
{
switch (Node)
{
case ShaderIrOperAbuf Abuf:
return Abuf.Offs == GlslDecl.VertexIdAttr
? OperType.I32
: OperType.F32;
case ShaderIrOperCbuf Cbuf: return OperType.F32;
case ShaderIrOperGpr Gpr: return OperType.F32;
case ShaderIrOperImm Imm: return OperType.I32;
case ShaderIrOperImmf Immf: return OperType.F32;
case ShaderIrOperPred Pred: return OperType.Bool;
case ShaderIrOp Op:
if (Op.Inst > ShaderIrInst.B_Start &&
Op.Inst < ShaderIrInst.B_End)
{
return OperType.Bool;
}
else if (Op.Inst > ShaderIrInst.F_Start &&
Op.Inst < ShaderIrInst.F_End)
{
return OperType.F32;
}
else if (Op.Inst > ShaderIrInst.I_Start &&
Op.Inst < ShaderIrInst.I_End)
{
return OperType.I32;
}
break;
}
throw new ArgumentException(nameof(Node));
}
private bool IsValidOutOper(ShaderIrNode Node)
{
if (Node is ShaderIrOperGpr Gpr && Gpr.IsConst)
{
return false;
}
else if (Node is ShaderIrOperPred Pred && Pred.IsConst)
{
return false;
}
return true;
}
private Instruction GetDstOperName(ShaderIrNode Node)
{
if (Node is ShaderIrOperAbuf Abuf)
{
return GetOutAbufName(Abuf);
}
else if (Node is ShaderIrOperGpr Gpr)
{
return GetName(Gpr, true);
}
else if (Node is ShaderIrOperPred Pred)
{
return GetName(Pred, true);
}
throw new ArgumentException(nameof(Node));
}
private Instruction GetOutAbufName(ShaderIrOperAbuf Abuf)
{
return GetName(Decl.OutAttributes, Abuf, true);
}
private Instruction GetSrcExpr(ShaderIrNode Node, bool Entry = false)
{
switch (Node)
{
case ShaderIrOperAbuf Abuf: return GetName (Abuf, false);
case ShaderIrOperCbuf Cbuf: return GetName (Cbuf, false);
case ShaderIrOperGpr Gpr: return GetName (Gpr, false);
case ShaderIrOperImm Imm: return GetValue(Imm);
case ShaderIrOperImmf Immf: return GetValue(Immf);
case ShaderIrOperPred Pred: return GetName (Pred, false);
case ShaderIrOp Op:
Instruction Expr;
if (InstsExpr.TryGetValue(Op.Inst, out GetInstExpr GetExpr))
{
Expr = GetExpr(Op);
}
else
{
throw new NotImplementedException(Op.Inst.ToString());
}
//GetExpr does not add it to Code
Code.Add(Expr);
return Expr;
default: throw new ArgumentException(nameof(Node));
}
}
private Instruction GetName(ShaderIrOperAbuf Abuf, bool Pointer)
{
if (Decl.ShaderType == GalShaderType.Vertex)
{
switch (Abuf.Offs)
{
case GlslDecl.VertexIdAttr: return GetVariableValue(TypeInt, "gl_VertexID", Pointer);
case GlslDecl.InstanceIdAttr: return GetVariableValue(TypeInt, "gl_InstanceID", Pointer);
}
}
else if (Decl.ShaderType == GalShaderType.TessEvaluation)
{
switch (Abuf.Offs)
{
case GlslDecl.TessCoordAttrX: return GetVariableValue(TypeFloat, "gl_TessCoord", 0, Pointer);
case GlslDecl.TessCoordAttrY: return GetVariableValue(TypeFloat, "gl_TessCoord", 1, Pointer);
case GlslDecl.TessCoordAttrZ: return GetVariableValue(TypeFloat, "gl_TessCoord", 2, Pointer);
}
}
return GetName(Decl.InAttributes, Abuf, Pointer);
}
private Instruction GetName(IReadOnlyDictionary<int, ShaderDeclInfo> Dict, ShaderIrOperAbuf Abuf, bool Pointer)
{
int Index = Abuf.Offs >> 4;
int Elem = (Abuf.Offs >> 2) & 3;
if (!Dict.TryGetValue(Index, out ShaderDeclInfo DeclInfo))
{
throw new InvalidOperationException();
}
//Guess types are float-based
if (DeclInfo.Size > 1)
{
return GetVariableValue(TypeFloat, DeclInfo.Name, Elem, Pointer);
}
else
{
return GetVariableValue(TypeFloat, DeclInfo.Name, Pointer);
}
}
private Instruction GetName(ShaderIrOperGpr Gpr, bool Pointer)
{
//Gprs are always float, right?
if (Gpr.IsConst)
{
if (Pointer)
{
throw new InvalidOperationException("Can't return pointer to a constant");
}
return AllocConstant(TypeFloat, new LiteralNumber(0f));
}
else
{
return GetNameWithSwizzle(TypeFloat, Decl.Gprs, Gpr.Index, Pointer);
}
}
private Instruction GetValue(ShaderIrOperImm Imm)
{
return AllocConstant(TypeInt, new LiteralNumber(Imm.Value));
}
private Instruction GetValue(ShaderIrOperImmf Immf)
{
return AllocConstant(TypeFloat, new LiteralNumber(Immf.Value));
}
private Instruction GetName(ShaderIrOperPred Pred, bool Pointer)
{
if (Pred.IsConst)
{
return TrueConstant;
}
else
{
return GetNameWithSwizzle(TypeBool, Decl.Preds, Pred.Index, Pointer);
}
}
private Instruction GetName(ShaderIrOperCbuf Cbuf, bool Pointer)
{
if (!Decl.Uniforms.TryGetValue(Cbuf.Index, out ShaderDeclInfo DeclInfo))
{
throw new InvalidOperationException();
}
Instruction PosConstant = AllocConstant(TypeInt, new LiteralNumber(Cbuf.Pos));
if (Cbuf.Offs != null)
{
//Note: We assume that the register value is always a multiple of 4.
//This may not be always the case.
Instruction ShiftConstant = AllocConstant(TypeInt, new LiteralNumber(2));
Instruction Source = GetSrcExpr(Cbuf.Offs);
Instruction Casted = new OpBitcast(TypeInt, Source);
Instruction Offset = new OpShiftRightLogical(TypeInt, Casted, ShiftConstant);
Instruction Index = new OpIAdd(TypeInt, PosConstant, Offset);
Code.Add(Source);
Code.Add(Casted);
Code.Add(Offset);
Code.Add(Index);
return GetVariableValue(TypeFloat, DeclInfo.Name, Index, Pointer);
}
else
{
return GetVariableValue(TypeFloat, DeclInfo.Name, PosConstant, Pointer);
}
}
private Instruction GetNameWithSwizzle(
Instruction Type,
IReadOnlyDictionary<int, ShaderDeclInfo> Dict,
int Index,
bool Pointer)
{
int VecIndex = Index >> 2;
if (Dict.TryGetValue(VecIndex, out ShaderDeclInfo DeclInfo))
{
if (DeclInfo.Size > 1 && Index < VecIndex + DeclInfo.Size)
{
return GetVariableValue(Type, DeclInfo.Name, Index & 3, Pointer);
}
}
if (!Dict.TryGetValue(Index, out DeclInfo))
{
throw new InvalidOperationException();
}
return GetVariableValue(Type, DeclInfo.Name, Pointer);
}
private void PrintHeader()
{
Assembler.Add(new OpCapability(Capability.Shader));
Assembler.Add(GlslExtension);
Assembler.Add(new OpMemoryModel(AddressingModel.Logical, MemoryModel.GLSL450));
}
private void PrintTypes()
{
Assembler.Add(TypesConstants);
}
private void PrintEntryPoint()
{
List<Instruction> Interface = new List<Instruction>();
foreach (SpirvVariable Variable in Variables)
{
if (Variable.Storage == StorageClass.Input
|| Variable.Storage == StorageClass.Output)
{
Interface.Add(Variable.Id);
}
}
Assembler.Add(new OpEntryPoint(
GetExecutionModel(),
Main,
"main",
Interface.ToArray()));
}
private ExecutionModel GetExecutionModel()
{
switch (Decl.ShaderType)
{
case GalShaderType.Vertex: return ExecutionModel.Vertex;
case GalShaderType.TessControl: return ExecutionModel.TessellationControl;
case GalShaderType.TessEvaluation: return ExecutionModel.TessellationEvaluation;
case GalShaderType.Geometry: return ExecutionModel.Geometry;
case GalShaderType.Fragment: return ExecutionModel.Fragment;
default:
throw new InvalidOperationException();
}
}
private Instruction AllocConstant(Instruction Type, Operand Value)
{
Instruction NewConstant = new OpConstant(Type, Value);
foreach (Instruction Constant in TypesConstants)
{
if (Constant.Equals(NewConstant))
{
return Constant;
}
}
TypesConstants.Add(NewConstant);
return NewConstant;
}
private Instruction GetVariableValue(Instruction ResultType, string Name, bool Pointer)
{
SpirvVariable Variable = GetVariable(Name);
if (Pointer)
{
return Variable.Id;
}
Instruction Value = new OpLoad(ResultType, Variable.Id);
Code.Add(Value);
return Value;
}
private Instruction GetVariableValue(Instruction ResultType, string Name, int Index, bool Pointer)
{
Instruction InstIndex = AllocConstant(TypeInt, new LiteralNumber(Index));
return GetVariableValue(ResultType, Name, InstIndex, Pointer);
}
private Instruction GetVariableValue(Instruction ResultType, string Name, Instruction Index, bool Pointer)
{
OpTypePointer AccessTypePointer;
OpAccessChain Component;
switch (Name)
{
case "gl_Position":
Instruction PositionIndex = AllocConstant(TypeInt, new LiteralNumber((int)0));
AccessTypePointer = (OpTypePointer)AllocType(new OpTypePointer(StorageClass.Output, ResultType));
Component = new OpAccessChain(AccessTypePointer, PerVertexVar, PositionIndex, Index);
break;
default:
SpirvVariable Base = GetVariable(Name);
AccessTypePointer = (OpTypePointer)AllocType(new OpTypePointer(Base.Storage, ResultType));
Component = new OpAccessChain(AccessTypePointer, Base.Id, Index);
break;
}
Code.Add(Component);
if (Pointer)
{
return Component;
}
else
{
Instruction Value = new OpLoad(ResultType, Component);
Code.Add(Value);
return Value;
}
}
private SpirvVariable GetVariable(string Name)
{
foreach (SpirvVariable Variable in Variables)
{
if (Variable.Name == Name)
{
return Variable;
}
}
throw new InvalidOperationException($"Variable {Name} not declared");
}
private Instruction AllocVariable(
Instruction Type,
StorageClass StorageClass,
string Name,
int Location = -1)
{
Instruction InstVariable = new OpVariable(Type, StorageClass);
VarsDeclaration.Add(InstVariable);
Names.Add(new OpName(InstVariable, Name));
if (Location >= 0)
{
Operand Literal = new LiteralNumber(Location);
Decorates.Add(new OpDecorate(InstVariable, Decoration.Location, Literal));
}
SpirvVariable Variable = new SpirvVariable();
Variable.Id = InstVariable;
Variable.Storage = StorageClass;
Variable.Name = Name;
Variable.Location = Location;
Variables.Add(Variable);
return InstVariable;
}
private Instruction AllocLocatedVariable(
Instruction Type,
StorageClass StorageClass,
string Name,
LocationAllocator Allocator)
{
return AllocVariable(Type, StorageClass, Name, Allocator());
}
private Instruction AllocType(Instruction NewType)
{
foreach (Instruction StoredType in TypesConstants)
{
if (StoredType.Equals(NewType))
{
return StoredType;
}
}
TypesConstants.Add(NewType);
return NewType;
}
private int AllocInAttributeLocation()
{
InAttributeCount += 1;
return InAttributeCount - 1;
}
private int AllocOutAttributeLocation()
{
OutAttributeCount += 1;
return OutAttributeCount - 1;
}
private int AllocUniformLocation()
{
UniformCount += 1;
return UniformCount - 1;
}
}
}