Replace LinkedList by IntrusiveList to avoid allocations on JIT (#931)
* Replace LinkedList by IntrusiveList to avoid allocations on JIT * Fix wrong replacements
This commit is contained in:
		
							parent
							
								
									e9a37ca6a8
								
							
						
					
					
						commit
						e5f78fb1d4
					
				
					 18 changed files with 365 additions and 198 deletions
				
			
		|  | @ -16,17 +16,17 @@ namespace ARMeilleure.CodeGen.Optimizations | |||
|             { | ||||
|                 modified = false; | ||||
| 
 | ||||
|                 foreach (BasicBlock block in cfg.Blocks) | ||||
|                 for (BasicBlock block = cfg.Blocks.First; block != null; block = block.ListNext) | ||||
|                 { | ||||
|                     LinkedListNode<Node> node = block.Operations.First; | ||||
|                     Node node = block.Operations.First; | ||||
| 
 | ||||
|                     while (node != null) | ||||
|                     { | ||||
|                         LinkedListNode<Node> nextNode = node.Next; | ||||
|                         Node nextNode = node.ListNext; | ||||
| 
 | ||||
|                         bool isUnused = IsUnused(node.Value); | ||||
|                         bool isUnused = IsUnused(node); | ||||
| 
 | ||||
|                         if (!(node.Value is Operation operation) || isUnused) | ||||
|                         if (!(node is Operation operation) || isUnused) | ||||
|                         { | ||||
|                             if (isUnused) | ||||
|                             { | ||||
|  | @ -80,13 +80,11 @@ namespace ARMeilleure.CodeGen.Optimizations | |||
|             } | ||||
|         } | ||||
| 
 | ||||
|         private static void RemoveNode(BasicBlock block, LinkedListNode<Node> llNode) | ||||
|         private static void RemoveNode(BasicBlock block, Node node) | ||||
|         { | ||||
|             // Remove a node from the nodes list, and also remove itself | ||||
|             // from all the use lists on the operands that this node uses. | ||||
|             block.Operations.Remove(llNode); | ||||
| 
 | ||||
|             Node node = llNode.Value; | ||||
|             block.Operations.Remove(node); | ||||
| 
 | ||||
|             for (int index = 0; index < node.SourcesCount; index++) | ||||
|             { | ||||
|  |  | |||
|  | @ -95,7 +95,7 @@ namespace ARMeilleure.CodeGen.RegisterAllocators | |||
| 
 | ||||
|                 bool hasCall = false; | ||||
| 
 | ||||
|                 foreach (Node node in block.Operations) | ||||
|                 for (Node node = block.Operations.First; node != null; node = node.ListNext) | ||||
|                 { | ||||
|                     if (node is Operation operation && operation.Instruction == Instruction.Call) | ||||
|                     { | ||||
|  | @ -176,10 +176,8 @@ namespace ARMeilleure.CodeGen.RegisterAllocators | |||
|                 intLocalFreeRegisters &= ~(intSpillTempRegisters | intCallerSavedRegisters); | ||||
|                 vecLocalFreeRegisters &= ~(vecSpillTempRegisters | vecCallerSavedRegisters); | ||||
| 
 | ||||
|                 for (LinkedListNode<Node> llNode = block.Operations.First; llNode != null; llNode = llNode.Next) | ||||
|                 for (Node node = block.Operations.First; node != null; node = node.ListNext) | ||||
|                 { | ||||
|                     Node node = llNode.Value; | ||||
| 
 | ||||
|                     int intLocalUse = 0; | ||||
|                     int vecLocalUse = 0; | ||||
| 
 | ||||
|  | @ -232,7 +230,7 @@ namespace ARMeilleure.CodeGen.RegisterAllocators | |||
| 
 | ||||
|                             Operation fillOp = new Operation(Instruction.Fill, temp, Const(info.SpillOffset)); | ||||
| 
 | ||||
|                             block.Operations.AddBefore(llNode, fillOp); | ||||
|                             block.Operations.AddBefore(node, fillOp); | ||||
|                         } | ||||
|                     } | ||||
| 
 | ||||
|  | @ -306,7 +304,9 @@ namespace ARMeilleure.CodeGen.RegisterAllocators | |||
| 
 | ||||
|                             Operation spillOp = new Operation(Instruction.Spill, null, Const(info.SpillOffset), temp); | ||||
| 
 | ||||
|                             llNode = block.Operations.AddAfter(llNode, spillOp); | ||||
|                             block.Operations.AddAfter(node, spillOp); | ||||
| 
 | ||||
|                             node = spillOp; | ||||
|                         } | ||||
|                     } | ||||
| 
 | ||||
|  |  | |||
|  | @ -28,7 +28,7 @@ namespace ARMeilleure.CodeGen.RegisterAllocators | |||
| 
 | ||||
|         private LiveInterval[] _parentIntervals; | ||||
| 
 | ||||
|         private List<LinkedListNode<Node>> _operationNodes; | ||||
|         private List<(IntrusiveList<Node>, Node)> _operationNodes; | ||||
| 
 | ||||
|         private int _operationsCount; | ||||
| 
 | ||||
|  | @ -583,15 +583,19 @@ namespace ARMeilleure.CodeGen.RegisterAllocators | |||
| 
 | ||||
|                 int splitPosition = kv.Key; | ||||
| 
 | ||||
|                 LinkedListNode<Node> node = GetOperationNode(splitPosition); | ||||
|                 (IntrusiveList<Node> nodes, Node node) = GetOperationNode(splitPosition); | ||||
| 
 | ||||
|                 Operation[] sequence = copyResolver.Sequence(); | ||||
| 
 | ||||
|                 node = node.List.AddBefore(node, sequence[0]); | ||||
|                 nodes.AddBefore(node, sequence[0]); | ||||
| 
 | ||||
|                 node = sequence[0]; | ||||
| 
 | ||||
|                 for (int index = 1; index < sequence.Length; index++) | ||||
|                 { | ||||
|                     node = node.List.AddAfter(node, sequence[index]); | ||||
|                     nodes.AddAfter(node, sequence[index]); | ||||
| 
 | ||||
|                     node = sequence[index]; | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | @ -605,10 +609,8 @@ namespace ARMeilleure.CodeGen.RegisterAllocators | |||
|                 return block.Index >= blocksCount; | ||||
|             } | ||||
| 
 | ||||
|             for (LinkedListNode<BasicBlock> node = cfg.Blocks.First; node != null; node = node.Next) | ||||
|             for (BasicBlock block = cfg.Blocks.First; block != null; block = block.ListNext) | ||||
|             { | ||||
|                 BasicBlock block = node.Value; | ||||
| 
 | ||||
|                 if (IsSplitEdgeBlock(block)) | ||||
|                 { | ||||
|                     continue; | ||||
|  | @ -666,13 +668,17 @@ namespace ARMeilleure.CodeGen.RegisterAllocators | |||
|                     } | ||||
|                     else if (successor.Predecessors.Count == 1) | ||||
|                     { | ||||
|                         LinkedListNode<Node> prependNode = successor.Operations.AddFirst(sequence[0]); | ||||
|                         successor.Operations.AddFirst(sequence[0]); | ||||
| 
 | ||||
|                         Node prependNode = sequence[0]; | ||||
| 
 | ||||
|                         for (int index = 1; index < sequence.Length; index++) | ||||
|                         { | ||||
|                             Operation operation = sequence[index]; | ||||
| 
 | ||||
|                             prependNode = successor.Operations.AddAfter(prependNode, operation); | ||||
|                             successor.Operations.AddAfter(prependNode, operation); | ||||
| 
 | ||||
|                             prependNode = operation; | ||||
|                         } | ||||
|                     } | ||||
|                     else | ||||
|  | @ -695,7 +701,7 @@ namespace ARMeilleure.CodeGen.RegisterAllocators | |||
| 
 | ||||
|             foreach (int usePosition in current.UsePositions()) | ||||
|             { | ||||
|                 Node operation = GetOperationNode(usePosition).Value; | ||||
|                 (_, Node operation) = GetOperationNode(usePosition); | ||||
| 
 | ||||
|                 for (int index = 0; index < operation.SourcesCount; index++) | ||||
|                 { | ||||
|  | @ -729,14 +735,14 @@ namespace ARMeilleure.CodeGen.RegisterAllocators | |||
|                 interval.Local.Type); | ||||
|         } | ||||
| 
 | ||||
|         private LinkedListNode<Node> GetOperationNode(int position) | ||||
|         private (IntrusiveList<Node>, Node) GetOperationNode(int position) | ||||
|         { | ||||
|             return _operationNodes[position / InstructionGap]; | ||||
|         } | ||||
| 
 | ||||
|         private void NumberLocals(ControlFlowGraph cfg) | ||||
|         { | ||||
|             _operationNodes = new List<LinkedListNode<Node>>(); | ||||
|             _operationNodes = new List<(IntrusiveList<Node>, Node)>(); | ||||
| 
 | ||||
|             _intervals = new List<LiveInterval>(); | ||||
| 
 | ||||
|  | @ -754,13 +760,11 @@ namespace ARMeilleure.CodeGen.RegisterAllocators | |||
|             { | ||||
|                 BasicBlock block = cfg.PostOrderBlocks[index]; | ||||
| 
 | ||||
|                 for (LinkedListNode<Node> node = block.Operations.First; node != null; node = node.Next) | ||||
|                 for (Node node = block.Operations.First; node != null; node = node.ListNext) | ||||
|                 { | ||||
|                     _operationNodes.Add(node); | ||||
|                     _operationNodes.Add((block.Operations, node)); | ||||
| 
 | ||||
|                     Node operation = node.Value; | ||||
| 
 | ||||
|                     foreach (Operand dest in Destinations(operation)) | ||||
|                     foreach (Operand dest in Destinations(node)) | ||||
|                     { | ||||
|                         if (dest.Kind == OperandKind.LocalVariable && visited.Add(dest)) | ||||
|                         { | ||||
|  | @ -776,7 +780,7 @@ namespace ARMeilleure.CodeGen.RegisterAllocators | |||
|                 if (block.Operations.Count == 0) | ||||
|                 { | ||||
|                     // Pretend we have a dummy instruction on the empty block. | ||||
|                     _operationNodes.Add(null); | ||||
|                     _operationNodes.Add((null, null)); | ||||
| 
 | ||||
|                     _operationsCount += InstructionGap; | ||||
|                 } | ||||
|  | @ -795,12 +799,12 @@ namespace ARMeilleure.CodeGen.RegisterAllocators | |||
|             BitMap[] blkLiveKill = new BitMap[cfg.Blocks.Count]; | ||||
| 
 | ||||
|             // Compute local live sets. | ||||
|             foreach (BasicBlock block in cfg.Blocks) | ||||
|             for (BasicBlock block = cfg.Blocks.First; block != null; block = block.ListNext) | ||||
|             { | ||||
|                 BitMap liveGen  = new BitMap(mapSize); | ||||
|                 BitMap liveKill = new BitMap(mapSize); | ||||
| 
 | ||||
|                 foreach (Node node in block.Operations) | ||||
|                 for (Node node = block.Operations.First; node != null; node = node.ListNext) | ||||
|                 { | ||||
|                     foreach (Operand source in Sources(node)) | ||||
|                     { | ||||
|  | @ -979,13 +983,13 @@ namespace ARMeilleure.CodeGen.RegisterAllocators | |||
| 
 | ||||
|         private static IEnumerable<Node> BottomOperations(BasicBlock block) | ||||
|         { | ||||
|             LinkedListNode<Node> node = block.Operations.Last; | ||||
|             Node node = block.Operations.Last; | ||||
| 
 | ||||
|             while (node != null && !(node.Value is PhiNode)) | ||||
|             while (node != null && !(node is PhiNode)) | ||||
|             { | ||||
|                 yield return node.Value; | ||||
|                 yield return node; | ||||
| 
 | ||||
|                 node = node.Previous; | ||||
|                 node = node.ListPrevious; | ||||
|             } | ||||
|         } | ||||
| 
 | ||||
|  |  | |||
|  | @ -157,11 +157,11 @@ namespace ARMeilleure.CodeGen.X86 | |||
| 
 | ||||
|                 UnwindInfo unwindInfo = WritePrologue(context); | ||||
| 
 | ||||
|                 foreach (BasicBlock block in cfg.Blocks) | ||||
|                 for (BasicBlock block = cfg.Blocks.First; block != null; block = block.ListNext) | ||||
|                 { | ||||
|                     context.EnterBlock(block); | ||||
| 
 | ||||
|                     foreach (Node node in block.Operations) | ||||
|                     for (Node node = block.Operations.First; node != null; node = node.ListNext) | ||||
|                     { | ||||
|                         if (node is Operation operation) | ||||
|                         { | ||||
|  |  | |||
|  | @ -8,8 +8,6 @@ using static ARMeilleure.IntermediateRepresentation.OperandHelper; | |||
| 
 | ||||
| namespace ARMeilleure.CodeGen.X86 | ||||
| { | ||||
|     using LLNode = LinkedListNode<Node>; | ||||
| 
 | ||||
|     static class PreAllocator | ||||
|     { | ||||
|         public static void RunPass(CompilerContext cctx, StackAllocator stackAlloc, out int maxCallArgs) | ||||
|  | @ -20,24 +18,24 @@ namespace ARMeilleure.CodeGen.X86 | |||
| 
 | ||||
|             Operand[] preservedArgs = new Operand[CallingConvention.GetArgumentsOnRegsCount()]; | ||||
| 
 | ||||
|             foreach (BasicBlock block in cctx.Cfg.Blocks) | ||||
|             for (BasicBlock block = cctx.Cfg.Blocks.First; block != null; block = block.ListNext) | ||||
|             { | ||||
|                 LLNode nextNode; | ||||
|                 Node nextNode; | ||||
| 
 | ||||
|                 for (LLNode node = block.Operations.First; node != null; node = nextNode) | ||||
|                 for (Node node = block.Operations.First; node != null; node = nextNode) | ||||
|                 { | ||||
|                     nextNode = node.Next; | ||||
|                     nextNode = node.ListNext; | ||||
| 
 | ||||
|                     if (!(node.Value is Operation operation)) | ||||
|                     if (!(node is Operation operation)) | ||||
|                     { | ||||
|                         continue; | ||||
|                     } | ||||
| 
 | ||||
|                     HandleConstantCopy(node, operation); | ||||
|                     HandleConstantCopy(block.Operations, node, operation); | ||||
| 
 | ||||
|                     HandleSameDestSrc1Copy(node, operation); | ||||
|                     HandleSameDestSrc1Copy(block.Operations, node, operation); | ||||
| 
 | ||||
|                     HandleFixedRegisterCopy(node, operation); | ||||
|                     HandleFixedRegisterCopy(block.Operations, node, operation); | ||||
| 
 | ||||
|                     switch (operation.Instruction) | ||||
|                     { | ||||
|  | @ -62,51 +60,51 @@ namespace ARMeilleure.CodeGen.X86 | |||
|                             // being called, as mandated by the ABI. | ||||
|                             if (callConv == CallConvName.Windows) | ||||
|                             { | ||||
|                                 node = HandleCallWindowsAbi(stackAlloc, node, operation); | ||||
|                                 node = HandleCallWindowsAbi(block.Operations, stackAlloc, node, operation); | ||||
|                             } | ||||
|                             else /* if (callConv == CallConvName.SystemV) */ | ||||
|                             { | ||||
|                                 node = HandleCallSystemVAbi(node, operation); | ||||
|                                 node = HandleCallSystemVAbi(block.Operations, node, operation); | ||||
|                             } | ||||
|                             break; | ||||
| 
 | ||||
|                         case Instruction.ConvertToFPUI: | ||||
|                             HandleConvertToFPUI(node, operation); | ||||
|                             HandleConvertToFPUI(block.Operations, node, operation); | ||||
|                             break; | ||||
| 
 | ||||
|                         case Instruction.LoadArgument: | ||||
|                             if (callConv == CallConvName.Windows) | ||||
|                             { | ||||
|                                 HandleLoadArgumentWindowsAbi(cctx, node, preservedArgs, operation); | ||||
|                                 HandleLoadArgumentWindowsAbi(cctx, block.Operations, node, preservedArgs, operation); | ||||
|                             } | ||||
|                             else /* if (callConv == CallConvName.SystemV) */ | ||||
|                             { | ||||
|                                 HandleLoadArgumentSystemVAbi(cctx, node, preservedArgs, operation); | ||||
|                                 HandleLoadArgumentSystemVAbi(cctx, block.Operations, node, preservedArgs, operation); | ||||
|                             } | ||||
|                             break; | ||||
| 
 | ||||
|                         case Instruction.Negate: | ||||
|                             if (!operation.GetSource(0).Type.IsInteger()) | ||||
|                             { | ||||
|                                 node = HandleNegate(node, operation); | ||||
|                                 node = HandleNegate(block.Operations, node, operation); | ||||
|                             } | ||||
|                             break; | ||||
| 
 | ||||
|                         case Instruction.Return: | ||||
|                             if (callConv == CallConvName.Windows) | ||||
|                             { | ||||
|                                 HandleReturnWindowsAbi(cctx, node, preservedArgs, operation); | ||||
|                                 HandleReturnWindowsAbi(cctx, block.Operations, node, preservedArgs, operation); | ||||
|                             } | ||||
|                             else /* if (callConv == CallConvName.SystemV) */ | ||||
|                             { | ||||
|                                 HandleReturnSystemVAbi(node, operation); | ||||
|                                 HandleReturnSystemVAbi(block.Operations, node, operation); | ||||
|                             } | ||||
|                             break; | ||||
| 
 | ||||
|                         case Instruction.VectorInsert8: | ||||
|                             if (!HardwareCapabilities.SupportsSse41) | ||||
|                             { | ||||
|                                 node = HandleVectorInsert8(node, operation); | ||||
|                                 node = HandleVectorInsert8(block.Operations, node, operation); | ||||
|                             } | ||||
|                             break; | ||||
|                     } | ||||
|  | @ -114,7 +112,7 @@ namespace ARMeilleure.CodeGen.X86 | |||
|             } | ||||
|         } | ||||
| 
 | ||||
|         private static void HandleConstantCopy(LLNode node, Operation operation) | ||||
|         private static void HandleConstantCopy(IntrusiveList<Node> nodes, Node node, Operation operation) | ||||
|         { | ||||
|             if (operation.SourcesCount == 0 || IsIntrinsic(operation.Instruction)) | ||||
|             { | ||||
|  | @ -135,7 +133,7 @@ namespace ARMeilleure.CodeGen.X86 | |||
|                     // - Insert a copy with the constant value (as integer) to a GPR. | ||||
|                     // - Insert a copy from the GPR to a XMM register. | ||||
|                     // - Replace the constant use with the XMM register. | ||||
|                     src1 = AddXmmCopy(node, src1); | ||||
|                     src1 = AddXmmCopy(nodes, node, src1); | ||||
| 
 | ||||
|                     operation.SetSource(0, src1); | ||||
|                 } | ||||
|  | @ -164,7 +162,7 @@ namespace ARMeilleure.CodeGen.X86 | |||
| 
 | ||||
|                     if (src1.Kind == OperandKind.Constant) | ||||
|                     { | ||||
|                         src1 = AddCopy(node, src1); | ||||
|                         src1 = AddCopy(nodes, node, src1); | ||||
| 
 | ||||
|                         operation.SetSource(0, src1); | ||||
|                     } | ||||
|  | @ -182,25 +180,23 @@ namespace ARMeilleure.CodeGen.X86 | |||
|             { | ||||
|                 if (!src2.Type.IsInteger()) | ||||
|                 { | ||||
|                     src2 = AddXmmCopy(node, src2); | ||||
|                     src2 = AddXmmCopy(nodes, node, src2); | ||||
| 
 | ||||
|                     operation.SetSource(1, src2); | ||||
|                 } | ||||
|                 else if (!HasConstSrc2(inst) || IsLongConst(src2)) | ||||
|                 { | ||||
|                     src2 = AddCopy(node, src2); | ||||
|                     src2 = AddCopy(nodes, node, src2); | ||||
| 
 | ||||
|                     operation.SetSource(1, src2); | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
| 
 | ||||
|         private static LLNode HandleFixedRegisterCopy(LLNode node, Operation operation) | ||||
|         private static Node HandleFixedRegisterCopy(IntrusiveList<Node> nodes, Node node, Operation operation) | ||||
|         { | ||||
|             Operand dest = operation.Destination; | ||||
| 
 | ||||
|             LinkedList<Node> nodes = node.List; | ||||
| 
 | ||||
|             switch (operation.Instruction) | ||||
|             { | ||||
|                 case Instruction.CompareAndSwap128: | ||||
|  | @ -359,7 +355,7 @@ namespace ARMeilleure.CodeGen.X86 | |||
|             return node; | ||||
|         } | ||||
| 
 | ||||
|         private static LLNode HandleSameDestSrc1Copy(LLNode node, Operation operation) | ||||
|         private static Node HandleSameDestSrc1Copy(IntrusiveList<Node> nodes, Node node, Operation operation) | ||||
|         { | ||||
|             if (operation.Destination == null || operation.SourcesCount == 0) | ||||
|             { | ||||
|  | @ -371,8 +367,6 @@ namespace ARMeilleure.CodeGen.X86 | |||
|             Operand dest = operation.Destination; | ||||
|             Operand src1 = operation.GetSource(0); | ||||
| 
 | ||||
|             LinkedList<Node> nodes = node.List; | ||||
| 
 | ||||
|             // The multiply instruction (that maps to IMUL) is somewhat special, it has | ||||
|             // a three operand form where the second source is a immediate value. | ||||
|             bool threeOperandForm = inst == Instruction.Multiply && operation.GetSource(1).Kind == OperandKind.Constant; | ||||
|  | @ -441,7 +435,7 @@ namespace ARMeilleure.CodeGen.X86 | |||
|             return node; | ||||
|         } | ||||
| 
 | ||||
|         private static LLNode HandleConvertToFPUI(LLNode node, Operation operation) | ||||
|         private static Node HandleConvertToFPUI(IntrusiveList<Node> nodes, Node node, Operation operation) | ||||
|         { | ||||
|             // Unsigned integer to FP conversions are not supported on X86. | ||||
|             // We need to turn them into signed integer to FP conversions, and | ||||
|  | @ -451,9 +445,7 @@ namespace ARMeilleure.CodeGen.X86 | |||
| 
 | ||||
|             Debug.Assert(source.Type.IsInteger(), $"Invalid source type \"{source.Type}\"."); | ||||
| 
 | ||||
|             LinkedList<Node> nodes = node.List; | ||||
| 
 | ||||
|             LLNode currentNode = node; | ||||
|             Node currentNode = node; | ||||
| 
 | ||||
|             if (source.Type == OperandType.I32) | ||||
|             { | ||||
|  | @ -494,12 +486,12 @@ namespace ARMeilleure.CodeGen.X86 | |||
|                 node = nodes.AddAfter(node, new Operation(Instruction.Add, dest, dest, lsbF)); | ||||
|             } | ||||
| 
 | ||||
|             Delete(currentNode, operation); | ||||
|             Delete(nodes, currentNode, operation); | ||||
| 
 | ||||
|             return node; | ||||
|         } | ||||
| 
 | ||||
|         private static LLNode HandleNegate(LLNode node, Operation operation) | ||||
|         private static Node HandleNegate(IntrusiveList<Node> nodes, Node node, Operation operation) | ||||
|         { | ||||
|             // There's no SSE FP negate instruction, so we need to transform that into | ||||
|             // a XOR of the value to be negated with a mask with the highest bit set. | ||||
|  | @ -510,9 +502,7 @@ namespace ARMeilleure.CodeGen.X86 | |||
|             Debug.Assert(dest.Type == OperandType.FP32 || | ||||
|                          dest.Type == OperandType.FP64, $"Invalid destination type \"{dest.Type}\"."); | ||||
| 
 | ||||
|             LinkedList<Node> nodes = node.List; | ||||
| 
 | ||||
|             LLNode currentNode = node; | ||||
|             Node currentNode = node; | ||||
| 
 | ||||
|             Operand res = Local(dest.Type); | ||||
| 
 | ||||
|  | @ -531,12 +521,12 @@ namespace ARMeilleure.CodeGen.X86 | |||
| 
 | ||||
|             node = nodes.AddAfter(node, new Operation(Instruction.Copy, dest, res)); | ||||
| 
 | ||||
|             Delete(currentNode, operation); | ||||
|             Delete(nodes, currentNode, operation); | ||||
| 
 | ||||
|             return node; | ||||
|         } | ||||
| 
 | ||||
|         private static LLNode HandleVectorInsert8(LLNode node, Operation operation) | ||||
|         private static Node HandleVectorInsert8(IntrusiveList<Node> nodes, Node node, Operation operation) | ||||
|         { | ||||
|             // Handle vector insertion, when SSE 4.1 is not supported. | ||||
|             Operand dest = operation.Destination; | ||||
|  | @ -550,9 +540,7 @@ namespace ARMeilleure.CodeGen.X86 | |||
| 
 | ||||
|             Debug.Assert(index < 16); | ||||
| 
 | ||||
|             LinkedList<Node> nodes = node.List; | ||||
| 
 | ||||
|             LLNode currentNode = node; | ||||
|             Node currentNode = node; | ||||
| 
 | ||||
|             Operand temp1 = Local(OperandType.I32); | ||||
|             Operand temp2 = Local(OperandType.I32); | ||||
|  | @ -580,17 +568,15 @@ namespace ARMeilleure.CodeGen.X86 | |||
| 
 | ||||
|             node = nodes.AddAfter(node, vinsOp); | ||||
| 
 | ||||
|             Delete(currentNode, operation); | ||||
|             Delete(nodes, currentNode, operation); | ||||
| 
 | ||||
|             return node; | ||||
|         } | ||||
| 
 | ||||
|         private static LLNode HandleCallWindowsAbi(StackAllocator stackAlloc, LLNode node, Operation operation) | ||||
|         private static Node HandleCallWindowsAbi(IntrusiveList<Node> nodes, StackAllocator stackAlloc, Node node, Operation operation) | ||||
|         { | ||||
|             Operand dest = operation.Destination; | ||||
| 
 | ||||
|             LinkedList<Node> nodes = node.List; | ||||
| 
 | ||||
|             // Handle struct arguments. | ||||
|             int retArgs = 0; | ||||
| 
 | ||||
|  | @ -661,7 +647,7 @@ namespace ARMeilleure.CodeGen.X86 | |||
| 
 | ||||
|                     Operation storeOp = new Operation(Instruction.Store, null, stackAddr, source); | ||||
| 
 | ||||
|                     HandleConstantCopy(nodes.AddBefore(node, storeOp), storeOp); | ||||
|                     HandleConstantCopy(nodes, nodes.AddBefore(node, storeOp), storeOp); | ||||
| 
 | ||||
|                     operation.SetSource(index, stackAddr); | ||||
|                 } | ||||
|  | @ -687,7 +673,7 @@ namespace ARMeilleure.CodeGen.X86 | |||
| 
 | ||||
|                 Operation copyOp = new Operation(Instruction.Copy, argReg, source); | ||||
| 
 | ||||
|                 HandleConstantCopy(nodes.AddBefore(node, copyOp), copyOp); | ||||
|                 HandleConstantCopy(nodes, nodes.AddBefore(node, copyOp), copyOp); | ||||
| 
 | ||||
|                 sources[1 + retArgs + index] = argReg; | ||||
|             } | ||||
|  | @ -702,7 +688,7 @@ namespace ARMeilleure.CodeGen.X86 | |||
| 
 | ||||
|                 Operation spillOp = new Operation(Instruction.SpillArg, null, offset, source); | ||||
| 
 | ||||
|                 HandleConstantCopy(nodes.AddBefore(node, spillOp), spillOp); | ||||
|                 HandleConstantCopy(nodes, nodes.AddBefore(node, spillOp), spillOp); | ||||
|             } | ||||
| 
 | ||||
|             if (dest != null) | ||||
|  | @ -738,15 +724,14 @@ namespace ARMeilleure.CodeGen.X86 | |||
|             return node; | ||||
|         } | ||||
| 
 | ||||
|         private static LLNode HandleCallSystemVAbi(LLNode node, Operation operation) | ||||
|         private static Node HandleCallSystemVAbi(IntrusiveList<Node> nodes, Node node, Operation operation) | ||||
|         { | ||||
|             Operand dest = operation.Destination; | ||||
| 
 | ||||
|             LinkedList<Node> nodes = node.List; | ||||
| 
 | ||||
|             List<Operand> sources = new List<Operand>(); | ||||
| 
 | ||||
|             sources.Add(operation.GetSource(0)); | ||||
|             List<Operand> sources = new List<Operand> | ||||
|             { | ||||
|                 operation.GetSource(0) | ||||
|             }; | ||||
| 
 | ||||
|             int argsCount = operation.SourcesCount - 1; | ||||
| 
 | ||||
|  | @ -797,7 +782,7 @@ namespace ARMeilleure.CodeGen.X86 | |||
| 
 | ||||
|                     Operation copyOp = new Operation(Instruction.Copy, argReg, source); | ||||
| 
 | ||||
|                     HandleConstantCopy(nodes.AddBefore(node, copyOp), copyOp); | ||||
|                     HandleConstantCopy(nodes, nodes.AddBefore(node, copyOp), copyOp); | ||||
| 
 | ||||
|                     sources.Add(argReg); | ||||
|                 } | ||||
|  | @ -807,7 +792,7 @@ namespace ARMeilleure.CodeGen.X86 | |||
| 
 | ||||
|                     Operation spillOp = new Operation(Instruction.SpillArg, null, offset, source); | ||||
| 
 | ||||
|                     HandleConstantCopy(nodes.AddBefore(node, spillOp), spillOp); | ||||
|                     HandleConstantCopy(nodes, nodes.AddBefore(node, spillOp), spillOp); | ||||
| 
 | ||||
|                     stackOffset += source.Type.GetSizeInBytes(); | ||||
|                 } | ||||
|  | @ -846,7 +831,8 @@ namespace ARMeilleure.CodeGen.X86 | |||
| 
 | ||||
|         private static void HandleLoadArgumentWindowsAbi( | ||||
|             CompilerContext cctx, | ||||
|             LLNode node, | ||||
|             IntrusiveList<Node> nodes, | ||||
|             Node node, | ||||
|             Operand[] preservedArgs, | ||||
|             Operation operation) | ||||
|         { | ||||
|  | @ -896,9 +882,9 @@ namespace ARMeilleure.CodeGen.X86 | |||
|                     ? Instruction.Load | ||||
|                     : Instruction.Copy, dest, preservedArgs[index]); | ||||
| 
 | ||||
|                 node.List.AddBefore(node, argCopyOp); | ||||
|                 nodes.AddBefore(node, argCopyOp); | ||||
| 
 | ||||
|                 Delete(node, operation); | ||||
|                 Delete(nodes, node, operation); | ||||
|             } | ||||
|             else | ||||
|             { | ||||
|  | @ -908,7 +894,8 @@ namespace ARMeilleure.CodeGen.X86 | |||
| 
 | ||||
|         private static void HandleLoadArgumentSystemVAbi( | ||||
|             CompilerContext cctx, | ||||
|             LLNode node, | ||||
|             IntrusiveList<Node> nodes, | ||||
|             Node node, | ||||
|             Operand[] preservedArgs, | ||||
|             Operation operation) | ||||
|         { | ||||
|  | @ -994,9 +981,9 @@ namespace ARMeilleure.CodeGen.X86 | |||
| 
 | ||||
|                 Operation argCopyOp = new Operation(Instruction.Copy, dest, preservedArgs[index]); | ||||
| 
 | ||||
|                 node.List.AddBefore(node, argCopyOp); | ||||
|                 nodes.AddBefore(node, argCopyOp); | ||||
| 
 | ||||
|                 Delete(node, operation); | ||||
|                 Delete(nodes, node, operation); | ||||
|             } | ||||
|             else | ||||
|             { | ||||
|  | @ -1006,7 +993,8 @@ namespace ARMeilleure.CodeGen.X86 | |||
| 
 | ||||
|         private static void HandleReturnWindowsAbi( | ||||
|             CompilerContext cctx, | ||||
|             LLNode node, | ||||
|             IntrusiveList<Node> nodes, | ||||
|             Node node, | ||||
|             Operand[] preservedArgs, | ||||
|             Operation operation) | ||||
|         { | ||||
|  | @ -1049,19 +1037,19 @@ namespace ARMeilleure.CodeGen.X86 | |||
|             { | ||||
|                 Operation retStoreOp = new Operation(Instruction.Store, null, retReg, source); | ||||
| 
 | ||||
|                 node.List.AddBefore(node, retStoreOp); | ||||
|                 nodes.AddBefore(node, retStoreOp); | ||||
|             } | ||||
|             else | ||||
|             { | ||||
|                 Operation retCopyOp = new Operation(Instruction.Copy, retReg, source); | ||||
| 
 | ||||
|                 node.List.AddBefore(node, retCopyOp); | ||||
|                 nodes.AddBefore(node, retCopyOp); | ||||
|             } | ||||
| 
 | ||||
|             operation.SetSources(new Operand[0]); | ||||
|         } | ||||
| 
 | ||||
|         private static void HandleReturnSystemVAbi(LLNode node, Operation operation) | ||||
|         private static void HandleReturnSystemVAbi(IntrusiveList<Node> nodes, Node node, Operation operation) | ||||
|         { | ||||
|             if (operation.SourcesCount == 0) | ||||
|             { | ||||
|  | @ -1075,8 +1063,8 @@ namespace ARMeilleure.CodeGen.X86 | |||
|                 Operand retLReg = Gpr(CallingConvention.GetIntReturnRegister(),     OperandType.I64); | ||||
|                 Operand retHReg = Gpr(CallingConvention.GetIntReturnRegisterHigh(), OperandType.I64); | ||||
| 
 | ||||
|                 node.List.AddBefore(node, new Operation(Instruction.VectorExtract, retLReg, source, Const(0))); | ||||
|                 node.List.AddBefore(node, new Operation(Instruction.VectorExtract, retHReg, source, Const(1))); | ||||
|                 nodes.AddBefore(node, new Operation(Instruction.VectorExtract, retLReg, source, Const(0))); | ||||
|                 nodes.AddBefore(node, new Operation(Instruction.VectorExtract, retHReg, source, Const(1))); | ||||
|             } | ||||
|             else | ||||
|             { | ||||
|  | @ -1086,30 +1074,30 @@ namespace ARMeilleure.CodeGen.X86 | |||
| 
 | ||||
|                 Operation retCopyOp = new Operation(Instruction.Copy, retReg, source); | ||||
| 
 | ||||
|                 node.List.AddBefore(node, retCopyOp); | ||||
|                 nodes.AddBefore(node, retCopyOp); | ||||
|             } | ||||
|         } | ||||
| 
 | ||||
|         private static Operand AddXmmCopy(LLNode node, Operand source) | ||||
|         private static Operand AddXmmCopy(IntrusiveList<Node> nodes, Node node, Operand source) | ||||
|         { | ||||
|             Operand temp = Local(source.Type); | ||||
| 
 | ||||
|             Operand intConst = AddCopy(node, GetIntConst(source)); | ||||
|             Operand intConst = AddCopy(nodes, node, GetIntConst(source)); | ||||
| 
 | ||||
|             Operation copyOp = new Operation(Instruction.VectorCreateScalar, temp, intConst); | ||||
| 
 | ||||
|             node.List.AddBefore(node, copyOp); | ||||
|             nodes.AddBefore(node, copyOp); | ||||
| 
 | ||||
|             return temp; | ||||
|         } | ||||
| 
 | ||||
|         private static Operand AddCopy(LLNode node, Operand source) | ||||
|         private static Operand AddCopy(IntrusiveList<Node> nodes, Node node, Operand source) | ||||
|         { | ||||
|             Operand temp = Local(source.Type); | ||||
| 
 | ||||
|             Operation copyOp = new Operation(Instruction.Copy, temp, source); | ||||
| 
 | ||||
|             node.List.AddBefore(node, copyOp); | ||||
|             nodes.AddBefore(node, copyOp); | ||||
| 
 | ||||
|             return temp; | ||||
|         } | ||||
|  | @ -1142,7 +1130,7 @@ namespace ARMeilleure.CodeGen.X86 | |||
|             return value == (int)value; | ||||
|         } | ||||
| 
 | ||||
|         private static void Delete(LLNode node, Operation operation) | ||||
|         private static void Delete(IntrusiveList<Node> nodes, Node node, Operation operation) | ||||
|         { | ||||
|             operation.Destination = null; | ||||
| 
 | ||||
|  | @ -1151,7 +1139,7 @@ namespace ARMeilleure.CodeGen.X86 | |||
|                 operation.SetSource(index, null); | ||||
|             } | ||||
| 
 | ||||
|             node.List.Remove(node); | ||||
|             nodes.Remove(node); | ||||
|         } | ||||
| 
 | ||||
|         private static Operand Gpr(X86Register register, OperandType type) | ||||
|  |  | |||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 gdkchan
						gdkchan