AsmJsByteCodeDumper.cpp 46 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083
  1. //-------------------------------------------------------------------------------------------------------
  2. // Copyright (C) Microsoft Corporation and contributors. All rights reserved.
  3. // Licensed under the MIT license. See LICENSE.txt file in the project root for full license information.
  4. //-------------------------------------------------------------------------------------------------------
  5. #include "RuntimeByteCodePch.h"
  6. #ifdef ASMJS_PLAT
  7. #if DBG_DUMP
  8. #include "Language/AsmJsModule.h"
  9. #include "ByteCode/AsmJsByteCodeDumper.h"
  10. namespace Js
  11. {
  12. void AsmJsByteCodeDumper::Dump(FunctionBody* body, const WAsmJs::TypedRegisterAllocator* typedRegister, AsmJsFunc* asmFunc)
  13. {
  14. ByteCodeReader reader;
  15. reader.Create(body);
  16. StatementReader<FunctionBody::StatementMapList> statementReader;
  17. statementReader.Create(body);
  18. body->DumpFullFunctionName();
  19. Output::Print(_u(" Asm.js ("));
  20. AsmJsFunctionInfo* funcInfo = body->GetAsmJsFunctionInfo();
  21. const ArgSlot argCount = funcInfo->GetArgCount();
  22. for (ArgSlot i = 0; i < argCount; i++)
  23. {
  24. AsmJsVarType var = funcInfo->GetArgType(i);
  25. if (i > 0)
  26. {
  27. Output::Print(_u(", "));
  28. }
  29. if (var.isDouble())
  30. {
  31. Output::Print(_u("+In%hu"), i);
  32. }
  33. else if (var.isFloat())
  34. {
  35. Output::Print(_u("flt(In%hu)"), i);
  36. }
  37. else if (var.isInt())
  38. {
  39. Output::Print(_u("In%hu|0"), i);
  40. }
  41. else if (var.isInt64())
  42. {
  43. Output::Print(_u("int64(In%hu)"), i);
  44. }
  45. else if (var.isSIMD())
  46. {
  47. switch (var.which())
  48. {
  49. case AsmJsType::Int32x4:
  50. Output::Print(_u("I4(In%hu)"), i);
  51. break;
  52. case AsmJsType::Int8x16:
  53. Output::Print(_u("I16(In%hu)"), i);
  54. break;
  55. case AsmJsType::Float32x4:
  56. Output::Print(_u("F4(In%hu)"), i);
  57. break;
  58. case AsmJsType::Float64x2:
  59. Output::Print(_u("D2(In%hu)"), i);
  60. break;
  61. case AsmJsType::Int64x2:
  62. Output::Print(_u("I2(In%hu)"), i);
  63. break;
  64. }
  65. }
  66. else
  67. {
  68. Assert(UNREACHED);
  69. }
  70. }
  71. Output::Print(_u(") "));
  72. Output::Print(_u("(size: %d [%d])\n"), body->GetByteCodeCount(), body->GetByteCodeWithoutLDACount());
  73. if (!typedRegister && asmFunc)
  74. {
  75. typedRegister = &asmFunc->GetTypedRegisterAllocator();
  76. }
  77. if (typedRegister)
  78. {
  79. typedRegister->DumpLocalsInfo();
  80. }
  81. if (asmFunc)
  82. {
  83. DumpConstants(asmFunc, body);
  84. }
  85. if (typedRegister)
  86. {
  87. Output::Print(_u(" Implicit Arg Ins:\n ======== =====\n "));
  88. uint32 iArgs[WAsmJs::LIMIT];
  89. typedRegister->GetArgumentStartIndex(iArgs);
  90. uint32 iArg = iArgs[WAsmJs::INT32];
  91. uint32 lArg = iArgs[WAsmJs::INT64];
  92. uint32 dArg = iArgs[WAsmJs::FLOAT64];
  93. uint32 fArg = iArgs[WAsmJs::FLOAT32];
  94. uint32 simdArg = iArgs[WAsmJs::SIMD];
  95. for (ArgSlot i = 0; i < argCount; i++)
  96. {
  97. AsmJsVarType var = funcInfo->GetArgType(i);
  98. if (var.isDouble())
  99. {
  100. Output::Print(_u(" D%d In%d"), dArg++, i);
  101. }
  102. else if (var.isFloat())
  103. {
  104. Output::Print(_u(" F%d In%d"), fArg++, i);
  105. }
  106. else if (var.isInt())
  107. {
  108. Output::Print(_u(" I%d In%d"), iArg++, i);
  109. }
  110. else if (var.isInt64())
  111. {
  112. Output::Print(_u(" L%d In%d"), lArg++, i);
  113. }
  114. else if (var.isSIMD())
  115. {
  116. Output::Print(_u(" SIMD%d In%d"), simdArg++, i);
  117. }
  118. else
  119. {
  120. Assert(UNREACHED);
  121. }
  122. Output::Print(_u("\n "));
  123. }
  124. Output::Print(_u("\n"));
  125. }
  126. uint32 statementIndex = 0;
  127. while (true)
  128. {
  129. while (statementReader.AtStatementBoundary(&reader))
  130. {
  131. body->PrintStatementSourceLine(statementIndex);
  132. statementIndex = statementReader.MoveNextStatementBoundary();
  133. }
  134. int byteOffset = reader.GetCurrentOffset();
  135. LayoutSize layoutSize;
  136. OpCodeAsmJs op = reader.ReadAsmJsOp(layoutSize);
  137. if (op == OpCodeAsmJs::EndOfBlock)
  138. {
  139. Assert(reader.GetCurrentOffset() == body->GetByteCode()->GetLength());
  140. break;
  141. }
  142. Output::Print(_u(" %04x %2s"), byteOffset, layoutSize == LargeLayout ? _u("L-") : layoutSize == MediumLayout ? _u("M-") : _u(""));
  143. DumpOp(op, layoutSize, reader, body);
  144. if (Js::Configuration::Global.flags.Verbose)
  145. {
  146. int layoutStart = byteOffset + 2; // Account for the prefix op
  147. int endByteOffset = reader.GetCurrentOffset();
  148. Output::SkipToColumn(70);
  149. if (layoutSize == LargeLayout)
  150. {
  151. Output::Print(_u("%02X "),
  152. op > Js::OpCodeAsmJs::MaxByteSizedOpcodes ?
  153. Js::OpCodeAsmJs::ExtendedLargeLayoutPrefix : Js::OpCodeAsmJs::LargeLayoutPrefix);
  154. }
  155. else if (layoutSize == MediumLayout)
  156. {
  157. Output::Print(_u("%02X "),
  158. op > Js::OpCodeAsmJs::MaxByteSizedOpcodes ?
  159. Js::OpCodeAsmJs::ExtendedMediumLayoutPrefix : Js::OpCodeAsmJs::MediumLayoutPrefix);
  160. }
  161. else
  162. {
  163. Assert(layoutSize == SmallLayout);
  164. if (op > Js::OpCodeAsmJs::MaxByteSizedOpcodes)
  165. {
  166. Output::Print(_u("%02X "), Js::OpCodeAsmJs::ExtendedOpcodePrefix);
  167. }
  168. else
  169. {
  170. Output::Print(_u(" "));
  171. layoutStart--; // don't have a prefix
  172. }
  173. }
  174. Output::Print(_u("%02x"), (byte)op);
  175. for (int i = layoutStart; i < endByteOffset; i++)
  176. {
  177. Output::Print(_u(" %02x"), reader.GetRawByte(i));
  178. }
  179. }
  180. Output::Print(_u("\n"));
  181. }
  182. if (statementReader.AtStatementBoundary(&reader))
  183. {
  184. body->PrintStatementSourceLine(statementIndex);
  185. statementIndex = statementReader.MoveNextStatementBoundary();
  186. }
  187. Output::Print(_u("\n"));
  188. Output::Flush();
  189. }
  190. template<typename T, typename Func>
  191. void PrintTypedConstants(byte* table, WAsmJs::Types type, uint nConsts, Func printValFunc)
  192. {
  193. T* constTable = (T*)table;
  194. if (nConsts > 0)
  195. {
  196. char16 buf[32];
  197. WAsmJs::RegisterSpace::GetTypeDebugName(type, buf, 32);
  198. Output::Print(_u(" Constant %s:\n ======== =======\n "), buf);
  199. WAsmJs::RegisterSpace::GetTypeDebugName(type, buf, 32, true);
  200. for (uint i = 0; i < nConsts; i++)
  201. {
  202. Output::Print(_u(" %s%d "), buf, i);
  203. printValFunc(*constTable);
  204. Output::Print(_u("\n "));
  205. ++constTable;
  206. }
  207. }
  208. }
  209. void AsmJsByteCodeDumper::DumpConstants(AsmJsFunc* func, FunctionBody* body)
  210. {
  211. byte* table = (byte*)body->GetConstTable();
  212. auto constSrcInfos = func->GetTypedRegisterAllocator().GetConstSourceInfos();
  213. for (int i = 0; i < WAsmJs::LIMIT; ++i)
  214. {
  215. WAsmJs::Types type = (WAsmJs::Types)i;
  216. if (func->GetTypedRegisterAllocator().IsTypeExcluded(type))
  217. {
  218. continue;
  219. }
  220. uint constCount = func->GetTypedRegisterAllocator().GetRegisterSpace(type)->GetConstCount();
  221. if (constCount > 0)
  222. {
  223. uint offset = constSrcInfos.srcByteOffsets[i];
  224. byte* tableOffseted = table + offset;
  225. switch (type)
  226. {
  227. case WAsmJs::INT32: PrintTypedConstants<int>(tableOffseted, type, constCount, [](int v) {Output::Print(_u("%d"), v);}); break;
  228. case WAsmJs::INT64: PrintTypedConstants<int64>(tableOffseted, type, constCount, [](int64 v) {Output::Print(_u("%lld"), v);}); break;
  229. case WAsmJs::FLOAT32: PrintTypedConstants<float>(tableOffseted, type, constCount, [](float v) {Output::Print(_u("%.4f"), v);}); break;
  230. case WAsmJs::FLOAT64: PrintTypedConstants<double>(tableOffseted, type, constCount, [](double v) {Output::Print(_u("%.4f"), v);}); break;
  231. case WAsmJs::SIMD: PrintTypedConstants<AsmJsSIMDValue>(tableOffseted, type, constCount, [](AsmJsSIMDValue v) {
  232. Output::Print(_u("\n I32(%d, %d, %d, %d),"), v.i32[SIMD_X], v.i32[SIMD_Y], v.i32[SIMD_Z], v.i32[SIMD_W]);
  233. Output::Print(_u("\n F32(%.4f, %.4f, %.4f, %.4f),"), v.f32[SIMD_X], v.f32[SIMD_Y], v.f32[SIMD_Z], v.f32[SIMD_W]);
  234. Output::Print(_u("\n D64(%.4f, %.4f),"), v.f64[SIMD_X], v.f64[SIMD_Y]);
  235. Output::Print(_u("\n I8(%d, %d, %d, %d, %d, %d, %d, %d, %d, %d, %d, %d, %d, %d, %d, %d)"),
  236. v.i8[0], v.i8[1], v.i8[2], v.i8[3], v.i8[4], v.i8[5], v.i8[6], v.i8[7],
  237. v.i8[8], v.i8[9], v.i8[10], v.i8[11], v.i8[12], v.i8[13], v.i8[14], v.i8[15]);
  238. });
  239. break;
  240. default:
  241. Assert(false);
  242. break;
  243. }
  244. }
  245. Output::Print(_u("\n"));
  246. }
  247. }
  248. void AsmJsByteCodeDumper::DumpOp(OpCodeAsmJs op, LayoutSize layoutSize, ByteCodeReader& reader, FunctionBody* dumpFunction)
  249. {
  250. Output::Print(_u("%-20s"), OpCodeUtilAsmJs::GetOpCodeName(op));
  251. OpLayoutTypeAsmJs nType = OpCodeUtilAsmJs::GetOpCodeLayout(op);
  252. switch (layoutSize * OpLayoutTypeAsmJs::Count + nType)
  253. {
  254. #define LAYOUT_TYPE(layout) \
  255. case OpLayoutTypeAsmJs::layout: \
  256. Assert(layoutSize == SmallLayout); \
  257. Dump##layout(op, reader.layout(), dumpFunction, reader); \
  258. break;
  259. #define LAYOUT_TYPE_WMS(layout) \
  260. case SmallLayout * OpLayoutTypeAsmJs::Count + OpLayoutTypeAsmJs::layout: \
  261. Dump##layout(op, reader.layout##_Small(), dumpFunction, reader); \
  262. break; \
  263. case MediumLayout * OpLayoutTypeAsmJs::Count + OpLayoutTypeAsmJs::layout: \
  264. Dump##layout(op, reader.layout##_Medium(), dumpFunction, reader); \
  265. break; \
  266. case LargeLayout * OpLayoutTypeAsmJs::Count + OpLayoutTypeAsmJs::layout: \
  267. Dump##layout(op, reader.layout##_Large(), dumpFunction, reader); \
  268. break;
  269. #include "LayoutTypesAsmJs.h"
  270. default:
  271. AssertMsg(false, "Unknown OpLayout");
  272. break;
  273. }
  274. }
  275. void AsmJsByteCodeDumper::DumpIntReg(RegSlot reg)
  276. {
  277. Output::Print(_u(" I%d "), (int)reg);
  278. }
  279. void AsmJsByteCodeDumper::DumpLongReg(RegSlot reg)
  280. {
  281. Output::Print(_u(" L%d "), (int)reg);
  282. }
  283. void AsmJsByteCodeDumper::DumpDoubleReg(RegSlot reg)
  284. {
  285. Output::Print(_u(" D%d "), (int)reg);
  286. }
  287. void AsmJsByteCodeDumper::DumpFloatReg(RegSlot reg)
  288. {
  289. Output::Print(_u(" F%d "), (int)reg);
  290. }
  291. void AsmJsByteCodeDumper::DumpR8Float(float value)
  292. {
  293. Output::Print(_u(" float:%f "), value);
  294. }
  295. // Float32x4
  296. void AsmJsByteCodeDumper::DumpFloat32x4Reg(RegSlot reg)
  297. {
  298. Output::Print(_u(" F4_%d "), (int)reg);
  299. }
  300. // Int32x4
  301. void AsmJsByteCodeDumper::DumpInt32x4Reg(RegSlot reg)
  302. {
  303. Output::Print(_u(" I4_%d "), (int)reg);
  304. }
  305. void AsmJsByteCodeDumper::DumpUint32x4Reg(RegSlot reg)
  306. {
  307. Output::Print(_u(" U4_%d "), (int)reg);
  308. }
  309. void AsmJsByteCodeDumper::DumpInt16x8Reg(RegSlot reg)
  310. {
  311. Output::Print(_u(" I8_%d "), (int)reg);
  312. }
  313. // Int8x16
  314. void AsmJsByteCodeDumper::DumpInt8x16Reg(RegSlot reg)
  315. {
  316. Output::Print(_u(" I16_%d "), (int)reg);
  317. }
  318. void AsmJsByteCodeDumper::DumpUint16x8Reg(RegSlot reg)
  319. {
  320. Output::Print(_u(" U8_%d "), (int)reg);
  321. }
  322. void AsmJsByteCodeDumper::DumpUint8x16Reg(RegSlot reg)
  323. {
  324. Output::Print(_u(" U16_%d "), (int)reg);
  325. }
  326. // Bool32x4
  327. void AsmJsByteCodeDumper::DumpBool32x4Reg(RegSlot reg)
  328. {
  329. Output::Print(_u(" B4_%d "), (int)reg);
  330. }
  331. // Bool16x8
  332. void AsmJsByteCodeDumper::DumpBool16x8Reg(RegSlot reg)
  333. {
  334. Output::Print(_u(" B8_%d "), (int)reg);
  335. }
  336. // Bool32x4
  337. void AsmJsByteCodeDumper::DumpBool8x16Reg(RegSlot reg)
  338. {
  339. Output::Print(_u(" B16_%d "), (int)reg);
  340. }
  341. // Float64x2
  342. void AsmJsByteCodeDumper::DumpFloat64x2Reg(RegSlot reg)
  343. {
  344. Output::Print(_u(" D2_%d "), (int)reg);
  345. }
  346. // Int64x2
  347. void AsmJsByteCodeDumper::DumpInt64x2Reg(RegSlot reg)
  348. {
  349. Output::Print(_u(" I2_%d "), (int)reg);
  350. }
  351. template <class T>
  352. void AsmJsByteCodeDumper::DumpElementSlot(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)
  353. {
  354. switch (op)
  355. {
  356. case OpCodeAsmJs::LdSlot:
  357. case OpCodeAsmJs::LdSlotArr:
  358. {
  359. #ifdef ENABLE_WASM
  360. auto asmInfo = dumpFunction->GetAsmJsFunctionInfo();
  361. auto wasmInfo = asmInfo->GetWasmReaderInfo();
  362. if (wasmInfo)
  363. {
  364. uint index = (uint)data->SlotIndex;
  365. if (index - wasmInfo->m_module->GetFuncOffset() < wasmInfo->m_module->GetWasmFunctionCount())
  366. {
  367. uint funcIndex = data->SlotIndex - wasmInfo->m_module->GetFuncOffset();
  368. auto loadedFunc = wasmInfo->m_module->GetWasmFunctionInfo(funcIndex);
  369. Output::Print(_u(" R%d = %s"), data->Value, loadedFunc->GetBody()->GetDisplayName());
  370. break;
  371. }
  372. }
  373. #endif
  374. Output::Print(_u(" R%d = R%d[%d]"), data->Value, data->Instance, data->SlotIndex);
  375. break;
  376. }
  377. case OpCodeAsmJs::LdArr_Func:
  378. case OpCodeAsmJs::LdArr_WasmFunc:
  379. Output::Print(_u(" R%d = R%d[I%d]"), data->Value, data->Instance, data->SlotIndex);
  380. break;
  381. case OpCodeAsmJs::StSlot_Int:
  382. Output::Print(_u(" R%d[%d] = I%d"), data->Instance, data->SlotIndex, data->Value);
  383. break;
  384. case OpCodeAsmJs::StSlot_Long:
  385. Output::Print(_u(" R%d[%d] = L%d"), data->Instance, data->SlotIndex, data->Value);
  386. break;
  387. case OpCodeAsmJs::StSlot_Flt:
  388. Output::Print(_u(" R%d[%d] = F%d"), data->Instance, data->SlotIndex, data->Value);
  389. break;
  390. case OpCodeAsmJs::StSlot_Db:
  391. Output::Print(_u(" R%d[%d] = D%d"), data->Instance, data->SlotIndex, data->Value);
  392. break;
  393. case OpCodeAsmJs::LdSlot_Int:
  394. Output::Print(_u(" I%d = R%d[%d]"), data->Value, data->Instance, data->SlotIndex);
  395. break;
  396. case OpCodeAsmJs::LdSlot_Long:
  397. Output::Print(_u(" L%d = R%d[%d]"), data->Value, data->Instance, data->SlotIndex);
  398. break;
  399. case OpCodeAsmJs::LdSlot_Flt:
  400. Output::Print(_u(" F%d = R%d[%d]"), data->Value, data->Instance, data->SlotIndex);
  401. break;
  402. case OpCodeAsmJs::LdSlot_Db:
  403. Output::Print(_u(" D%d = R%d[%d]"), data->Value, data->Instance, data->SlotIndex);
  404. break;
  405. case OpCodeAsmJs::Simd128_LdSlot_F4:
  406. Output::Print(_u(" F4_%d = R%d[%d]"), data->Value, data->Instance, data->SlotIndex);
  407. break;
  408. case OpCodeAsmJs::Simd128_LdSlot_I4:
  409. Output::Print(_u(" I4_%d = R%d[%d]"), data->Value, data->Instance, data->SlotIndex);
  410. break;
  411. case OpCodeAsmJs::Simd128_LdSlot_I8:
  412. Output::Print(_u(" I8_%d = R%d[%d]"), data->Value, data->Instance, data->SlotIndex);
  413. break;
  414. case OpCodeAsmJs::Simd128_LdSlot_I16:
  415. Output::Print(_u(" I16_%d = R%d[%d]"), data->Value, data->Instance, data->SlotIndex);
  416. break;
  417. case OpCodeAsmJs::Simd128_LdSlot_U4:
  418. Output::Print(_u(" U4_%d = R%d[%d]"), data->Value, data->Instance, data->SlotIndex);
  419. break;
  420. case OpCodeAsmJs::Simd128_LdSlot_U8:
  421. Output::Print(_u(" U8_%d = R%d[%d]"), data->Value, data->Instance, data->SlotIndex);
  422. break;
  423. case OpCodeAsmJs::Simd128_LdSlot_U16:
  424. Output::Print(_u(" U16_%d = R%d[%d]"), data->Value, data->Instance, data->SlotIndex);
  425. break;
  426. case OpCodeAsmJs::Simd128_LdSlot_B4:
  427. Output::Print(_u(" B4_%d = R%d[%d]"), data->Value, data->Instance, data->SlotIndex);
  428. break;
  429. case OpCodeAsmJs::Simd128_LdSlot_B8:
  430. Output::Print(_u(" B8_%d = R%d[%d]"), data->Value, data->Instance, data->SlotIndex);
  431. break;
  432. case OpCodeAsmJs::Simd128_LdSlot_B16:
  433. Output::Print(_u(" B16_%d = R%d[%d]"), data->Value, data->Instance, data->SlotIndex);
  434. break;
  435. #if 0
  436. case OpCodeAsmJs::Simd128_LdSlot_D2:
  437. Output::Print(_u(" D2_%d = R%d[%d]"), data->Value, data->Instance, data->SlotIndex);
  438. break;
  439. #endif // 0
  440. case OpCodeAsmJs::Simd128_StSlot_F4:
  441. Output::Print(_u(" R%d[%d] = F4_%d"), data->Instance, data->SlotIndex, data->Value);
  442. break;
  443. case OpCodeAsmJs::Simd128_StSlot_I4:
  444. Output::Print(_u(" R%d[%d] = I4_%d"), data->Instance, data->SlotIndex, data->Value);
  445. break;
  446. case OpCodeAsmJs::Simd128_StSlot_I8:
  447. Output::Print(_u(" R%d[%d] = I8_%d"), data->Instance, data->SlotIndex, data->Value);
  448. break;
  449. case OpCodeAsmJs::Simd128_StSlot_I16:
  450. Output::Print(_u(" R%d[%d] = I16_%d"), data->Instance, data->SlotIndex, data->Value);
  451. break;
  452. case OpCodeAsmJs::Simd128_StSlot_U4:
  453. Output::Print(_u(" R%d[%d] = U4_%d"), data->Instance, data->SlotIndex, data->Value);
  454. break;
  455. case OpCodeAsmJs::Simd128_StSlot_U8:
  456. Output::Print(_u(" R%d[%d] = U8_%d"), data->Instance, data->SlotIndex, data->Value);
  457. break;
  458. case OpCodeAsmJs::Simd128_StSlot_U16:
  459. Output::Print(_u(" R%d[%d] = U16_%d"), data->Instance, data->SlotIndex, data->Value);
  460. break;
  461. case OpCodeAsmJs::Simd128_StSlot_B4:
  462. Output::Print(_u(" R%d[%d] = B4_%d"), data->Instance, data->SlotIndex, data->Value);
  463. break;
  464. case OpCodeAsmJs::Simd128_StSlot_B8:
  465. Output::Print(_u(" R%d[%d] = B8_%d"), data->Instance, data->SlotIndex, data->Value);
  466. break;
  467. case OpCodeAsmJs::Simd128_StSlot_B16:
  468. Output::Print(_u(" R%d[%d] = B16_%d"), data->Instance, data->SlotIndex, data->Value);
  469. break;
  470. #if 0
  471. case OpCodeAsmJs::Simd128_StSlot_D2:
  472. Output::Print(_u(" R%d[%d] = D2_%d"), data->Instance, data->SlotIndex, data->Value);
  473. break;
  474. #endif // 0
  475. default:
  476. {
  477. AssertMsg(false, "Unknown OpCode for OpLayoutElementSlot");
  478. break;
  479. }
  480. }
  481. }
  482. void AsmJsByteCodeDumper::InitializeWAsmJsMemTag(ArrayBufferView::ViewType type, _Out_ WAsmJsMemTag * tag)
  483. {
  484. switch (type)
  485. {
  486. case ArrayBufferView::TYPE_INT8:
  487. tag->heapTag = _u("HEAP8"); tag->valueTag = 'I'; break;
  488. case ArrayBufferView::TYPE_UINT8:
  489. tag->heapTag = _u("HEAPU8"); tag->valueTag = 'U'; break;
  490. case ArrayBufferView::TYPE_INT16:
  491. tag->heapTag = _u("HEAP16"); tag->valueTag = 'I'; break;
  492. case ArrayBufferView::TYPE_UINT16:
  493. tag->heapTag = _u("HEAPU16"); tag->valueTag = 'U'; break;
  494. case ArrayBufferView::TYPE_INT32:
  495. tag->heapTag = _u("HEAP32"); tag->valueTag = 'I'; break;
  496. case ArrayBufferView::TYPE_UINT32:
  497. tag->heapTag = _u("HEAPU32"); tag->valueTag = 'U'; break;
  498. case ArrayBufferView::TYPE_FLOAT32:
  499. tag->heapTag = _u("HEAPF32"); tag->valueTag = 'F'; break;
  500. case ArrayBufferView::TYPE_FLOAT64:
  501. tag->heapTag = _u("HEAPF64"); tag->valueTag = 'D'; break;
  502. case ArrayBufferView::TYPE_INT64:
  503. tag->heapTag = _u("HEAPI64"); tag->valueTag = 'L'; break;
  504. case ArrayBufferView::TYPE_INT8_TO_INT64:
  505. tag->heapTag = _u("HEAP8"); tag->valueTag = 'L'; break;
  506. case ArrayBufferView::TYPE_UINT8_TO_INT64:
  507. tag->heapTag = _u("HEAPU8"); tag->valueTag = 'L'; break;
  508. case ArrayBufferView::TYPE_INT16_TO_INT64:
  509. tag->heapTag = _u("HEAP16"); tag->valueTag = 'L'; break;
  510. case ArrayBufferView::TYPE_UINT16_TO_INT64:
  511. tag->heapTag = _u("HEAPU16"); tag->valueTag = 'L'; break;
  512. case ArrayBufferView::TYPE_INT32_TO_INT64:
  513. tag->heapTag = _u("HEAP32"); tag->valueTag = 'L'; break;
  514. case ArrayBufferView::TYPE_UINT32_TO_INT64:
  515. tag->heapTag = _u("HEAPU32"); tag->valueTag = 'L'; break;
  516. default:
  517. Assume(UNREACHED);
  518. }
  519. }
  520. template <class T>
  521. void AsmJsByteCodeDumper::DumpAsmTypedArr(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)
  522. {
  523. WAsmJsMemTag tag;
  524. InitializeWAsmJsMemTag(data->ViewType, &tag);
  525. switch (op)
  526. {
  527. case OpCodeAsmJs::LdArr:
  528. Output::Print(_u(" %c%d = %s[I%d]"), tag.valueTag, data->Value, tag.heapTag, data->SlotIndex); break;
  529. case OpCodeAsmJs::LdArrConst:
  530. Output::Print(_u(" %c%d = %s[%d]"), tag.valueTag, data->Value, tag.heapTag, data->SlotIndex); break;
  531. case OpCodeAsmJs::StArr:
  532. Output::Print(_u(" %s[I%d] = %c%d"), tag.heapTag, data->SlotIndex, tag.valueTag, data->Value); break;
  533. case OpCodeAsmJs::StArrConst:
  534. Output::Print(_u(" %s[%d] = %c%d"), tag.heapTag, data->SlotIndex, tag.valueTag, data->Value); break;
  535. default:
  536. Assume(UNREACHED);
  537. }
  538. }
  539. template <class T>
  540. void AsmJsByteCodeDumper::DumpWasmMemAccess(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)
  541. {
  542. WAsmJsMemTag tag;
  543. InitializeWAsmJsMemTag(data->ViewType, &tag);
  544. switch (op)
  545. {
  546. case OpCodeAsmJs::LdArrWasm:
  547. Output::Print(_u(" %c%d = %s[I%d + %d]"), tag.valueTag, data->Value, tag.heapTag, data->SlotIndex, data->Offset); break;
  548. case OpCodeAsmJs::StArrWasm:
  549. Output::Print(_u(" %s[I%d + %d] = %c%d"), tag.heapTag, data->SlotIndex, data->Offset, tag.valueTag, data->Value); break;
  550. default:
  551. Assume(UNREACHED);
  552. }
  553. }
  554. void AsmJsByteCodeDumper::DumpStartCall(OpCodeAsmJs op, const unaligned OpLayoutStartCall* data, FunctionBody * dumpFunction, ByteCodeReader& reader)
  555. {
  556. Assert(op == OpCodeAsmJs::StartCall || op == OpCodeAsmJs::I_StartCall);
  557. Output::Print(_u(" ArgSize: %d"), data->ArgCount);
  558. }
  559. template <class T>
  560. void AsmJsByteCodeDumper::DumpAsmCall(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)
  561. {
  562. if (data->Return != Constants::NoRegister)
  563. {
  564. DumpReg((RegSlot)data->Return);
  565. Output::Print(_u("="));
  566. }
  567. Output::Print(_u(" R%d(ArgCount: %d)"), data->Function, data->ArgCount);
  568. }
  569. template <class T>
  570. void AsmJsByteCodeDumper::DumpProfiledAsmCall(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)
  571. {
  572. if (data->Return != Constants::NoRegister)
  573. {
  574. DumpReg((RegSlot)data->Return);
  575. Output::Print(_u("="));
  576. }
  577. Output::Print(_u(" R%d(ArgCount: %d, profileId: %d)"), data->Function, data->ArgCount, data->profileId);
  578. }
  579. template <class T>
  580. void AsmJsByteCodeDumper::DumpAsmUnsigned1(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)
  581. {
  582. DumpU4(data->C1);
  583. }
  584. template <class T>
  585. void AsmJsByteCodeDumper::DumpWasmLoopStart(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)
  586. {
  587. DumpU4(data->loopId);
  588. }
  589. void AsmJsByteCodeDumper::DumpEmpty(OpCodeAsmJs op, const unaligned OpLayoutEmpty* data, FunctionBody * dumpFunction, ByteCodeReader& reader)
  590. {
  591. // empty
  592. }
  593. void AsmJsByteCodeDumper::DumpAsmBr(OpCodeAsmJs op, const unaligned OpLayoutAsmBr* data, FunctionBody * dumpFunction, ByteCodeReader& reader)
  594. {
  595. DumpOffset(data->RelativeJumpOffset, reader);
  596. }
  597. template <class T>
  598. void AsmJsByteCodeDumper::DumpAsmReg1(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)
  599. {
  600. DumpReg(data->R0);
  601. }
  602. template <class T>
  603. void AsmJsByteCodeDumper::DumpAsmReg2(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)
  604. {
  605. DumpReg(data->R0);
  606. DumpReg(data->R1);
  607. }
  608. template <class T>
  609. void AsmJsByteCodeDumper::DumpAsmReg3(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)
  610. {
  611. DumpReg(data->R0);
  612. DumpReg(data->R1);
  613. DumpReg(data->R2);
  614. }
  615. template <class T>
  616. void AsmJsByteCodeDumper::DumpAsmReg4(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)
  617. {
  618. DumpReg(data->R0);
  619. DumpReg(data->R1);
  620. DumpReg(data->R2);
  621. DumpReg(data->R3);
  622. }
  623. template <class T>
  624. void AsmJsByteCodeDumper::DumpAsmReg5(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)
  625. {
  626. DumpReg(data->R0);
  627. DumpReg(data->R1);
  628. DumpReg(data->R2);
  629. DumpReg(data->R3);
  630. DumpReg(data->R4);
  631. }
  632. template <class T>
  633. void AsmJsByteCodeDumper::DumpAsmReg6(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)
  634. {
  635. DumpReg(data->R0);
  636. DumpReg(data->R1);
  637. DumpReg(data->R2);
  638. DumpReg(data->R3);
  639. DumpReg(data->R4);
  640. DumpReg(data->R5);
  641. }
  642. template <class T>
  643. void AsmJsByteCodeDumper::DumpAsmReg7(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)
  644. {
  645. DumpReg(data->R0);
  646. DumpReg(data->R1);
  647. DumpReg(data->R2);
  648. DumpReg(data->R3);
  649. DumpReg(data->R4);
  650. DumpReg(data->R5);
  651. DumpReg(data->R6);
  652. }
  653. template <class T>
  654. void AsmJsByteCodeDumper::DumpAsmReg9(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)
  655. {
  656. DumpReg(data->R0);
  657. DumpReg(data->R1);
  658. DumpReg(data->R2);
  659. DumpReg(data->R3);
  660. DumpReg(data->R4);
  661. DumpReg(data->R5);
  662. DumpReg(data->R6);
  663. DumpReg(data->R7);
  664. DumpReg(data->R8);
  665. }
  666. template <class T>
  667. void AsmJsByteCodeDumper::DumpAsmReg10(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)
  668. {
  669. DumpReg(data->R0);
  670. DumpReg(data->R1);
  671. DumpReg(data->R2);
  672. DumpReg(data->R3);
  673. DumpReg(data->R4);
  674. DumpReg(data->R5);
  675. DumpReg(data->R6);
  676. DumpReg(data->R7);
  677. DumpReg(data->R8);
  678. DumpReg(data->R9);
  679. }
  680. template <class T>
  681. void AsmJsByteCodeDumper::DumpAsmReg11(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)
  682. {
  683. DumpReg(data->R0);
  684. DumpReg(data->R1);
  685. DumpReg(data->R2);
  686. DumpReg(data->R3);
  687. DumpReg(data->R4);
  688. DumpReg(data->R5);
  689. DumpReg(data->R6);
  690. DumpReg(data->R7);
  691. DumpReg(data->R8);
  692. DumpReg(data->R9);
  693. DumpReg(data->R10);
  694. }
  695. template <class T>
  696. void AsmJsByteCodeDumper::DumpAsmReg17(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)
  697. {
  698. DumpReg(data->R0);
  699. DumpReg(data->R1);
  700. DumpReg(data->R2);
  701. DumpReg(data->R3);
  702. DumpReg(data->R4);
  703. DumpReg(data->R5);
  704. DumpReg(data->R6);
  705. DumpReg(data->R7);
  706. DumpReg(data->R8);
  707. DumpReg(data->R9);
  708. DumpReg(data->R10);
  709. DumpReg(data->R11);
  710. DumpReg(data->R12);
  711. DumpReg(data->R13);
  712. DumpReg(data->R14);
  713. DumpReg(data->R15);
  714. DumpReg(data->R16);
  715. }
  716. template <class T>
  717. void AsmJsByteCodeDumper::DumpAsmReg18(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)
  718. {
  719. DumpReg(data->R0);
  720. DumpReg(data->R1);
  721. DumpReg(data->R2);
  722. DumpReg(data->R3);
  723. DumpReg(data->R4);
  724. DumpReg(data->R5);
  725. DumpReg(data->R6);
  726. DumpReg(data->R7);
  727. DumpReg(data->R8);
  728. DumpReg(data->R9);
  729. DumpReg(data->R10);
  730. DumpReg(data->R11);
  731. DumpReg(data->R12);
  732. DumpReg(data->R13);
  733. DumpReg(data->R14);
  734. DumpReg(data->R15);
  735. DumpReg(data->R16);
  736. DumpReg(data->R17);
  737. }
  738. template <class T>
  739. void AsmJsByteCodeDumper::DumpAsmReg19(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)
  740. {
  741. DumpReg(data->R0);
  742. DumpReg(data->R1);
  743. DumpReg(data->R2);
  744. DumpReg(data->R3);
  745. DumpReg(data->R4);
  746. DumpReg(data->R5);
  747. DumpReg(data->R6);
  748. DumpReg(data->R7);
  749. DumpReg(data->R8);
  750. DumpReg(data->R9);
  751. DumpReg(data->R10);
  752. DumpReg(data->R11);
  753. DumpReg(data->R12);
  754. DumpReg(data->R13);
  755. DumpReg(data->R14);
  756. DumpReg(data->R15);
  757. DumpReg(data->R16);
  758. DumpReg(data->R17);
  759. DumpReg(data->R18);
  760. }
  761. #define LAYOUT_TYPE_WMS_REG2(layout, t0, t1) \
  762. template <class T> void AsmJsByteCodeDumper::Dump##layout(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)\
  763. {\
  764. Dump##t0##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t0(), 0));\
  765. Dump##t1##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t1(), 1));\
  766. }
  767. #define LAYOUT_TYPE_WMS_REG3(layout, t0, t1, t2) \
  768. template <class T> void AsmJsByteCodeDumper::Dump##layout(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)\
  769. {\
  770. Dump##t0##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t0(), 0));\
  771. Dump##t1##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t1(), 1));\
  772. Dump##t2##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t2(), 2));\
  773. }
  774. #define LAYOUT_TYPE_WMS_REG4(layout, t0, t1, t2, t3)\
  775. template <class T> void AsmJsByteCodeDumper::Dump##layout(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)\
  776. {\
  777. Dump##t0##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t0(), 0));\
  778. Dump##t1##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t1(), 1));\
  779. Dump##t2##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t2(), 2));\
  780. Dump##t3##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t3(), 3));\
  781. };
  782. #define LAYOUT_TYPE_WMS_REG5(layout, t0, t1, t2, t3, t4)\
  783. template <class T> void AsmJsByteCodeDumper::Dump##layout(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)\
  784. {\
  785. Dump##t0##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t0(), 0));\
  786. Dump##t1##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t1(), 1));\
  787. Dump##t2##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t2(), 2));\
  788. Dump##t3##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t3(), 3));\
  789. Dump##t4##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t4(), 4));\
  790. };
  791. #define LAYOUT_TYPE_WMS_REG6(layout, t0, t1, t2, t3, t4, t5)\
  792. template <class T> void AsmJsByteCodeDumper::Dump##layout(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)\
  793. {\
  794. Dump##t0##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t0(), 0));\
  795. Dump##t1##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t1(), 1));\
  796. Dump##t2##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t2(), 2));\
  797. Dump##t3##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t3(), 3));\
  798. Dump##t4##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t4(), 4));\
  799. Dump##t5##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t5(), 5));\
  800. };
  801. #define LAYOUT_TYPE_WMS_REG7(layout, t0, t1, t2, t3, t4, t5, t6)\
  802. template <class T> void AsmJsByteCodeDumper::Dump##layout(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)\
  803. {\
  804. Dump##t0##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t0(), 0));\
  805. Dump##t1##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t1(), 1));\
  806. Dump##t2##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t2(), 2));\
  807. Dump##t3##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t3(), 3));\
  808. Dump##t4##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t4(), 4));\
  809. Dump##t5##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t5(), 5));\
  810. Dump##t6##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t6(), 6));\
  811. };
  812. #define LAYOUT_TYPE_WMS_REG9(layout, t0, t1, t2, t3, t4, t5, t6, t7, t8)\
  813. template <class T> void AsmJsByteCodeDumper::Dump##layout(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)\
  814. {\
  815. Dump##t0##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t0(), 0));\
  816. Dump##t1##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t1(), 1));\
  817. Dump##t2##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t2(), 2));\
  818. Dump##t3##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t3(), 3));\
  819. Dump##t4##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t4(), 4));\
  820. Dump##t5##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t5(), 5));\
  821. Dump##t6##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t6(), 6));\
  822. Dump##t7##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t7(), 7));\
  823. Dump##t8##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t8(), 8));\
  824. };
  825. #define LAYOUT_TYPE_WMS_REG10(layout, t0, t1, t2, t3, t4, t5, t6, t7, t8, t9)\
  826. template <class T> void AsmJsByteCodeDumper::Dump##layout(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)\
  827. {\
  828. Dump##t0##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t0(), 0));\
  829. Dump##t1##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t1(), 1));\
  830. Dump##t2##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t2(), 2));\
  831. Dump##t3##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t3(), 3));\
  832. Dump##t4##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t4(), 4));\
  833. Dump##t5##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t5(), 5));\
  834. Dump##t6##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t6(), 6));\
  835. Dump##t7##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t7(), 7));\
  836. Dump##t8##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t8(), 8));\
  837. Dump##t9##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t9(), 9));\
  838. };
  839. #define LAYOUT_TYPE_WMS_REG11(layout, t0, t1, t2, t3, t4, t5, t6, t7, t8, t9, t10)\
  840. template <class T> void AsmJsByteCodeDumper::Dump##layout(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)\
  841. {\
  842. Dump##t0##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t0(), 0));\
  843. Dump##t1##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t1(), 1));\
  844. Dump##t2##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t2(), 2));\
  845. Dump##t3##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t3(), 3));\
  846. Dump##t4##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t4(), 4));\
  847. Dump##t5##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t5(), 5));\
  848. Dump##t6##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t6(), 6));\
  849. Dump##t7##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t7(), 7));\
  850. Dump##t8##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t8(), 8));\
  851. Dump##t9##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t9(), 9));\
  852. Dump##t10##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t10(), 10));\
  853. };
  854. #define LAYOUT_TYPE_WMS_REG17(layout, t0, t1, t2, t3, t4, t5, t6, t7, t8, t9, t10, t11, t12, t13, t14, t15, t16)\
  855. template <class T> void AsmJsByteCodeDumper::Dump##layout(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)\
  856. {\
  857. Dump##t0##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t0(), 0));\
  858. Dump##t1##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t1(), 1));\
  859. Dump##t2##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t2(), 2));\
  860. Dump##t3##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t3(), 3));\
  861. Dump##t4##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t4(), 4));\
  862. Dump##t5##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t5(), 5));\
  863. Dump##t6##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t6(), 6));\
  864. Dump##t7##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t7(), 7));\
  865. Dump##t8##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t8(), 8));\
  866. Dump##t9##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t9(), 9));\
  867. Dump##t10##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t10(), 10));\
  868. Dump##t11##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t11(), 11));\
  869. Dump##t12##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t12(), 12));\
  870. Dump##t13##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t13(), 13));\
  871. Dump##t14##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t14(), 14));\
  872. Dump##t15##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t15(), 15));\
  873. Dump##t16##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t16(), 16));\
  874. };
  875. #define LAYOUT_TYPE_WMS_REG18(layout, t0, t1, t2, t3, t4, t5, t6, t7, t8, t9, t10, t11, t12, t13, t14, t15, t16, t17)\
  876. template <class T> void AsmJsByteCodeDumper::Dump##layout(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)\
  877. {\
  878. Dump##t0##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t0(), 0));\
  879. Dump##t1##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t1(), 1));\
  880. Dump##t2##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t2(), 2));\
  881. Dump##t3##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t3(), 3));\
  882. Dump##t4##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t4(), 4));\
  883. Dump##t5##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t5(), 5));\
  884. Dump##t6##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t6(), 6));\
  885. Dump##t7##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t7(), 7));\
  886. Dump##t8##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t8(), 8));\
  887. Dump##t9##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t9(), 9));\
  888. Dump##t10##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t10(), 10));\
  889. Dump##t11##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t11(), 11));\
  890. Dump##t12##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t12(), 12));\
  891. Dump##t13##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t13(), 13));\
  892. Dump##t14##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t14(), 14));\
  893. Dump##t15##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t15(), 15));\
  894. Dump##t16##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t16(), 16));\
  895. Dump##t17##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t17(), 17));\
  896. };
  897. #define LAYOUT_TYPE_WMS_REG19(layout, t0, t1, t2, t3, t4, t5, t6, t7, t8, t9, t10, t11, t12, t13, t14, t15, t16, t17, t18)\
  898. template <class T> void AsmJsByteCodeDumper::Dump##layout(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)\
  899. {\
  900. Dump##t0##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t0(), 0));\
  901. Dump##t1##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t1(), 1));\
  902. Dump##t2##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t2(), 2));\
  903. Dump##t3##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t3(), 3));\
  904. Dump##t4##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t4(), 4));\
  905. Dump##t5##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t5(), 5));\
  906. Dump##t6##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t6(), 6));\
  907. Dump##t7##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t7(), 7));\
  908. Dump##t8##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t8(), 8));\
  909. Dump##t9##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t9(), 9));\
  910. Dump##t10##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t10(), 10));\
  911. Dump##t11##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t11(), 11));\
  912. Dump##t12##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t12(), 12));\
  913. Dump##t13##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t13(), 13));\
  914. Dump##t14##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t14(), 14));\
  915. Dump##t15##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t15(), 15));\
  916. Dump##t16##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t16(), 16));\
  917. Dump##t17##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t17(), 17));\
  918. Dump##t18##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t18(), 18));\
  919. };
  920. #include "LayoutTypesAsmJs.h"
  921. template <class T>
  922. void AsmJsByteCodeDumper::DumpBrInt1(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)
  923. {
  924. DumpOffset(data->RelativeJumpOffset, reader);
  925. DumpIntReg(data->I1);
  926. }
  927. template <class T>
  928. void AsmJsByteCodeDumper::DumpBrInt2(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)
  929. {
  930. DumpOffset(data->RelativeJumpOffset, reader);
  931. DumpIntReg(data->I1);
  932. DumpIntReg(data->I2);
  933. }
  934. template <class T>
  935. void AsmJsByteCodeDumper::DumpBrInt1Const1(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)
  936. {
  937. DumpOffset(data->RelativeJumpOffset, reader);
  938. DumpIntReg(data->I1);
  939. DumpI4(data->C1);
  940. }
  941. template <class T>
  942. void AsmJsByteCodeDumper::DumpAsmShuffle(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)
  943. {
  944. DumpFloat32x4Reg(data->R0);
  945. DumpFloat32x4Reg(data->R1);
  946. DumpFloat32x4Reg(data->R2);
  947. const uint NUM_LANES = 16;
  948. for (uint i = 0; i < NUM_LANES; i++)
  949. {
  950. DumpU4(data->INDICES[i]);
  951. }
  952. }
  953. template <class T>
  954. void AsmJsByteCodeDumper::DumpAsmSimdTypedArr(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)
  955. {
  956. const char16* heapTag = nullptr;
  957. switch (data->ViewType)
  958. {
  959. case ArrayBufferView::TYPE_INT8:
  960. case ArrayBufferView::TYPE_INT8_TO_INT64:
  961. heapTag = _u("HEAP8"); break;
  962. case ArrayBufferView::TYPE_UINT8:
  963. case ArrayBufferView::TYPE_UINT8_TO_INT64:
  964. heapTag = _u("HEAPU8"); break;
  965. case ArrayBufferView::TYPE_INT16:
  966. case ArrayBufferView::TYPE_INT16_TO_INT64:
  967. heapTag = _u("HEAP16"); break;
  968. case ArrayBufferView::TYPE_UINT16:
  969. case ArrayBufferView::TYPE_UINT16_TO_INT64:
  970. heapTag = _u("HEAPU16"); break;
  971. case ArrayBufferView::TYPE_INT32:
  972. case ArrayBufferView::TYPE_INT32_TO_INT64:
  973. heapTag = _u("HEAP32"); break;
  974. case ArrayBufferView::TYPE_UINT32:
  975. case ArrayBufferView::TYPE_UINT32_TO_INT64:
  976. heapTag = _u("HEAPU32"); break;
  977. case ArrayBufferView::TYPE_FLOAT32:
  978. heapTag = _u("HEAPF32"); break;
  979. case ArrayBufferView::TYPE_FLOAT64:
  980. heapTag = _u("HEAPF64"); break;
  981. case ArrayBufferView::TYPE_INT64:
  982. heapTag = _u("HEAPI64"); break;
  983. default:
  984. Assert(false);
  985. __assume(false);
  986. break;
  987. }
  988. #define SIMD_DUMP_ARR_I4 DumpInt32x4Reg
  989. #define SIMD_DUMP_ARR_I8 DumpInt16x8Reg
  990. #define SIMD_DUMP_ARR_I16 DumpInt8x16Reg
  991. #define SIMD_DUMP_ARR_U4 DumpUint32x4Reg
  992. #define SIMD_DUMP_ARR_U8 DumpUint16x8Reg
  993. #define SIMD_DUMP_ARR_U16 DumpUint8x16Reg
  994. #define SIMD_DUMP_ARR_F4 DumpFloat32x4Reg
  995. #define SIMD_DUMP_ARR_D2 DumpFloat64x2Reg
  996. #define SIMD_DUMP_ARR_I2 DumpInt64x2Reg
  997. #define SIMD_DUMP_REG(type) SIMD_DUMP_ARR_##type(data->Value)
  998. #define SIMD_DUMP_ARR_VALUE(type) \
  999. case OpCodeAsmJs::Simd128_LdArr_##type:\
  1000. SIMD_DUMP_REG(type); Output::Print(_u("= %s[I%d] "), heapTag, data->SlotIndex); break;\
  1001. case OpCodeAsmJs::Simd128_LdArrConst_##type:\
  1002. SIMD_DUMP_REG(type); Output::Print(_u("= %s[%d] "), heapTag, data->SlotIndex); break;\
  1003. case OpCodeAsmJs::Simd128_StArr_##type:\
  1004. Output::Print(_u("%s[I%d] = "), heapTag, data->SlotIndex); SIMD_DUMP_REG(type); break;\
  1005. case OpCodeAsmJs::Simd128_StArrConst_##type:\
  1006. Output::Print(_u("%s[%d] = "), heapTag, data->SlotIndex); SIMD_DUMP_REG(type); break;\
  1007. switch (op)
  1008. {
  1009. SIMD_DUMP_ARR_VALUE(I4)
  1010. SIMD_DUMP_ARR_VALUE(I8)
  1011. SIMD_DUMP_ARR_VALUE(I16)
  1012. SIMD_DUMP_ARR_VALUE(U4)
  1013. SIMD_DUMP_ARR_VALUE(U8)
  1014. SIMD_DUMP_ARR_VALUE(U16)
  1015. SIMD_DUMP_ARR_VALUE(F4)
  1016. //SIMD_DUMP_ARR_VALUE(D2)
  1017. default:
  1018. Assert(false);
  1019. __assume(false);
  1020. break;
  1021. }
  1022. // data width
  1023. Output::Print(_u(" %d bytes "), data->DataWidth);
  1024. }
  1025. }
  1026. #endif
  1027. #endif