InstEmitMemory32.cs 8.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325
  1. using ChocolArm64.Decoders;
  2. using ChocolArm64.State;
  3. using ChocolArm64.Translation;
  4. using System;
  5. using System.Reflection.Emit;
  6. using static ChocolArm64.Instructions.InstEmit32Helper;
  7. using static ChocolArm64.Instructions.InstEmitMemoryHelper;
  8. namespace ChocolArm64.Instructions
  9. {
  10. static partial class InstEmit32
  11. {
  12. private const int ByteSizeLog2 = 0;
  13. private const int HWordSizeLog2 = 1;
  14. private const int WordSizeLog2 = 2;
  15. private const int DWordSizeLog2 = 3;
  16. [Flags]
  17. enum AccessType
  18. {
  19. Store = 0,
  20. Signed = 1,
  21. Load = 2,
  22. LoadZx = Load,
  23. LoadSx = Load | Signed,
  24. }
  25. public static void Ldm(ILEmitterCtx context)
  26. {
  27. OpCode32MemMult op = (OpCode32MemMult)context.CurrOp;
  28. EmitLoadFromRegister(context, op.Rn);
  29. bool writesToPc = (op.RegisterMask & (1 << RegisterAlias.Aarch32Pc)) != 0;
  30. bool writeBack = op.PostOffset != 0 && (op.Rn != RegisterAlias.Aarch32Pc || !writesToPc);
  31. if (writeBack)
  32. {
  33. context.Emit(OpCodes.Dup);
  34. }
  35. context.EmitLdc_I4(op.Offset);
  36. context.Emit(OpCodes.Add);
  37. context.EmitSttmp();
  38. if (writeBack)
  39. {
  40. context.EmitLdc_I4(op.PostOffset);
  41. context.Emit(OpCodes.Add);
  42. EmitStoreToRegister(context, op.Rn);
  43. }
  44. int mask = op.RegisterMask;
  45. int offset = 0;
  46. for (int register = 0; mask != 0; mask >>= 1, register++)
  47. {
  48. if ((mask & 1) != 0)
  49. {
  50. context.EmitLdarg(TranslatedSub.MemoryArgIdx);
  51. context.EmitLdtmp();
  52. context.EmitLdc_I4(offset);
  53. context.Emit(OpCodes.Add);
  54. EmitReadZxCall(context, WordSizeLog2);
  55. EmitStoreToRegister(context, register);
  56. offset += 4;
  57. }
  58. }
  59. }
  60. public static void Ldr(ILEmitterCtx context)
  61. {
  62. EmitLoadOrStore(context, WordSizeLog2, AccessType.LoadZx);
  63. }
  64. public static void Ldrb(ILEmitterCtx context)
  65. {
  66. EmitLoadOrStore(context, ByteSizeLog2, AccessType.LoadZx);
  67. }
  68. public static void Ldrd(ILEmitterCtx context)
  69. {
  70. EmitLoadOrStore(context, DWordSizeLog2, AccessType.LoadZx);
  71. }
  72. public static void Ldrh(ILEmitterCtx context)
  73. {
  74. EmitLoadOrStore(context, HWordSizeLog2, AccessType.LoadZx);
  75. }
  76. public static void Ldrsb(ILEmitterCtx context)
  77. {
  78. EmitLoadOrStore(context, ByteSizeLog2, AccessType.LoadSx);
  79. }
  80. public static void Ldrsh(ILEmitterCtx context)
  81. {
  82. EmitLoadOrStore(context, HWordSizeLog2, AccessType.LoadSx);
  83. }
  84. public static void Stm(ILEmitterCtx context)
  85. {
  86. OpCode32MemMult op = (OpCode32MemMult)context.CurrOp;
  87. EmitLoadFromRegister(context, op.Rn);
  88. context.EmitLdc_I4(op.Offset);
  89. context.Emit(OpCodes.Add);
  90. context.EmitSttmp();
  91. int mask = op.RegisterMask;
  92. int offset = 0;
  93. for (int register = 0; mask != 0; mask >>= 1, register++)
  94. {
  95. if ((mask & 1) != 0)
  96. {
  97. context.EmitLdarg(TranslatedSub.MemoryArgIdx);
  98. context.EmitLdtmp();
  99. context.EmitLdc_I4(offset);
  100. context.Emit(OpCodes.Add);
  101. EmitLoadFromRegister(context, register);
  102. EmitWriteCall(context, WordSizeLog2);
  103. //Note: If Rn is also specified on the register list,
  104. //and Rn is the first register on this list, then the
  105. //value that is written to memory is the unmodified value,
  106. //before the write back. If it is on the list, but it's
  107. //not the first one, then the value written to memory
  108. //varies between CPUs.
  109. if (offset == 0 && op.PostOffset != 0)
  110. {
  111. //Emit write back after the first write.
  112. EmitLoadFromRegister(context, op.Rn);
  113. context.EmitLdc_I4(op.PostOffset);
  114. context.Emit(OpCodes.Add);
  115. EmitStoreToRegister(context, op.Rn);
  116. }
  117. offset += 4;
  118. }
  119. }
  120. }
  121. public static void Str(ILEmitterCtx context)
  122. {
  123. EmitLoadOrStore(context, WordSizeLog2, AccessType.Store);
  124. }
  125. public static void Strb(ILEmitterCtx context)
  126. {
  127. EmitLoadOrStore(context, ByteSizeLog2, AccessType.Store);
  128. }
  129. public static void Strd(ILEmitterCtx context)
  130. {
  131. EmitLoadOrStore(context, DWordSizeLog2, AccessType.Store);
  132. }
  133. public static void Strh(ILEmitterCtx context)
  134. {
  135. EmitLoadOrStore(context, HWordSizeLog2, AccessType.Store);
  136. }
  137. private static void EmitLoadOrStore(ILEmitterCtx context, int size, AccessType accType)
  138. {
  139. OpCode32Mem op = (OpCode32Mem)context.CurrOp;
  140. if (op.Index || op.WBack)
  141. {
  142. EmitLoadFromRegister(context, op.Rn);
  143. context.EmitLdc_I4(op.Imm);
  144. context.Emit(op.Add ? OpCodes.Add : OpCodes.Sub);
  145. context.EmitSttmp();
  146. }
  147. context.EmitLdarg(TranslatedSub.MemoryArgIdx);
  148. if (op.Index)
  149. {
  150. context.EmitLdtmp();
  151. }
  152. else
  153. {
  154. EmitLoadFromRegister(context, op.Rn);
  155. }
  156. if ((accType & AccessType.Load) != 0)
  157. {
  158. if ((accType & AccessType.Signed) != 0)
  159. {
  160. EmitReadSx32Call(context, size);
  161. }
  162. else
  163. {
  164. EmitReadZxCall(context, size);
  165. }
  166. if (op.WBack)
  167. {
  168. context.EmitLdtmp();
  169. EmitStoreToRegister(context, op.Rn);
  170. }
  171. if (size == DWordSizeLog2)
  172. {
  173. context.Emit(OpCodes.Dup);
  174. context.EmitLdflg((int)PState.EBit);
  175. ILLabel lblBigEndian = new ILLabel();
  176. ILLabel lblEnd = new ILLabel();
  177. context.Emit(OpCodes.Brtrue_S, lblBigEndian);
  178. //Little endian mode.
  179. context.Emit(OpCodes.Conv_U4);
  180. EmitStoreToRegister(context, op.Rt);
  181. context.EmitLsr(32);
  182. context.Emit(OpCodes.Conv_U4);
  183. EmitStoreToRegister(context, op.Rt | 1);
  184. context.Emit(OpCodes.Br_S, lblEnd);
  185. //Big endian mode.
  186. context.MarkLabel(lblBigEndian);
  187. context.EmitLsr(32);
  188. context.Emit(OpCodes.Conv_U4);
  189. EmitStoreToRegister(context, op.Rt);
  190. context.Emit(OpCodes.Conv_U4);
  191. EmitStoreToRegister(context, op.Rt | 1);
  192. context.MarkLabel(lblEnd);
  193. }
  194. else
  195. {
  196. EmitStoreToRegister(context, op.Rt);
  197. }
  198. }
  199. else
  200. {
  201. if (op.WBack)
  202. {
  203. context.EmitLdtmp();
  204. EmitStoreToRegister(context, op.Rn);
  205. }
  206. EmitLoadFromRegister(context, op.Rt);
  207. if (size == DWordSizeLog2)
  208. {
  209. context.Emit(OpCodes.Conv_U8);
  210. context.EmitLdflg((int)PState.EBit);
  211. ILLabel lblBigEndian = new ILLabel();
  212. ILLabel lblEnd = new ILLabel();
  213. context.Emit(OpCodes.Brtrue_S, lblBigEndian);
  214. //Little endian mode.
  215. EmitLoadFromRegister(context, op.Rt | 1);
  216. context.Emit(OpCodes.Conv_U8);
  217. context.EmitLsl(32);
  218. context.Emit(OpCodes.Or);
  219. context.Emit(OpCodes.Br_S, lblEnd);
  220. //Big endian mode.
  221. context.MarkLabel(lblBigEndian);
  222. context.EmitLsl(32);
  223. EmitLoadFromRegister(context, op.Rt | 1);
  224. context.Emit(OpCodes.Conv_U8);
  225. context.Emit(OpCodes.Or);
  226. context.MarkLabel(lblEnd);
  227. }
  228. EmitWriteCall(context, size);
  229. }
  230. }
  231. }
  232. }