InstEmitMemoryHelper.cs 4.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138
  1. using ChocolArm64.Decoders;
  2. using ChocolArm64.Memory;
  3. using ChocolArm64.Translation;
  4. using System;
  5. using System.Reflection.Emit;
  6. namespace ChocolArm64.Instructions
  7. {
  8. static class InstEmitMemoryHelper
  9. {
  10. private enum Extension
  11. {
  12. Zx,
  13. Sx32,
  14. Sx64
  15. }
  16. public static void EmitReadZxCall(ILEmitterCtx context, int size)
  17. {
  18. EmitReadCall(context, Extension.Zx, size);
  19. }
  20. public static void EmitReadSx32Call(ILEmitterCtx context, int size)
  21. {
  22. EmitReadCall(context, Extension.Sx32, size);
  23. }
  24. public static void EmitReadSx64Call(ILEmitterCtx context, int size)
  25. {
  26. EmitReadCall(context, Extension.Sx64, size);
  27. }
  28. private static void EmitReadCall(ILEmitterCtx context, Extension ext, int size)
  29. {
  30. bool isSimd = GetIsSimd(context);
  31. string name = null;
  32. if (size < 0 || size > (isSimd ? 4 : 3))
  33. {
  34. throw new ArgumentOutOfRangeException(nameof(size));
  35. }
  36. if (isSimd)
  37. {
  38. switch (size)
  39. {
  40. case 0: name = nameof(MemoryManager.ReadVector8); break;
  41. case 1: name = nameof(MemoryManager.ReadVector16); break;
  42. case 2: name = nameof(MemoryManager.ReadVector32); break;
  43. case 3: name = nameof(MemoryManager.ReadVector64); break;
  44. case 4: name = nameof(MemoryManager.ReadVector128); break;
  45. }
  46. }
  47. else
  48. {
  49. switch (size)
  50. {
  51. case 0: name = nameof(MemoryManager.ReadByte); break;
  52. case 1: name = nameof(MemoryManager.ReadUInt16); break;
  53. case 2: name = nameof(MemoryManager.ReadUInt32); break;
  54. case 3: name = nameof(MemoryManager.ReadUInt64); break;
  55. }
  56. }
  57. context.EmitCall(typeof(MemoryManager), name);
  58. if (!isSimd)
  59. {
  60. if (ext == Extension.Sx32 ||
  61. ext == Extension.Sx64)
  62. {
  63. switch (size)
  64. {
  65. case 0: context.Emit(OpCodes.Conv_I1); break;
  66. case 1: context.Emit(OpCodes.Conv_I2); break;
  67. case 2: context.Emit(OpCodes.Conv_I4); break;
  68. }
  69. }
  70. if (size < 3)
  71. {
  72. context.Emit(ext == Extension.Sx64
  73. ? OpCodes.Conv_I8
  74. : OpCodes.Conv_U8);
  75. }
  76. }
  77. }
  78. public static void EmitWriteCall(ILEmitterCtx context, int size)
  79. {
  80. bool isSimd = GetIsSimd(context);
  81. string name = null;
  82. if (size < 0 || size > (isSimd ? 4 : 3))
  83. {
  84. throw new ArgumentOutOfRangeException(nameof(size));
  85. }
  86. if (size < 3 && !isSimd)
  87. {
  88. context.Emit(OpCodes.Conv_I4);
  89. }
  90. if (isSimd)
  91. {
  92. switch (size)
  93. {
  94. case 0: name = nameof(MemoryManager.WriteVector8); break;
  95. case 1: name = nameof(MemoryManager.WriteVector16); break;
  96. case 2: name = nameof(MemoryManager.WriteVector32); break;
  97. case 3: name = nameof(MemoryManager.WriteVector64); break;
  98. case 4: name = nameof(MemoryManager.WriteVector128); break;
  99. }
  100. }
  101. else
  102. {
  103. switch (size)
  104. {
  105. case 0: name = nameof(MemoryManager.WriteByte); break;
  106. case 1: name = nameof(MemoryManager.WriteUInt16); break;
  107. case 2: name = nameof(MemoryManager.WriteUInt32); break;
  108. case 3: name = nameof(MemoryManager.WriteUInt64); break;
  109. }
  110. }
  111. context.EmitCall(typeof(MemoryManager), name);
  112. }
  113. private static bool GetIsSimd(ILEmitterCtx context)
  114. {
  115. return context.CurrOp is IOpCodeSimd64 &&
  116. !(context.CurrOp is OpCodeSimdMemMs64 ||
  117. context.CurrOp is OpCodeSimdMemSs64);
  118. }
  119. }
  120. }