mirror of
https://github.com/GreemDev/Ryujinx
synced 2025-01-12 05:51:25 +01:00
49f970d5bd
* Add support for CAL and RET shader instructions * Remove unused stuff * Fix a bug that could cause the wrong values to be passed to a function * Avoid repopulating function id dictionary every time * PR feedback * Fix vertex shader A/B merge
151 lines
No EOL
4.4 KiB
C#
151 lines
No EOL
4.4 KiB
C#
using Ryujinx.Graphics.Shader.IntermediateRepresentation;
|
|
using System.Collections.Generic;
|
|
|
|
namespace Ryujinx.Graphics.Shader.Translation
|
|
{
|
|
class ControlFlowGraph
|
|
{
|
|
public BasicBlock[] Blocks { get; }
|
|
public BasicBlock[] PostOrderBlocks { get; }
|
|
public int[] PostOrderMap { get; }
|
|
|
|
public ControlFlowGraph(BasicBlock[] blocks)
|
|
{
|
|
Blocks = blocks;
|
|
|
|
HashSet<BasicBlock> visited = new HashSet<BasicBlock>();
|
|
|
|
Stack<BasicBlock> blockStack = new Stack<BasicBlock>();
|
|
|
|
List<BasicBlock> postOrderBlocks = new List<BasicBlock>(blocks.Length);
|
|
|
|
PostOrderMap = new int[blocks.Length];
|
|
|
|
visited.Add(blocks[0]);
|
|
|
|
blockStack.Push(blocks[0]);
|
|
|
|
while (blockStack.TryPop(out BasicBlock block))
|
|
{
|
|
if (block.Next != null && visited.Add(block.Next))
|
|
{
|
|
blockStack.Push(block);
|
|
blockStack.Push(block.Next);
|
|
}
|
|
else if (block.Branch != null && visited.Add(block.Branch))
|
|
{
|
|
blockStack.Push(block);
|
|
blockStack.Push(block.Branch);
|
|
}
|
|
else
|
|
{
|
|
PostOrderMap[block.Index] = postOrderBlocks.Count;
|
|
|
|
postOrderBlocks.Add(block);
|
|
}
|
|
}
|
|
|
|
PostOrderBlocks = postOrderBlocks.ToArray();
|
|
}
|
|
|
|
public static ControlFlowGraph Create(Operation[] operations)
|
|
{
|
|
Dictionary<Operand, BasicBlock> labels = new Dictionary<Operand, BasicBlock>();
|
|
|
|
List<BasicBlock> blocks = new List<BasicBlock>();
|
|
|
|
BasicBlock currentBlock = null;
|
|
|
|
void NextBlock(BasicBlock nextBlock)
|
|
{
|
|
if (currentBlock != null && !EndsWithUnconditionalInst(currentBlock.GetLastOp()))
|
|
{
|
|
currentBlock.Next = nextBlock;
|
|
}
|
|
|
|
currentBlock = nextBlock;
|
|
}
|
|
|
|
void NewNextBlock()
|
|
{
|
|
BasicBlock block = new BasicBlock(blocks.Count);
|
|
|
|
blocks.Add(block);
|
|
|
|
NextBlock(block);
|
|
}
|
|
|
|
bool needsNewBlock = true;
|
|
|
|
for (int index = 0; index < operations.Length; index++)
|
|
{
|
|
Operation operation = operations[index];
|
|
|
|
if (operation.Inst == Instruction.MarkLabel)
|
|
{
|
|
Operand label = operation.Dest;
|
|
|
|
if (labels.TryGetValue(label, out BasicBlock nextBlock))
|
|
{
|
|
nextBlock.Index = blocks.Count;
|
|
|
|
blocks.Add(nextBlock);
|
|
|
|
NextBlock(nextBlock);
|
|
}
|
|
else
|
|
{
|
|
NewNextBlock();
|
|
|
|
labels.Add(label, currentBlock);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (needsNewBlock)
|
|
{
|
|
NewNextBlock();
|
|
}
|
|
|
|
currentBlock.Operations.AddLast(operation);
|
|
}
|
|
|
|
needsNewBlock = operation.Inst == Instruction.Branch ||
|
|
operation.Inst == Instruction.BranchIfTrue ||
|
|
operation.Inst == Instruction.BranchIfFalse;
|
|
|
|
if (needsNewBlock)
|
|
{
|
|
Operand label = operation.Dest;
|
|
|
|
if (!labels.TryGetValue(label, out BasicBlock branchBlock))
|
|
{
|
|
branchBlock = new BasicBlock();
|
|
|
|
labels.Add(label, branchBlock);
|
|
}
|
|
|
|
currentBlock.Branch = branchBlock;
|
|
}
|
|
}
|
|
|
|
return new ControlFlowGraph(blocks.ToArray());
|
|
}
|
|
|
|
private static bool EndsWithUnconditionalInst(INode node)
|
|
{
|
|
if (node is Operation operation)
|
|
{
|
|
switch (operation.Inst)
|
|
{
|
|
case Instruction.Branch:
|
|
case Instruction.Discard:
|
|
case Instruction.Return:
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
}
|
|
} |