AsmJsByteCodeDumper.cpp 45 KB

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