2018-04-08 14:17:35 -05:00
|
|
|
namespace Ryujinx.Graphics.Gal.Shader
|
|
|
|
{
|
|
|
|
static class ShaderDecoder
|
|
|
|
{
|
2018-05-17 13:25:42 -05:00
|
|
|
private const bool AddDbgComments = true;
|
|
|
|
|
2018-05-22 20:43:31 -05:00
|
|
|
public static ShaderIrBlock DecodeBasicBlock(IGalMemory Memory, long Position)
|
2018-04-08 14:17:35 -05:00
|
|
|
{
|
|
|
|
ShaderIrBlock Block = new ShaderIrBlock();
|
|
|
|
|
2018-05-22 20:43:31 -05:00
|
|
|
while (true)
|
2018-04-08 14:17:35 -05:00
|
|
|
{
|
2018-05-22 20:43:31 -05:00
|
|
|
Block.Position = Position;
|
2018-05-17 13:25:42 -05:00
|
|
|
|
2018-05-22 20:43:31 -05:00
|
|
|
Block.MarkLabel(Position);
|
2018-05-17 13:25:42 -05:00
|
|
|
|
|
|
|
//Ignore scheduling instructions, which are written every 32 bytes.
|
2018-05-22 20:43:31 -05:00
|
|
|
if ((Position & 0x1f) == 0)
|
2018-04-10 14:50:32 -05:00
|
|
|
{
|
2018-05-22 20:43:31 -05:00
|
|
|
Position += 8;
|
2018-04-10 14:50:32 -05:00
|
|
|
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2018-05-22 20:43:31 -05:00
|
|
|
uint Word0 = (uint)Memory.ReadInt32(Position + 0);
|
|
|
|
uint Word1 = (uint)Memory.ReadInt32(Position + 4);
|
|
|
|
|
|
|
|
Position += 8;
|
2018-04-08 14:17:35 -05:00
|
|
|
|
|
|
|
long OpCode = Word0 | (long)Word1 << 32;
|
|
|
|
|
|
|
|
ShaderDecodeFunc Decode = ShaderOpCodeTable.GetDecoder(OpCode);
|
|
|
|
|
2018-05-17 13:25:42 -05:00
|
|
|
if (AddDbgComments)
|
|
|
|
{
|
2018-05-22 20:43:31 -05:00
|
|
|
string DbgOpCode = $"0x{Position:x16}: 0x{OpCode:x16} ";
|
2018-05-17 13:25:42 -05:00
|
|
|
|
|
|
|
Block.AddNode(new ShaderIrCmnt(DbgOpCode + (Decode?.Method.Name ?? "???")));
|
|
|
|
}
|
|
|
|
|
2018-04-08 14:17:35 -05:00
|
|
|
if (Decode == null)
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
Decode(Block, OpCode);
|
|
|
|
|
2018-05-17 13:25:42 -05:00
|
|
|
if (Block.GetLastNode() is ShaderIrOp Op && Op.Inst == ShaderIrInst.Exit)
|
2018-04-08 14:17:35 -05:00
|
|
|
{
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return Block;
|
|
|
|
}
|
|
|
|
|
|
|
|
private static bool IsFlowChange(ShaderIrInst Inst)
|
|
|
|
{
|
|
|
|
return Inst == ShaderIrInst.Exit;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|