 c1bdf19061
			
		
	
	
		c1bdf19061
		
			
		
	
	
	
	
		
			
			* Implement ARM32 memory instructions: LDM, LDR, LDRB, LDRD, LDRH, LDRSB, LDRSH, STM, STR, STRB, STRD, STRH (immediate and register + immediate variants), implement CMP (immediate and register shifted by immediate variants) * Rename some opcode classes and flag masks for consistency * Fix a few suboptimal ARM32 codegen issues, only loads should be considered on decoder when checking if Rt == PC, and only NZCV flags should be considered for comparison optimizations * Take into account Rt2 for LDRD instructions aswell when checking if the instruction changes PC * Re-align arm32 instructions on the opcode table
		
			
				
	
	
		
			684 lines
		
	
	
	
		
			19 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			684 lines
		
	
	
	
		
			19 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| using ChocolArm64.Decoders;
 | |
| using ChocolArm64.Instructions;
 | |
| using ChocolArm64.State;
 | |
| using System;
 | |
| using System.Collections.Generic;
 | |
| using System.Reflection;
 | |
| using System.Reflection.Emit;
 | |
| 
 | |
| namespace ChocolArm64.Translation
 | |
| {
 | |
|     class ILEmitterCtx
 | |
|     {
 | |
|         private TranslatorCache _cache;
 | |
| 
 | |
|         private Dictionary<long, ILLabel> _labels;
 | |
| 
 | |
|         private long _subPosition;
 | |
| 
 | |
|         private int _opcIndex;
 | |
| 
 | |
|         private Block _currBlock;
 | |
| 
 | |
|         public Block    CurrBlock => _currBlock;
 | |
|         public OpCode64 CurrOp    => _currBlock?.OpCodes[_opcIndex];
 | |
| 
 | |
|         public Aarch32Mode Mode { get; } = Aarch32Mode.User; //TODO
 | |
| 
 | |
|         private Dictionary<Block, ILBlock> _visitedBlocks;
 | |
| 
 | |
|         private Queue<Block> _branchTargets;
 | |
| 
 | |
|         private List<ILBlock> _ilBlocks;
 | |
| 
 | |
|         private ILBlock _ilBlock;
 | |
| 
 | |
|         private OpCode64 _optOpLastCompare;
 | |
|         private OpCode64 _optOpLastFlagSet;
 | |
| 
 | |
|         //This is the index of the temporary register, used to store temporary
 | |
|         //values needed by some functions, since IL doesn't have a swap instruction.
 | |
|         //You can use any value here as long it doesn't conflict with the indices
 | |
|         //for the other registers. Any value >= 64 or < 0 will do.
 | |
|         private const int IntTmpIndex     = -1;
 | |
|         private const int RorTmpIndex     = -2;
 | |
|         private const int CmpOptTmp1Index = -3;
 | |
|         private const int CmpOptTmp2Index = -4;
 | |
|         private const int VecTmp1Index    = -5;
 | |
|         private const int VecTmp2Index    = -6;
 | |
| 
 | |
|         public ILEmitterCtx(TranslatorCache cache, Block graph)
 | |
|         {
 | |
|             _cache     = cache ?? throw new ArgumentNullException(nameof(cache));
 | |
|             _currBlock = graph ?? throw new ArgumentNullException(nameof(graph));
 | |
| 
 | |
|             _labels = new Dictionary<long, ILLabel>();
 | |
| 
 | |
|             _visitedBlocks = new Dictionary<Block, ILBlock>();
 | |
| 
 | |
|             _visitedBlocks.Add(graph, new ILBlock());
 | |
| 
 | |
|             _branchTargets = new Queue<Block>();
 | |
| 
 | |
|             _ilBlocks = new List<ILBlock>();
 | |
| 
 | |
|             _subPosition = graph.Position;
 | |
| 
 | |
|             ResetBlockState();
 | |
| 
 | |
|             AdvanceOpCode();
 | |
|         }
 | |
| 
 | |
|         public ILBlock[] GetILBlocks()
 | |
|         {
 | |
|             EmitAllOpCodes();
 | |
| 
 | |
|             return _ilBlocks.ToArray();
 | |
|         }
 | |
| 
 | |
|         private void EmitAllOpCodes()
 | |
|         {
 | |
|             do
 | |
|             {
 | |
|                 EmitOpCode();
 | |
|             }
 | |
|             while (AdvanceOpCode());
 | |
|         }
 | |
| 
 | |
|         private void EmitOpCode()
 | |
|         {
 | |
|             if (_currBlock == null)
 | |
|             {
 | |
|                 return;
 | |
|             }
 | |
| 
 | |
|             if (_opcIndex == 0)
 | |
|             {
 | |
|                 MarkLabel(GetLabel(_currBlock.Position));
 | |
| 
 | |
|                 EmitSynchronization();
 | |
|             }
 | |
| 
 | |
|             //On AARCH32 mode, (almost) all instruction can be conditionally
 | |
|             //executed, and the required condition is encoded on the opcode.
 | |
|             //We handle that here, skipping the instruction if the condition
 | |
|             //is not met. We can just ignore it when the condition is "Always",
 | |
|             //because in this case the instruction is always going to be executed.
 | |
|             //Condition "Never" is also ignored because this is a special encoding
 | |
|             //used by some unconditional instructions.
 | |
|             ILLabel lblSkip = null;
 | |
| 
 | |
|             if (CurrOp is OpCode32 op && op.Cond < Condition.Al)
 | |
|             {
 | |
|                 lblSkip = new ILLabel();
 | |
| 
 | |
|                 EmitCondBranch(lblSkip, GetInverseCond(op.Cond));
 | |
|             }
 | |
| 
 | |
|             CurrOp.Emitter(this);
 | |
| 
 | |
|             if (lblSkip != null)
 | |
|             {
 | |
|                 MarkLabel(lblSkip);
 | |
| 
 | |
|                 //If this is the last op on the block, and there's no "next" block
 | |
|                 //after this one, then we have to return right now, with the address
 | |
|                 //of the next instruction to be executed (in the case that the condition
 | |
|                 //is false, and the branch was not taken, as all basic blocks should end with
 | |
|                 //some kind of branch).
 | |
|                 if (CurrOp == CurrBlock.GetLastOp() && CurrBlock.Next == null)
 | |
|                 {
 | |
|                     EmitStoreState();
 | |
|                     EmitLdc_I8(CurrOp.Position + CurrOp.OpCodeSizeInBytes);
 | |
| 
 | |
|                     Emit(OpCodes.Ret);
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             _ilBlock.Add(new ILBarrier());
 | |
|         }
 | |
| 
 | |
|         private Condition GetInverseCond(Condition cond)
 | |
|         {
 | |
|             //Bit 0 of all conditions is basically a negation bit, so
 | |
|             //inverting this bit has the effect of inverting the condition.
 | |
|             return (Condition)((int)cond ^ 1);
 | |
|         }
 | |
| 
 | |
|         private void EmitSynchronization()
 | |
|         {
 | |
|             EmitLdarg(TranslatedSub.StateArgIdx);
 | |
| 
 | |
|             EmitLdc_I4(_currBlock.OpCodes.Count);
 | |
| 
 | |
|             EmitPrivateCall(typeof(CpuThreadState), nameof(CpuThreadState.Synchronize));
 | |
| 
 | |
|             EmitLdc_I4(0);
 | |
| 
 | |
|             ILLabel lblContinue = new ILLabel();
 | |
| 
 | |
|             Emit(OpCodes.Bne_Un_S, lblContinue);
 | |
| 
 | |
|             EmitLdc_I8(0);
 | |
| 
 | |
|             Emit(OpCodes.Ret);
 | |
| 
 | |
|             MarkLabel(lblContinue);
 | |
|         }
 | |
| 
 | |
|         private bool AdvanceOpCode()
 | |
|         {
 | |
|             if (_currBlock == null)
 | |
|             {
 | |
|                 return false;
 | |
|             }
 | |
| 
 | |
|             while (++_opcIndex >= _currBlock.OpCodes.Count)
 | |
|             {
 | |
|                 if (!AdvanceBlock())
 | |
|                 {
 | |
|                     return false;
 | |
|                 }
 | |
| 
 | |
|                 ResetBlockState();
 | |
|             }
 | |
| 
 | |
|             return true;
 | |
|         }
 | |
| 
 | |
|         private bool AdvanceBlock()
 | |
|         {
 | |
|             if (_currBlock.Branch != null)
 | |
|             {
 | |
|                 if (_visitedBlocks.TryAdd(_currBlock.Branch, _ilBlock.Branch))
 | |
|                 {
 | |
|                     _branchTargets.Enqueue(_currBlock.Branch);
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             if (_currBlock.Next != null)
 | |
|             {
 | |
|                 if (_visitedBlocks.TryAdd(_currBlock.Next, _ilBlock.Next))
 | |
|                 {
 | |
|                     _currBlock = _currBlock.Next;
 | |
| 
 | |
|                     return true;
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     Emit(OpCodes.Br, GetLabel(_currBlock.Next.Position));
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             return _branchTargets.TryDequeue(out _currBlock);
 | |
|         }
 | |
| 
 | |
|         private void ResetBlockState()
 | |
|         {
 | |
|             _ilBlock = _visitedBlocks[_currBlock];
 | |
| 
 | |
|             _ilBlocks.Add(_ilBlock);
 | |
| 
 | |
|             _ilBlock.Next   = GetOrCreateILBlock(_currBlock.Next);
 | |
|             _ilBlock.Branch = GetOrCreateILBlock(_currBlock.Branch);
 | |
| 
 | |
|             _opcIndex = -1;
 | |
| 
 | |
|             _optOpLastFlagSet = null;
 | |
|             _optOpLastCompare = null;
 | |
|         }
 | |
| 
 | |
|         private ILBlock GetOrCreateILBlock(Block block)
 | |
|         {
 | |
|             if (block == null)
 | |
|             {
 | |
|                 return null;
 | |
|             }
 | |
| 
 | |
|             if (_visitedBlocks.TryGetValue(block, out ILBlock ilBlock))
 | |
|             {
 | |
|                 return ilBlock;
 | |
|             }
 | |
| 
 | |
|             return new ILBlock();
 | |
|         }
 | |
| 
 | |
|         public bool TryOptEmitSubroutineCall()
 | |
|         {
 | |
|             if (_currBlock.Next == null)
 | |
|             {
 | |
|                 return false;
 | |
|             }
 | |
| 
 | |
|             if (CurrOp.Emitter != InstEmit.Bl)
 | |
|             {
 | |
|                 return false;
 | |
|             }
 | |
| 
 | |
|             if (!_cache.TryGetSubroutine(((OpCodeBImmAl64)CurrOp).Imm, out TranslatedSub subroutine))
 | |
|             {
 | |
|                 return false;
 | |
|             }
 | |
| 
 | |
|             for (int index = 0; index < TranslatedSub.FixedArgTypes.Length; index++)
 | |
|             {
 | |
|                 EmitLdarg(index);
 | |
|             }
 | |
| 
 | |
|             foreach (Register reg in subroutine.SubArgs)
 | |
|             {
 | |
|                 switch (reg.Type)
 | |
|                 {
 | |
|                     case RegisterType.Flag:   Ldloc(reg.Index, IoType.Flag);   break;
 | |
|                     case RegisterType.Int:    Ldloc(reg.Index, IoType.Int);    break;
 | |
|                     case RegisterType.Vector: Ldloc(reg.Index, IoType.Vector); break;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             EmitCall(subroutine.Method);
 | |
| 
 | |
|             subroutine.AddCaller(_subPosition);
 | |
| 
 | |
|             return true;
 | |
|         }
 | |
| 
 | |
|         public void TryOptMarkCondWithoutCmp()
 | |
|         {
 | |
|             _optOpLastCompare = CurrOp;
 | |
| 
 | |
|             InstEmitAluHelper.EmitAluLoadOpers(this);
 | |
| 
 | |
|             Stloc(CmpOptTmp2Index, IoType.Int);
 | |
|             Stloc(CmpOptTmp1Index, IoType.Int);
 | |
|         }
 | |
| 
 | |
|         private Dictionary<Condition, OpCode> _branchOps = new Dictionary<Condition, OpCode>()
 | |
|         {
 | |
|             { Condition.Eq,   OpCodes.Beq    },
 | |
|             { Condition.Ne,   OpCodes.Bne_Un },
 | |
|             { Condition.GeUn, OpCodes.Bge_Un },
 | |
|             { Condition.LtUn, OpCodes.Blt_Un },
 | |
|             { Condition.GtUn, OpCodes.Bgt_Un },
 | |
|             { Condition.LeUn, OpCodes.Ble_Un },
 | |
|             { Condition.Ge,   OpCodes.Bge    },
 | |
|             { Condition.Lt,   OpCodes.Blt    },
 | |
|             { Condition.Gt,   OpCodes.Bgt    },
 | |
|             { Condition.Le,   OpCodes.Ble    }
 | |
|         };
 | |
| 
 | |
|         public void EmitCondBranch(ILLabel target, Condition cond)
 | |
|         {
 | |
|             OpCode ilOp;
 | |
| 
 | |
|             int intCond = (int)cond;
 | |
| 
 | |
|             if (_optOpLastCompare != null &&
 | |
|                 _optOpLastCompare == _optOpLastFlagSet && _branchOps.ContainsKey(cond))
 | |
|             {
 | |
|                 Ldloc(CmpOptTmp1Index, IoType.Int, _optOpLastCompare.RegisterSize);
 | |
|                 Ldloc(CmpOptTmp2Index, IoType.Int, _optOpLastCompare.RegisterSize);
 | |
| 
 | |
|                 ilOp = _branchOps[cond];
 | |
|             }
 | |
|             else if (intCond < 14)
 | |
|             {
 | |
|                 int condTrue = intCond >> 1;
 | |
| 
 | |
|                 switch (condTrue)
 | |
|                 {
 | |
|                     case 0: EmitLdflg((int)PState.ZBit); break;
 | |
|                     case 1: EmitLdflg((int)PState.CBit); break;
 | |
|                     case 2: EmitLdflg((int)PState.NBit); break;
 | |
|                     case 3: EmitLdflg((int)PState.VBit); break;
 | |
| 
 | |
|                     case 4:
 | |
|                         EmitLdflg((int)PState.CBit);
 | |
|                         EmitLdflg((int)PState.ZBit);
 | |
| 
 | |
|                         Emit(OpCodes.Not);
 | |
|                         Emit(OpCodes.And);
 | |
|                         break;
 | |
| 
 | |
|                     case 5:
 | |
|                     case 6:
 | |
|                         EmitLdflg((int)PState.NBit);
 | |
|                         EmitLdflg((int)PState.VBit);
 | |
| 
 | |
|                         Emit(OpCodes.Ceq);
 | |
| 
 | |
|                         if (condTrue == 6)
 | |
|                         {
 | |
|                             EmitLdflg((int)PState.ZBit);
 | |
| 
 | |
|                             Emit(OpCodes.Not);
 | |
|                             Emit(OpCodes.And);
 | |
|                         }
 | |
|                         break;
 | |
|                 }
 | |
| 
 | |
|                 ilOp = (intCond & 1) != 0
 | |
|                     ? OpCodes.Brfalse
 | |
|                     : OpCodes.Brtrue;
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 ilOp = OpCodes.Br;
 | |
|             }
 | |
| 
 | |
|             Emit(ilOp, target);
 | |
|         }
 | |
| 
 | |
|         public void EmitCast(IntType intType)
 | |
|         {
 | |
|             switch (intType)
 | |
|             {
 | |
|                 case IntType.UInt8:  Emit(OpCodes.Conv_U1); break;
 | |
|                 case IntType.UInt16: Emit(OpCodes.Conv_U2); break;
 | |
|                 case IntType.UInt32: Emit(OpCodes.Conv_U4); break;
 | |
|                 case IntType.UInt64: Emit(OpCodes.Conv_U8); break;
 | |
|                 case IntType.Int8:   Emit(OpCodes.Conv_I1); break;
 | |
|                 case IntType.Int16:  Emit(OpCodes.Conv_I2); break;
 | |
|                 case IntType.Int32:  Emit(OpCodes.Conv_I4); break;
 | |
|                 case IntType.Int64:  Emit(OpCodes.Conv_I8); break;
 | |
|             }
 | |
| 
 | |
|             bool sz64 = CurrOp.RegisterSize != RegisterSize.Int32;
 | |
| 
 | |
|             if (sz64 == (intType == IntType.UInt64 ||
 | |
|                          intType == IntType.Int64))
 | |
|             {
 | |
|                 return;
 | |
|             }
 | |
| 
 | |
|             if (sz64)
 | |
|             {
 | |
|                 Emit(intType >= IntType.Int8
 | |
|                     ? OpCodes.Conv_I8
 | |
|                     : OpCodes.Conv_U8);
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 Emit(OpCodes.Conv_U4);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public void EmitLsl(int amount) => EmitILShift(amount, OpCodes.Shl);
 | |
|         public void EmitLsr(int amount) => EmitILShift(amount, OpCodes.Shr_Un);
 | |
|         public void EmitAsr(int amount) => EmitILShift(amount, OpCodes.Shr);
 | |
| 
 | |
|         private void EmitILShift(int amount, OpCode ilOp)
 | |
|         {
 | |
|             if (amount > 0)
 | |
|             {
 | |
|                 EmitLdc_I4(amount);
 | |
| 
 | |
|                 Emit(ilOp);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public void EmitRor(int amount)
 | |
|         {
 | |
|             if (amount > 0)
 | |
|             {
 | |
|                 Stloc(RorTmpIndex, IoType.Int);
 | |
|                 Ldloc(RorTmpIndex, IoType.Int);
 | |
| 
 | |
|                 EmitLdc_I4(amount);
 | |
| 
 | |
|                 Emit(OpCodes.Shr_Un);
 | |
| 
 | |
|                 Ldloc(RorTmpIndex, IoType.Int);
 | |
| 
 | |
|                 EmitLdc_I4(CurrOp.GetBitsCount() - amount);
 | |
| 
 | |
|                 Emit(OpCodes.Shl);
 | |
|                 Emit(OpCodes.Or);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public ILLabel GetLabel(long position)
 | |
|         {
 | |
|             if (!_labels.TryGetValue(position, out ILLabel output))
 | |
|             {
 | |
|                 output = new ILLabel();
 | |
| 
 | |
|                 _labels.Add(position, output);
 | |
|             }
 | |
| 
 | |
|             return output;
 | |
|         }
 | |
| 
 | |
|         public void MarkLabel(ILLabel label)
 | |
|         {
 | |
|             _ilBlock.Add(label);
 | |
|         }
 | |
| 
 | |
|         public void Emit(OpCode ilOp)
 | |
|         {
 | |
|             _ilBlock.Add(new ILOpCode(ilOp));
 | |
|         }
 | |
| 
 | |
|         public void Emit(OpCode ilOp, ILLabel label)
 | |
|         {
 | |
|             _ilBlock.Add(new ILOpCodeBranch(ilOp, label));
 | |
|         }
 | |
| 
 | |
|         public void Emit(string text)
 | |
|         {
 | |
|             _ilBlock.Add(new ILOpCodeLog(text));
 | |
|         }
 | |
| 
 | |
|         public void EmitLdarg(int index)
 | |
|         {
 | |
|             _ilBlock.Add(new ILOpCodeLoad(index, IoType.Arg));
 | |
|         }
 | |
| 
 | |
|         public void EmitLdintzr(int index)
 | |
|         {
 | |
|             if (index != RegisterAlias.Zr)
 | |
|             {
 | |
|                 EmitLdint(index);
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 EmitLdc_I(0);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public void EmitStintzr(int index)
 | |
|         {
 | |
|             if (index != RegisterAlias.Zr)
 | |
|             {
 | |
|                 EmitStint(index);
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 Emit(OpCodes.Pop);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public void EmitLoadState()
 | |
|         {
 | |
|             if (_ilBlock.Next == null)
 | |
|             {
 | |
|                 throw new InvalidOperationException("Can't load state for next block, because there's no next block.");
 | |
|             }
 | |
| 
 | |
|             _ilBlock.Add(new ILOpCodeLoadState(_ilBlock.Next));
 | |
|         }
 | |
| 
 | |
|         public void EmitStoreState()
 | |
|         {
 | |
|             _ilBlock.Add(new ILOpCodeStoreState(_ilBlock));
 | |
|         }
 | |
| 
 | |
|         public void EmitLdtmp() => EmitLdint(IntTmpIndex);
 | |
|         public void EmitSttmp() => EmitStint(IntTmpIndex);
 | |
| 
 | |
|         public void EmitLdvectmp() => EmitLdvec(VecTmp1Index);
 | |
|         public void EmitStvectmp() => EmitStvec(VecTmp1Index);
 | |
| 
 | |
|         public void EmitLdvectmp2() => EmitLdvec(VecTmp2Index);
 | |
|         public void EmitStvectmp2() => EmitStvec(VecTmp2Index);
 | |
| 
 | |
|         public void EmitLdint(int index) => Ldloc(index, IoType.Int);
 | |
|         public void EmitStint(int index) => Stloc(index, IoType.Int);
 | |
| 
 | |
|         public void EmitLdvec(int index) => Ldloc(index, IoType.Vector);
 | |
|         public void EmitStvec(int index) => Stloc(index, IoType.Vector);
 | |
| 
 | |
|         public void EmitLdflg(int index) => Ldloc(index, IoType.Flag);
 | |
|         public void EmitStflg(int index)
 | |
|         {
 | |
|             //Set this only if any of the NZCV flag bits were modified.
 | |
|             //This is used to ensure that, when emiting a direct IL branch
 | |
|             //instruction for compare + branch sequences, we're not expecting
 | |
|             //to use comparison values from an old instruction, when in fact
 | |
|             //the flags were already overwritten by another instruction further along.
 | |
|             if (index >= (int)PState.VBit)
 | |
|             {
 | |
|                 _optOpLastFlagSet = CurrOp;
 | |
|             }
 | |
| 
 | |
|             Stloc(index, IoType.Flag);
 | |
|         }
 | |
| 
 | |
|         private void Ldloc(int index, IoType ioType)
 | |
|         {
 | |
|             _ilBlock.Add(new ILOpCodeLoad(index, ioType, CurrOp.RegisterSize));
 | |
|         }
 | |
| 
 | |
|         private void Ldloc(int index, IoType ioType, RegisterSize registerSize)
 | |
|         {
 | |
|             _ilBlock.Add(new ILOpCodeLoad(index, ioType, registerSize));
 | |
|         }
 | |
| 
 | |
|         private void Stloc(int index, IoType ioType)
 | |
|         {
 | |
|             _ilBlock.Add(new ILOpCodeStore(index, ioType, CurrOp.RegisterSize));
 | |
|         }
 | |
| 
 | |
|         public void EmitCallPropGet(Type objType, string propName)
 | |
|         {
 | |
|             if (objType == null)
 | |
|             {
 | |
|                 throw new ArgumentNullException(nameof(objType));
 | |
|             }
 | |
| 
 | |
|             if (propName == null)
 | |
|             {
 | |
|                 throw new ArgumentNullException(nameof(propName));
 | |
|             }
 | |
| 
 | |
|             EmitCall(objType.GetMethod($"get_{propName}"));
 | |
|         }
 | |
| 
 | |
|         public void EmitCallPropSet(Type objType, string propName)
 | |
|         {
 | |
|             if (objType == null)
 | |
|             {
 | |
|                 throw new ArgumentNullException(nameof(objType));
 | |
|             }
 | |
| 
 | |
|             if (propName == null)
 | |
|             {
 | |
|                 throw new ArgumentNullException(nameof(propName));
 | |
|             }
 | |
| 
 | |
|             EmitCall(objType.GetMethod($"set_{propName}"));
 | |
|         }
 | |
| 
 | |
|         public void EmitCall(Type objType, string mthdName)
 | |
|         {
 | |
|             if (objType == null)
 | |
|             {
 | |
|                 throw new ArgumentNullException(nameof(objType));
 | |
|             }
 | |
| 
 | |
|             if (mthdName == null)
 | |
|             {
 | |
|                 throw new ArgumentNullException(nameof(mthdName));
 | |
|             }
 | |
| 
 | |
|             EmitCall(objType.GetMethod(mthdName));
 | |
|         }
 | |
| 
 | |
|         public void EmitPrivateCall(Type objType, string mthdName)
 | |
|         {
 | |
|             if (objType == null)
 | |
|             {
 | |
|                 throw new ArgumentNullException(nameof(objType));
 | |
|             }
 | |
| 
 | |
|             if (mthdName == null)
 | |
|             {
 | |
|                 throw new ArgumentNullException(nameof(mthdName));
 | |
|             }
 | |
| 
 | |
|             EmitCall(objType.GetMethod(mthdName, BindingFlags.Instance | BindingFlags.NonPublic));
 | |
|         }
 | |
| 
 | |
|         public void EmitCall(MethodInfo mthdInfo)
 | |
|         {
 | |
|             if (mthdInfo == null)
 | |
|             {
 | |
|                 throw new ArgumentNullException(nameof(mthdInfo));
 | |
|             }
 | |
| 
 | |
|             _ilBlock.Add(new ILOpCodeCall(mthdInfo));
 | |
|         }
 | |
| 
 | |
|         public void EmitLdc_I(long value)
 | |
|         {
 | |
|             if (CurrOp.RegisterSize == RegisterSize.Int32)
 | |
|             {
 | |
|                 EmitLdc_I4((int)value);
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 EmitLdc_I8(value);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public void EmitLdc_I4(int value)
 | |
|         {
 | |
|             _ilBlock.Add(new ILOpCodeConst(value));
 | |
|         }
 | |
| 
 | |
|         public void EmitLdc_I8(long value)
 | |
|         {
 | |
|             _ilBlock.Add(new ILOpCodeConst(value));
 | |
|         }
 | |
| 
 | |
|         public void EmitLdc_R4(float value)
 | |
|         {
 | |
|             _ilBlock.Add(new ILOpCodeConst(value));
 | |
|         }
 | |
| 
 | |
|         public void EmitLdc_R8(double value)
 | |
|         {
 | |
|             _ilBlock.Add(new ILOpCodeConst(value));
 | |
|         }
 | |
| 
 | |
|         public void EmitZnFlagCheck()
 | |
|         {
 | |
|             EmitZnCheck(OpCodes.Ceq, (int)PState.ZBit);
 | |
|             EmitZnCheck(OpCodes.Clt, (int)PState.NBit);
 | |
|         }
 | |
| 
 | |
|         private void EmitZnCheck(OpCode ilCmpOp, int flag)
 | |
|         {
 | |
|             Emit(OpCodes.Dup);
 | |
|             Emit(OpCodes.Ldc_I4_0);
 | |
| 
 | |
|             if (CurrOp.RegisterSize != RegisterSize.Int32)
 | |
|             {
 | |
|                 Emit(OpCodes.Conv_I8);
 | |
|             }
 | |
| 
 | |
|             Emit(ilCmpOp);
 | |
| 
 | |
|             EmitStflg(flag);
 | |
|         }
 | |
|     }
 | |
| }
 |