InstEmitMemory32.cs 8.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319
  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.EmitLdtmp();
  51. context.EmitLdc_I4(offset);
  52. context.Emit(OpCodes.Add);
  53. EmitReadZxCall(context, WordSizeLog2);
  54. EmitStoreToRegister(context, register);
  55. offset += 4;
  56. }
  57. }
  58. }
  59. public static void Ldr(ILEmitterCtx context)
  60. {
  61. EmitLoadOrStore(context, WordSizeLog2, AccessType.LoadZx);
  62. }
  63. public static void Ldrb(ILEmitterCtx context)
  64. {
  65. EmitLoadOrStore(context, ByteSizeLog2, AccessType.LoadZx);
  66. }
  67. public static void Ldrd(ILEmitterCtx context)
  68. {
  69. EmitLoadOrStore(context, DWordSizeLog2, AccessType.LoadZx);
  70. }
  71. public static void Ldrh(ILEmitterCtx context)
  72. {
  73. EmitLoadOrStore(context, HWordSizeLog2, AccessType.LoadZx);
  74. }
  75. public static void Ldrsb(ILEmitterCtx context)
  76. {
  77. EmitLoadOrStore(context, ByteSizeLog2, AccessType.LoadSx);
  78. }
  79. public static void Ldrsh(ILEmitterCtx context)
  80. {
  81. EmitLoadOrStore(context, HWordSizeLog2, AccessType.LoadSx);
  82. }
  83. public static void Stm(ILEmitterCtx context)
  84. {
  85. OpCode32MemMult op = (OpCode32MemMult)context.CurrOp;
  86. EmitLoadFromRegister(context, op.Rn);
  87. context.EmitLdc_I4(op.Offset);
  88. context.Emit(OpCodes.Add);
  89. context.EmitSttmp();
  90. int mask = op.RegisterMask;
  91. int offset = 0;
  92. for (int register = 0; mask != 0; mask >>= 1, register++)
  93. {
  94. if ((mask & 1) != 0)
  95. {
  96. context.EmitLdtmp();
  97. context.EmitLdc_I4(offset);
  98. context.Emit(OpCodes.Add);
  99. EmitLoadFromRegister(context, register);
  100. EmitWriteCall(context, WordSizeLog2);
  101. //Note: If Rn is also specified on the register list,
  102. //and Rn is the first register on this list, then the
  103. //value that is written to memory is the unmodified value,
  104. //before the write back. If it is on the list, but it's
  105. //not the first one, then the value written to memory
  106. //varies between CPUs.
  107. if (offset == 0 && op.PostOffset != 0)
  108. {
  109. //Emit write back after the first write.
  110. EmitLoadFromRegister(context, op.Rn);
  111. context.EmitLdc_I4(op.PostOffset);
  112. context.Emit(OpCodes.Add);
  113. EmitStoreToRegister(context, op.Rn);
  114. }
  115. offset += 4;
  116. }
  117. }
  118. }
  119. public static void Str(ILEmitterCtx context)
  120. {
  121. EmitLoadOrStore(context, WordSizeLog2, AccessType.Store);
  122. }
  123. public static void Strb(ILEmitterCtx context)
  124. {
  125. EmitLoadOrStore(context, ByteSizeLog2, AccessType.Store);
  126. }
  127. public static void Strd(ILEmitterCtx context)
  128. {
  129. EmitLoadOrStore(context, DWordSizeLog2, AccessType.Store);
  130. }
  131. public static void Strh(ILEmitterCtx context)
  132. {
  133. EmitLoadOrStore(context, HWordSizeLog2, AccessType.Store);
  134. }
  135. private static void EmitLoadOrStore(ILEmitterCtx context, int size, AccessType accType)
  136. {
  137. OpCode32Mem op = (OpCode32Mem)context.CurrOp;
  138. if (op.Index || op.WBack)
  139. {
  140. EmitLoadFromRegister(context, op.Rn);
  141. context.EmitLdc_I4(op.Imm);
  142. context.Emit(op.Add ? OpCodes.Add : OpCodes.Sub);
  143. context.EmitSttmp();
  144. }
  145. if (op.Index)
  146. {
  147. context.EmitLdtmp();
  148. }
  149. else
  150. {
  151. EmitLoadFromRegister(context, op.Rn);
  152. }
  153. if ((accType & AccessType.Load) != 0)
  154. {
  155. if ((accType & AccessType.Signed) != 0)
  156. {
  157. EmitReadSx32Call(context, size);
  158. }
  159. else
  160. {
  161. EmitReadZxCall(context, size);
  162. }
  163. if (op.WBack)
  164. {
  165. context.EmitLdtmp();
  166. EmitStoreToRegister(context, op.Rn);
  167. }
  168. if (size == DWordSizeLog2)
  169. {
  170. context.Emit(OpCodes.Dup);
  171. context.EmitLdflg((int)PState.EBit);
  172. ILLabel lblBigEndian = new ILLabel();
  173. ILLabel lblEnd = new ILLabel();
  174. context.Emit(OpCodes.Brtrue_S, lblBigEndian);
  175. //Little endian mode.
  176. context.Emit(OpCodes.Conv_U4);
  177. EmitStoreToRegister(context, op.Rt);
  178. context.EmitLsr(32);
  179. context.Emit(OpCodes.Conv_U4);
  180. EmitStoreToRegister(context, op.Rt | 1);
  181. context.Emit(OpCodes.Br_S, lblEnd);
  182. //Big endian mode.
  183. context.MarkLabel(lblBigEndian);
  184. context.EmitLsr(32);
  185. context.Emit(OpCodes.Conv_U4);
  186. EmitStoreToRegister(context, op.Rt);
  187. context.Emit(OpCodes.Conv_U4);
  188. EmitStoreToRegister(context, op.Rt | 1);
  189. context.MarkLabel(lblEnd);
  190. }
  191. else
  192. {
  193. EmitStoreToRegister(context, op.Rt);
  194. }
  195. }
  196. else
  197. {
  198. if (op.WBack)
  199. {
  200. context.EmitLdtmp();
  201. EmitStoreToRegister(context, op.Rn);
  202. }
  203. EmitLoadFromRegister(context, op.Rt);
  204. if (size == DWordSizeLog2)
  205. {
  206. context.Emit(OpCodes.Conv_U8);
  207. context.EmitLdflg((int)PState.EBit);
  208. ILLabel lblBigEndian = new ILLabel();
  209. ILLabel lblEnd = new ILLabel();
  210. context.Emit(OpCodes.Brtrue_S, lblBigEndian);
  211. //Little endian mode.
  212. EmitLoadFromRegister(context, op.Rt | 1);
  213. context.Emit(OpCodes.Conv_U8);
  214. context.EmitLsl(32);
  215. context.Emit(OpCodes.Or);
  216. context.Emit(OpCodes.Br_S, lblEnd);
  217. //Big endian mode.
  218. context.MarkLabel(lblBigEndian);
  219. context.EmitLsl(32);
  220. EmitLoadFromRegister(context, op.Rt | 1);
  221. context.Emit(OpCodes.Conv_U8);
  222. context.Emit(OpCodes.Or);
  223. context.MarkLabel(lblEnd);
  224. }
  225. EmitWriteCall(context, size);
  226. }
  227. }
  228. }
  229. }