AInstEmitAluHelper.cs 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168
  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 EmitSubsCCheck(AILEmitterCtx Context)
  32. {
  33. //C = Rn == Rm || Rn > Rm = !(Rn < Rm)
  34. EmitDataLoadOpers(Context);
  35. Context.Emit(OpCodes.Clt_Un);
  36. Context.EmitLdc_I4(1);
  37. Context.Emit(OpCodes.Xor);
  38. Context.EmitStflg((int)APState.CBit);
  39. }
  40. public static void EmitSubsVCheck(AILEmitterCtx Context)
  41. {
  42. //V = (Rd ^ Rn) & (Rn ^ Rm) < 0
  43. Context.Emit(OpCodes.Dup);
  44. EmitDataLoadRn(Context);
  45. Context.Emit(OpCodes.Xor);
  46. EmitDataLoadOpers(Context);
  47. Context.Emit(OpCodes.Xor);
  48. Context.Emit(OpCodes.And);
  49. Context.EmitLdc_I(0);
  50. Context.Emit(OpCodes.Clt);
  51. Context.EmitStflg((int)APState.VBit);
  52. }
  53. public static void EmitDataLoadRm(AILEmitterCtx Context)
  54. {
  55. Context.EmitLdintzr(((IAOpCodeAluRs)Context.CurrOp).Rm);
  56. }
  57. public static void EmitDataLoadOpers(AILEmitterCtx Context)
  58. {
  59. EmitDataLoadRn(Context);
  60. EmitDataLoadOper2(Context);
  61. }
  62. public static void EmitDataLoadRn(AILEmitterCtx Context)
  63. {
  64. IAOpCodeAlu Op = (IAOpCodeAlu)Context.CurrOp;
  65. if (Op.DataOp == ADataOp.Logical || Op is IAOpCodeAluRs)
  66. {
  67. Context.EmitLdintzr(Op.Rn);
  68. }
  69. else
  70. {
  71. Context.EmitLdint(Op.Rn);
  72. }
  73. }
  74. public static void EmitDataLoadOper2(AILEmitterCtx Context)
  75. {
  76. switch (Context.CurrOp)
  77. {
  78. case IAOpCodeAluImm Op:
  79. Context.EmitLdc_I(Op.Imm);
  80. break;
  81. case IAOpCodeAluRs Op:
  82. Context.EmitLdintzr(Op.Rm);
  83. switch (Op.ShiftType)
  84. {
  85. case AShiftType.Lsl: Context.EmitLsl(Op.Shift); break;
  86. case AShiftType.Lsr: Context.EmitLsr(Op.Shift); break;
  87. case AShiftType.Asr: Context.EmitAsr(Op.Shift); break;
  88. case AShiftType.Ror: Context.EmitRor(Op.Shift); break;
  89. }
  90. break;
  91. case IAOpCodeAluRx Op:
  92. Context.EmitLdintzr(Op.Rm);
  93. Context.EmitCast(Op.IntType);
  94. Context.EmitLsl(Op.Shift);
  95. break;
  96. }
  97. }
  98. public static void EmitDataStore(AILEmitterCtx Context) => EmitDataStore(Context, false);
  99. public static void EmitDataStoreS(AILEmitterCtx Context) => EmitDataStore(Context, true);
  100. public static void EmitDataStore(AILEmitterCtx Context, bool SetFlags)
  101. {
  102. IAOpCodeAlu Op = (IAOpCodeAlu)Context.CurrOp;
  103. if (SetFlags || Op is IAOpCodeAluRs)
  104. {
  105. Context.EmitStintzr(Op.Rd);
  106. }
  107. else
  108. {
  109. Context.EmitStint(Op.Rd);
  110. }
  111. }
  112. public static void EmitSetNZCV(AILEmitterCtx Context, int NZCV)
  113. {
  114. Context.EmitLdc_I4((NZCV >> 0) & 1);
  115. Context.EmitStflg((int)APState.VBit);
  116. Context.EmitLdc_I4((NZCV >> 1) & 1);
  117. Context.EmitStflg((int)APState.CBit);
  118. Context.EmitLdc_I4((NZCV >> 2) & 1);
  119. Context.EmitStflg((int)APState.ZBit);
  120. Context.EmitLdc_I4((NZCV >> 3) & 1);
  121. Context.EmitStflg((int)APState.NBit);
  122. }
  123. }
  124. }