AsmJsByteCodeDumper.cpp 46 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085
  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. case OpCodeAsmJs::LdArrAtomic:
  548. Output::Print(_u(" %c%d = %s[I%d + %d]"), tag.valueTag, data->Value, tag.heapTag, data->SlotIndex, data->Offset); break;
  549. case OpCodeAsmJs::StArrWasm:
  550. case OpCodeAsmJs::StArrAtomic:
  551. Output::Print(_u(" %s[I%d + %d] = %c%d"), tag.heapTag, data->SlotIndex, data->Offset, tag.valueTag, data->Value); break;
  552. default:
  553. Assume(UNREACHED);
  554. }
  555. }
  556. void AsmJsByteCodeDumper::DumpStartCall(OpCodeAsmJs op, const unaligned OpLayoutStartCall* data, FunctionBody * dumpFunction, ByteCodeReader& reader)
  557. {
  558. Assert(op == OpCodeAsmJs::StartCall || op == OpCodeAsmJs::I_StartCall);
  559. Output::Print(_u(" ArgSize: %d"), data->ArgCount);
  560. }
  561. template <class T>
  562. void AsmJsByteCodeDumper::DumpAsmCall(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)"), data->Function, data->ArgCount);
  570. }
  571. template <class T>
  572. void AsmJsByteCodeDumper::DumpProfiledAsmCall(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)
  573. {
  574. if (data->Return != Constants::NoRegister)
  575. {
  576. DumpReg((RegSlot)data->Return);
  577. Output::Print(_u("="));
  578. }
  579. Output::Print(_u(" R%d(ArgCount: %d, profileId: %d)"), data->Function, data->ArgCount, data->profileId);
  580. }
  581. template <class T>
  582. void AsmJsByteCodeDumper::DumpAsmUnsigned1(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)
  583. {
  584. DumpU4(data->C1);
  585. }
  586. template <class T>
  587. void AsmJsByteCodeDumper::DumpWasmLoopStart(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)
  588. {
  589. DumpU4(data->loopId);
  590. }
  591. void AsmJsByteCodeDumper::DumpEmpty(OpCodeAsmJs op, const unaligned OpLayoutEmpty* data, FunctionBody * dumpFunction, ByteCodeReader& reader)
  592. {
  593. // empty
  594. }
  595. void AsmJsByteCodeDumper::DumpAsmBr(OpCodeAsmJs op, const unaligned OpLayoutAsmBr* data, FunctionBody * dumpFunction, ByteCodeReader& reader)
  596. {
  597. DumpOffset(data->RelativeJumpOffset, reader);
  598. }
  599. template <class T>
  600. void AsmJsByteCodeDumper::DumpAsmReg1(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)
  601. {
  602. DumpReg(data->R0);
  603. }
  604. template <class T>
  605. void AsmJsByteCodeDumper::DumpAsmReg2(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)
  606. {
  607. DumpReg(data->R0);
  608. DumpReg(data->R1);
  609. }
  610. template <class T>
  611. void AsmJsByteCodeDumper::DumpAsmReg3(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)
  612. {
  613. DumpReg(data->R0);
  614. DumpReg(data->R1);
  615. DumpReg(data->R2);
  616. }
  617. template <class T>
  618. void AsmJsByteCodeDumper::DumpAsmReg4(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)
  619. {
  620. DumpReg(data->R0);
  621. DumpReg(data->R1);
  622. DumpReg(data->R2);
  623. DumpReg(data->R3);
  624. }
  625. template <class T>
  626. void AsmJsByteCodeDumper::DumpAsmReg5(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)
  627. {
  628. DumpReg(data->R0);
  629. DumpReg(data->R1);
  630. DumpReg(data->R2);
  631. DumpReg(data->R3);
  632. DumpReg(data->R4);
  633. }
  634. template <class T>
  635. void AsmJsByteCodeDumper::DumpAsmReg6(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. }
  644. template <class T>
  645. void AsmJsByteCodeDumper::DumpAsmReg7(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)
  646. {
  647. DumpReg(data->R0);
  648. DumpReg(data->R1);
  649. DumpReg(data->R2);
  650. DumpReg(data->R3);
  651. DumpReg(data->R4);
  652. DumpReg(data->R5);
  653. DumpReg(data->R6);
  654. }
  655. template <class T>
  656. void AsmJsByteCodeDumper::DumpAsmReg9(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)
  657. {
  658. DumpReg(data->R0);
  659. DumpReg(data->R1);
  660. DumpReg(data->R2);
  661. DumpReg(data->R3);
  662. DumpReg(data->R4);
  663. DumpReg(data->R5);
  664. DumpReg(data->R6);
  665. DumpReg(data->R7);
  666. DumpReg(data->R8);
  667. }
  668. template <class T>
  669. void AsmJsByteCodeDumper::DumpAsmReg10(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)
  670. {
  671. DumpReg(data->R0);
  672. DumpReg(data->R1);
  673. DumpReg(data->R2);
  674. DumpReg(data->R3);
  675. DumpReg(data->R4);
  676. DumpReg(data->R5);
  677. DumpReg(data->R6);
  678. DumpReg(data->R7);
  679. DumpReg(data->R8);
  680. DumpReg(data->R9);
  681. }
  682. template <class T>
  683. void AsmJsByteCodeDumper::DumpAsmReg11(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)
  684. {
  685. DumpReg(data->R0);
  686. DumpReg(data->R1);
  687. DumpReg(data->R2);
  688. DumpReg(data->R3);
  689. DumpReg(data->R4);
  690. DumpReg(data->R5);
  691. DumpReg(data->R6);
  692. DumpReg(data->R7);
  693. DumpReg(data->R8);
  694. DumpReg(data->R9);
  695. DumpReg(data->R10);
  696. }
  697. template <class T>
  698. void AsmJsByteCodeDumper::DumpAsmReg17(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)
  699. {
  700. DumpReg(data->R0);
  701. DumpReg(data->R1);
  702. DumpReg(data->R2);
  703. DumpReg(data->R3);
  704. DumpReg(data->R4);
  705. DumpReg(data->R5);
  706. DumpReg(data->R6);
  707. DumpReg(data->R7);
  708. DumpReg(data->R8);
  709. DumpReg(data->R9);
  710. DumpReg(data->R10);
  711. DumpReg(data->R11);
  712. DumpReg(data->R12);
  713. DumpReg(data->R13);
  714. DumpReg(data->R14);
  715. DumpReg(data->R15);
  716. DumpReg(data->R16);
  717. }
  718. template <class T>
  719. void AsmJsByteCodeDumper::DumpAsmReg18(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)
  720. {
  721. DumpReg(data->R0);
  722. DumpReg(data->R1);
  723. DumpReg(data->R2);
  724. DumpReg(data->R3);
  725. DumpReg(data->R4);
  726. DumpReg(data->R5);
  727. DumpReg(data->R6);
  728. DumpReg(data->R7);
  729. DumpReg(data->R8);
  730. DumpReg(data->R9);
  731. DumpReg(data->R10);
  732. DumpReg(data->R11);
  733. DumpReg(data->R12);
  734. DumpReg(data->R13);
  735. DumpReg(data->R14);
  736. DumpReg(data->R15);
  737. DumpReg(data->R16);
  738. DumpReg(data->R17);
  739. }
  740. template <class T>
  741. void AsmJsByteCodeDumper::DumpAsmReg19(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)
  742. {
  743. DumpReg(data->R0);
  744. DumpReg(data->R1);
  745. DumpReg(data->R2);
  746. DumpReg(data->R3);
  747. DumpReg(data->R4);
  748. DumpReg(data->R5);
  749. DumpReg(data->R6);
  750. DumpReg(data->R7);
  751. DumpReg(data->R8);
  752. DumpReg(data->R9);
  753. DumpReg(data->R10);
  754. DumpReg(data->R11);
  755. DumpReg(data->R12);
  756. DumpReg(data->R13);
  757. DumpReg(data->R14);
  758. DumpReg(data->R15);
  759. DumpReg(data->R16);
  760. DumpReg(data->R17);
  761. DumpReg(data->R18);
  762. }
  763. #define LAYOUT_TYPE_WMS_REG2(layout, t0, t1) \
  764. template <class T> void AsmJsByteCodeDumper::Dump##layout(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)\
  765. {\
  766. Dump##t0##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t0(), 0));\
  767. Dump##t1##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t1(), 1));\
  768. }
  769. #define LAYOUT_TYPE_WMS_REG3(layout, t0, t1, t2) \
  770. template <class T> void AsmJsByteCodeDumper::Dump##layout(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)\
  771. {\
  772. Dump##t0##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t0(), 0));\
  773. Dump##t1##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t1(), 1));\
  774. Dump##t2##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t2(), 2));\
  775. }
  776. #define LAYOUT_TYPE_WMS_REG4(layout, t0, t1, t2, t3)\
  777. template <class T> void AsmJsByteCodeDumper::Dump##layout(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)\
  778. {\
  779. Dump##t0##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t0(), 0));\
  780. Dump##t1##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t1(), 1));\
  781. Dump##t2##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t2(), 2));\
  782. Dump##t3##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t3(), 3));\
  783. };
  784. #define LAYOUT_TYPE_WMS_REG5(layout, t0, t1, t2, t3, t4)\
  785. template <class T> void AsmJsByteCodeDumper::Dump##layout(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)\
  786. {\
  787. Dump##t0##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t0(), 0));\
  788. Dump##t1##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t1(), 1));\
  789. Dump##t2##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t2(), 2));\
  790. Dump##t3##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t3(), 3));\
  791. Dump##t4##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t4(), 4));\
  792. };
  793. #define LAYOUT_TYPE_WMS_REG6(layout, t0, t1, t2, t3, t4, t5)\
  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. };
  803. #define LAYOUT_TYPE_WMS_REG7(layout, t0, t1, t2, t3, t4, t5, t6)\
  804. template <class T> void AsmJsByteCodeDumper::Dump##layout(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)\
  805. {\
  806. Dump##t0##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t0(), 0));\
  807. Dump##t1##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t1(), 1));\
  808. Dump##t2##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t2(), 2));\
  809. Dump##t3##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t3(), 3));\
  810. Dump##t4##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t4(), 4));\
  811. Dump##t5##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t5(), 5));\
  812. Dump##t6##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t6(), 6));\
  813. };
  814. #define LAYOUT_TYPE_WMS_REG9(layout, t0, t1, t2, t3, t4, t5, t6, t7, t8)\
  815. template <class T> void AsmJsByteCodeDumper::Dump##layout(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)\
  816. {\
  817. Dump##t0##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t0(), 0));\
  818. Dump##t1##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t1(), 1));\
  819. Dump##t2##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t2(), 2));\
  820. Dump##t3##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t3(), 3));\
  821. Dump##t4##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t4(), 4));\
  822. Dump##t5##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t5(), 5));\
  823. Dump##t6##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t6(), 6));\
  824. Dump##t7##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t7(), 7));\
  825. Dump##t8##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t8(), 8));\
  826. };
  827. #define LAYOUT_TYPE_WMS_REG10(layout, t0, t1, t2, t3, t4, t5, t6, t7, t8, t9)\
  828. template <class T> void AsmJsByteCodeDumper::Dump##layout(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)\
  829. {\
  830. Dump##t0##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t0(), 0));\
  831. Dump##t1##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t1(), 1));\
  832. Dump##t2##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t2(), 2));\
  833. Dump##t3##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t3(), 3));\
  834. Dump##t4##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t4(), 4));\
  835. Dump##t5##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t5(), 5));\
  836. Dump##t6##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t6(), 6));\
  837. Dump##t7##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t7(), 7));\
  838. Dump##t8##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t8(), 8));\
  839. Dump##t9##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t9(), 9));\
  840. };
  841. #define LAYOUT_TYPE_WMS_REG11(layout, t0, t1, t2, t3, t4, t5, t6, t7, t8, t9, t10)\
  842. template <class T> void AsmJsByteCodeDumper::Dump##layout(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)\
  843. {\
  844. Dump##t0##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t0(), 0));\
  845. Dump##t1##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t1(), 1));\
  846. Dump##t2##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t2(), 2));\
  847. Dump##t3##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t3(), 3));\
  848. Dump##t4##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t4(), 4));\
  849. Dump##t5##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t5(), 5));\
  850. Dump##t6##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t6(), 6));\
  851. Dump##t7##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t7(), 7));\
  852. Dump##t8##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t8(), 8));\
  853. Dump##t9##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t9(), 9));\
  854. Dump##t10##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t10(), 10));\
  855. };
  856. #define LAYOUT_TYPE_WMS_REG17(layout, t0, t1, t2, t3, t4, t5, t6, t7, t8, t9, t10, t11, t12, t13, t14, t15, t16)\
  857. template <class T> void AsmJsByteCodeDumper::Dump##layout(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)\
  858. {\
  859. Dump##t0##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t0(), 0));\
  860. Dump##t1##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t1(), 1));\
  861. Dump##t2##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t2(), 2));\
  862. Dump##t3##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t3(), 3));\
  863. Dump##t4##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t4(), 4));\
  864. Dump##t5##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t5(), 5));\
  865. Dump##t6##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t6(), 6));\
  866. Dump##t7##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t7(), 7));\
  867. Dump##t8##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t8(), 8));\
  868. Dump##t9##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t9(), 9));\
  869. Dump##t10##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t10(), 10));\
  870. Dump##t11##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t11(), 11));\
  871. Dump##t12##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t12(), 12));\
  872. Dump##t13##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t13(), 13));\
  873. Dump##t14##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t14(), 14));\
  874. Dump##t15##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t15(), 15));\
  875. Dump##t16##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t16(), 16));\
  876. };
  877. #define LAYOUT_TYPE_WMS_REG18(layout, t0, t1, t2, t3, t4, t5, t6, t7, t8, t9, t10, t11, t12, t13, t14, t15, t16, t17)\
  878. template <class T> void AsmJsByteCodeDumper::Dump##layout(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)\
  879. {\
  880. Dump##t0##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t0(), 0));\
  881. Dump##t1##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t1(), 1));\
  882. Dump##t2##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t2(), 2));\
  883. Dump##t3##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t3(), 3));\
  884. Dump##t4##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t4(), 4));\
  885. Dump##t5##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t5(), 5));\
  886. Dump##t6##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t6(), 6));\
  887. Dump##t7##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t7(), 7));\
  888. Dump##t8##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t8(), 8));\
  889. Dump##t9##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t9(), 9));\
  890. Dump##t10##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t10(), 10));\
  891. Dump##t11##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t11(), 11));\
  892. Dump##t12##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t12(), 12));\
  893. Dump##t13##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t13(), 13));\
  894. Dump##t14##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t14(), 14));\
  895. Dump##t15##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t15(), 15));\
  896. Dump##t16##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t16(), 16));\
  897. Dump##t17##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t17(), 17));\
  898. };
  899. #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)\
  900. template <class T> void AsmJsByteCodeDumper::Dump##layout(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)\
  901. {\
  902. Dump##t0##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t0(), 0));\
  903. Dump##t1##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t1(), 1));\
  904. Dump##t2##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t2(), 2));\
  905. Dump##t3##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t3(), 3));\
  906. Dump##t4##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t4(), 4));\
  907. Dump##t5##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t5(), 5));\
  908. Dump##t6##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t6(), 6));\
  909. Dump##t7##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t7(), 7));\
  910. Dump##t8##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t8(), 8));\
  911. Dump##t9##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t9(), 9));\
  912. Dump##t10##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t10(), 10));\
  913. Dump##t11##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t11(), 11));\
  914. Dump##t12##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t12(), 12));\
  915. Dump##t13##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t13(), 13));\
  916. Dump##t14##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t14(), 14));\
  917. Dump##t15##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t15(), 15));\
  918. Dump##t16##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t16(), 16));\
  919. Dump##t17##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t17(), 17));\
  920. Dump##t18##Reg(data->LAYOUT_FIELDS_DEF(LAYOUT_PREFIX_##t18(), 18));\
  921. };
  922. #include "LayoutTypesAsmJs.h"
  923. template <class T>
  924. void AsmJsByteCodeDumper::DumpBrInt1(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)
  925. {
  926. DumpOffset(data->RelativeJumpOffset, reader);
  927. DumpIntReg(data->I1);
  928. }
  929. template <class T>
  930. void AsmJsByteCodeDumper::DumpBrInt2(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)
  931. {
  932. DumpOffset(data->RelativeJumpOffset, reader);
  933. DumpIntReg(data->I1);
  934. DumpIntReg(data->I2);
  935. }
  936. template <class T>
  937. void AsmJsByteCodeDumper::DumpBrInt1Const1(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)
  938. {
  939. DumpOffset(data->RelativeJumpOffset, reader);
  940. DumpIntReg(data->I1);
  941. DumpI4(data->C1);
  942. }
  943. template <class T>
  944. void AsmJsByteCodeDumper::DumpAsmShuffle(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)
  945. {
  946. DumpFloat32x4Reg(data->R0);
  947. DumpFloat32x4Reg(data->R1);
  948. DumpFloat32x4Reg(data->R2);
  949. const uint NUM_LANES = 16;
  950. for (uint i = 0; i < NUM_LANES; i++)
  951. {
  952. DumpU4(data->INDICES[i]);
  953. }
  954. }
  955. template <class T>
  956. void AsmJsByteCodeDumper::DumpAsmSimdTypedArr(OpCodeAsmJs op, const unaligned T * data, FunctionBody * dumpFunction, ByteCodeReader& reader)
  957. {
  958. const char16* heapTag = nullptr;
  959. switch (data->ViewType)
  960. {
  961. case ArrayBufferView::TYPE_INT8:
  962. case ArrayBufferView::TYPE_INT8_TO_INT64:
  963. heapTag = _u("HEAP8"); break;
  964. case ArrayBufferView::TYPE_UINT8:
  965. case ArrayBufferView::TYPE_UINT8_TO_INT64:
  966. heapTag = _u("HEAPU8"); break;
  967. case ArrayBufferView::TYPE_INT16:
  968. case ArrayBufferView::TYPE_INT16_TO_INT64:
  969. heapTag = _u("HEAP16"); break;
  970. case ArrayBufferView::TYPE_UINT16:
  971. case ArrayBufferView::TYPE_UINT16_TO_INT64:
  972. heapTag = _u("HEAPU16"); break;
  973. case ArrayBufferView::TYPE_INT32:
  974. case ArrayBufferView::TYPE_INT32_TO_INT64:
  975. heapTag = _u("HEAP32"); break;
  976. case ArrayBufferView::TYPE_UINT32:
  977. case ArrayBufferView::TYPE_UINT32_TO_INT64:
  978. heapTag = _u("HEAPU32"); break;
  979. case ArrayBufferView::TYPE_FLOAT32:
  980. heapTag = _u("HEAPF32"); break;
  981. case ArrayBufferView::TYPE_FLOAT64:
  982. heapTag = _u("HEAPF64"); break;
  983. case ArrayBufferView::TYPE_INT64:
  984. heapTag = _u("HEAPI64"); break;
  985. default:
  986. Assert(false);
  987. __assume(false);
  988. break;
  989. }
  990. #define SIMD_DUMP_ARR_I4 DumpInt32x4Reg
  991. #define SIMD_DUMP_ARR_I8 DumpInt16x8Reg
  992. #define SIMD_DUMP_ARR_I16 DumpInt8x16Reg
  993. #define SIMD_DUMP_ARR_U4 DumpUint32x4Reg
  994. #define SIMD_DUMP_ARR_U8 DumpUint16x8Reg
  995. #define SIMD_DUMP_ARR_U16 DumpUint8x16Reg
  996. #define SIMD_DUMP_ARR_F4 DumpFloat32x4Reg
  997. #define SIMD_DUMP_ARR_D2 DumpFloat64x2Reg
  998. #define SIMD_DUMP_ARR_I2 DumpInt64x2Reg
  999. #define SIMD_DUMP_REG(type) SIMD_DUMP_ARR_##type(data->Value)
  1000. #define SIMD_DUMP_ARR_VALUE(type) \
  1001. case OpCodeAsmJs::Simd128_LdArr_##type:\
  1002. SIMD_DUMP_REG(type); Output::Print(_u("= %s[I%d] "), heapTag, data->SlotIndex); break;\
  1003. case OpCodeAsmJs::Simd128_LdArrConst_##type:\
  1004. SIMD_DUMP_REG(type); Output::Print(_u("= %s[%d] "), heapTag, data->SlotIndex); break;\
  1005. case OpCodeAsmJs::Simd128_StArr_##type:\
  1006. Output::Print(_u("%s[I%d] = "), heapTag, data->SlotIndex); SIMD_DUMP_REG(type); break;\
  1007. case OpCodeAsmJs::Simd128_StArrConst_##type:\
  1008. Output::Print(_u("%s[%d] = "), heapTag, data->SlotIndex); SIMD_DUMP_REG(type); break;\
  1009. switch (op)
  1010. {
  1011. SIMD_DUMP_ARR_VALUE(I4)
  1012. SIMD_DUMP_ARR_VALUE(I8)
  1013. SIMD_DUMP_ARR_VALUE(I16)
  1014. SIMD_DUMP_ARR_VALUE(U4)
  1015. SIMD_DUMP_ARR_VALUE(U8)
  1016. SIMD_DUMP_ARR_VALUE(U16)
  1017. SIMD_DUMP_ARR_VALUE(F4)
  1018. //SIMD_DUMP_ARR_VALUE(D2)
  1019. default:
  1020. Assert(false);
  1021. __assume(false);
  1022. break;
  1023. }
  1024. // data width
  1025. Output::Print(_u(" %d bytes "), data->DataWidth);
  1026. }
  1027. }
  1028. #endif
  1029. #endif