AInstEmitAluHelper.cs 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187
  1. using ChocolArm64.Decoder;
  2. using ChocolArm64.State;
  3. using ChocolArm64.Translation;
  4. using System.Reflection.Emit;
  5. namespace ChocolArm64.Instruction
  6. {
  7. static class AInstEmitAluHelper
  8. {
  9. public static void EmitAddsCCheck(AILEmitterCtx Context)
  10. {
  11. //C = Rd < Rn
  12. Context.Emit(OpCodes.Dup);
  13. EmitDataLoadRn(Context);
  14. Context.Emit(OpCodes.Clt_Un);
  15. Context.EmitStflg((int)APState.CBit);
  16. }
  17. public static void EmitAddsVCheck(AILEmitterCtx Context)
  18. {
  19. //V = (Rd ^ Rn) & ~(Rn ^ Rm) < 0
  20. Context.Emit(OpCodes.Dup);
  21. EmitDataLoadRn(Context);
  22. Context.Emit(OpCodes.Xor);
  23. EmitDataLoadOpers(Context);
  24. Context.Emit(OpCodes.Xor);
  25. Context.Emit(OpCodes.Not);
  26. Context.Emit(OpCodes.And);
  27. Context.EmitLdc_I(0);
  28. Context.Emit(OpCodes.Clt);
  29. Context.EmitStflg((int)APState.VBit);
  30. }
  31. public static void EmitSbcsCCheck(AILEmitterCtx Context)
  32. {
  33. //C = (Rn == Rm && CIn) || Rn > Rm
  34. EmitDataLoadOpers(Context);
  35. Context.Emit(OpCodes.Ceq);
  36. Context.EmitLdflg((int)APState.CBit);
  37. Context.Emit(OpCodes.And);
  38. EmitDataLoadOpers(Context);
  39. Context.Emit(OpCodes.Cgt_Un);
  40. Context.Emit(OpCodes.Or);
  41. Context.EmitStflg((int)APState.CBit);
  42. }
  43. public static void EmitSubsCCheck(AILEmitterCtx Context)
  44. {
  45. //C = Rn == Rm || Rn > Rm = !(Rn < Rm)
  46. EmitDataLoadOpers(Context);
  47. Context.Emit(OpCodes.Clt_Un);
  48. Context.EmitLdc_I4(1);
  49. Context.Emit(OpCodes.Xor);
  50. Context.EmitStflg((int)APState.CBit);
  51. }
  52. public static void EmitSubsVCheck(AILEmitterCtx Context)
  53. {
  54. //V = (Rd ^ Rn) & (Rn ^ Rm) < 0
  55. Context.Emit(OpCodes.Dup);
  56. EmitDataLoadRn(Context);
  57. Context.Emit(OpCodes.Xor);
  58. EmitDataLoadOpers(Context);
  59. Context.Emit(OpCodes.Xor);
  60. Context.Emit(OpCodes.And);
  61. Context.EmitLdc_I(0);
  62. Context.Emit(OpCodes.Clt);
  63. Context.EmitStflg((int)APState.VBit);
  64. }
  65. public static void EmitDataLoadRm(AILEmitterCtx Context)
  66. {
  67. Context.EmitLdintzr(((IAOpCodeAluRs)Context.CurrOp).Rm);
  68. }
  69. public static void EmitDataLoadOpers(AILEmitterCtx Context)
  70. {
  71. EmitDataLoadRn(Context);
  72. EmitDataLoadOper2(Context);
  73. }
  74. public static void EmitDataLoadRn(AILEmitterCtx Context)
  75. {
  76. IAOpCodeAlu Op = (IAOpCodeAlu)Context.CurrOp;
  77. if (Op.DataOp == ADataOp.Logical || Op is IAOpCodeAluRs)
  78. {
  79. Context.EmitLdintzr(Op.Rn);
  80. }
  81. else
  82. {
  83. Context.EmitLdint(Op.Rn);
  84. }
  85. }
  86. public static void EmitDataLoadOper2(AILEmitterCtx Context)
  87. {
  88. switch (Context.CurrOp)
  89. {
  90. case IAOpCodeAluImm Op:
  91. Context.EmitLdc_I(Op.Imm);
  92. break;
  93. case IAOpCodeAluRs Op:
  94. Context.EmitLdintzr(Op.Rm);
  95. switch (Op.ShiftType)
  96. {
  97. case AShiftType.Lsl: Context.EmitLsl(Op.Shift); break;
  98. case AShiftType.Lsr: Context.EmitLsr(Op.Shift); break;
  99. case AShiftType.Asr: Context.EmitAsr(Op.Shift); break;
  100. case AShiftType.Ror: Context.EmitRor(Op.Shift); break;
  101. }
  102. break;
  103. case IAOpCodeAluRx Op:
  104. Context.EmitLdintzr(Op.Rm);
  105. Context.EmitCast(Op.IntType);
  106. Context.EmitLsl(Op.Shift);
  107. break;
  108. }
  109. }
  110. public static void EmitDataStore(AILEmitterCtx Context) => EmitDataStore(Context, false);
  111. public static void EmitDataStoreS(AILEmitterCtx Context) => EmitDataStore(Context, true);
  112. public static void EmitDataStore(AILEmitterCtx Context, bool SetFlags)
  113. {
  114. IAOpCodeAlu Op = (IAOpCodeAlu)Context.CurrOp;
  115. if (SetFlags || Op is IAOpCodeAluRs)
  116. {
  117. Context.EmitStintzr(Op.Rd);
  118. }
  119. else
  120. {
  121. Context.EmitStint(Op.Rd);
  122. }
  123. }
  124. public static void EmitSetNZCV(AILEmitterCtx Context, int NZCV)
  125. {
  126. Context.EmitLdc_I4((NZCV >> 0) & 1);
  127. Context.EmitStflg((int)APState.VBit);
  128. Context.EmitLdc_I4((NZCV >> 1) & 1);
  129. Context.EmitStflg((int)APState.CBit);
  130. Context.EmitLdc_I4((NZCV >> 2) & 1);
  131. Context.EmitStflg((int)APState.ZBit);
  132. Context.EmitLdc_I4((NZCV >> 3) & 1);
  133. Context.EmitStflg((int)APState.NBit);
  134. }
  135. }
  136. }