mirror of
				https://github.com/Ryujinx/Ryujinx.git
				synced 2025-10-25 07:22:25 -07:00 
			
		
		
		
	* Replace LinkedList by IntrusiveList to avoid allocations on JIT * Fix wrong replacements
		
			
				
	
	
		
			84 lines
		
	
	
		
			2.1 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			84 lines
		
	
	
		
			2.1 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| using System.Collections.Generic;
 | |
| 
 | |
| namespace ARMeilleure.IntermediateRepresentation
 | |
| {
 | |
|     class BasicBlock : IIntrusiveListNode<BasicBlock>
 | |
|     {
 | |
|         public int Index { get; set; }
 | |
| 
 | |
|         public BasicBlock ListPrevious { get; set; }
 | |
|         public BasicBlock ListNext { get; set; }
 | |
| 
 | |
|         public IntrusiveList<Node> Operations { get; }
 | |
| 
 | |
|         private BasicBlock _next;
 | |
|         private BasicBlock _branch;
 | |
| 
 | |
|         public BasicBlock Next
 | |
|         {
 | |
|             get => _next;
 | |
|             set => _next = AddSuccessor(_next, value);
 | |
|         }
 | |
| 
 | |
|         public BasicBlock Branch
 | |
|         {
 | |
|             get => _branch;
 | |
|             set => _branch = AddSuccessor(_branch, value);
 | |
|         }
 | |
| 
 | |
|         public List<BasicBlock> Predecessors { get; }
 | |
| 
 | |
|         public HashSet<BasicBlock> DominanceFrontiers { get; }
 | |
| 
 | |
|         public BasicBlock ImmediateDominator { get; set; }
 | |
| 
 | |
|         public BasicBlock()
 | |
|         {
 | |
|             Operations = new IntrusiveList<Node>();
 | |
| 
 | |
|             Predecessors = new List<BasicBlock>();
 | |
| 
 | |
|             DominanceFrontiers = new HashSet<BasicBlock>();
 | |
| 
 | |
|             Index = -1;
 | |
|         }
 | |
| 
 | |
|         public BasicBlock(int index) : this()
 | |
|         {
 | |
|             Index = index;
 | |
|         }
 | |
| 
 | |
|         private BasicBlock AddSuccessor(BasicBlock oldBlock, BasicBlock newBlock)
 | |
|         {
 | |
|             oldBlock?.Predecessors.Remove(this);
 | |
|             newBlock?.Predecessors.Add(this);
 | |
| 
 | |
|             return newBlock;
 | |
|         }
 | |
| 
 | |
|         public void Append(Node node)
 | |
|         {
 | |
|             // If the branch block is not null, then the list of operations
 | |
|             // should end with a branch instruction. We insert the new operation
 | |
|             // before this branch.
 | |
|             if (_branch != null || (Operations.Last != null && IsLeafBlock()))
 | |
|             {
 | |
|                 Operations.AddBefore(Operations.Last, node);
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 Operations.AddLast(node);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         private bool IsLeafBlock()
 | |
|         {
 | |
|             return _branch == null && _next == null;
 | |
|         }
 | |
| 
 | |
|         public Node GetLastOp()
 | |
|         {
 | |
|             return Operations.Last;
 | |
|         }
 | |
|     }
 | |
| } |