SimdUint8x16Lib.cpp 38 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934
  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 "RuntimeLibraryPch.h"
  6. namespace Js
  7. {
  8. Var SIMDUint8x16Lib::EntryUint8x16(RecyclableObject* function, CallInfo callInfo, ...)
  9. {
  10. PROBE_STACK(function->GetScriptContext(), Js::Constants::MinStackDefault);
  11. ARGUMENTS(args, callInfo);
  12. ScriptContext* scriptContext = function->GetScriptContext();
  13. AssertMsg(args.Info.Count > 0, "Should always have implicit 'this'");
  14. Assert(!(callInfo.Flags & CallFlags_New));
  15. Var undefinedVar = scriptContext->GetLibrary()->GetUndefined();
  16. const uint LANES = 16;
  17. uint8 values[LANES];
  18. for (uint i = 0; i < LANES; i++)
  19. {
  20. values[i] = JavascriptConversion::ToUInt8(args.Info.Count >= (i + 2) ? args[i + 1] : undefinedVar, scriptContext);
  21. }
  22. SIMDValue lanes = SIMDUint8x16Operation::OpUint8x16(values);
  23. return JavascriptSIMDUint8x16::New(&lanes, scriptContext);
  24. }
  25. Var SIMDUint8x16Lib::EntryCheck(RecyclableObject* function, CallInfo callInfo, ...)
  26. {
  27. PROBE_STACK(function->GetScriptContext(), Js::Constants::MinStackDefault);
  28. ARGUMENTS(args, callInfo);
  29. ScriptContext* scriptContext = function->GetScriptContext();
  30. AssertMsg(args.Info.Count > 0, "Should always have implicit 'this'");
  31. Assert(!(callInfo.Flags & CallFlags_New));
  32. if (args.Info.Count >= 2 && JavascriptSIMDUint8x16::Is(args[1]))
  33. {
  34. return args[1];
  35. }
  36. JavascriptError::ThrowTypeError(scriptContext, JSERR_SimdUint8x16TypeMismatch, _u("Uint8x16"));
  37. }
  38. Var SIMDUint8x16Lib::EntrySplat(RecyclableObject* function, CallInfo callInfo, ...)
  39. {
  40. PROBE_STACK(function->GetScriptContext(), Js::Constants::MinStackDefault);
  41. ARGUMENTS(args, callInfo);
  42. ScriptContext* scriptContext = function->GetScriptContext();
  43. AssertMsg(args.Info.Count > 0, "Should always have implicit 'this'");
  44. Assert(!(callInfo.Flags & CallFlags_New));
  45. Var undefinedVar = scriptContext->GetLibrary()->GetUndefined();
  46. uint8 value = JavascriptConversion::ToUInt8(args.Info.Count >= 2 ? args[1] : undefinedVar, scriptContext);
  47. SIMDValue lanes = SIMDInt8x16Operation::OpSplat(value);
  48. return JavascriptSIMDUint8x16::New(&lanes, scriptContext);
  49. }
  50. Var SIMDUint8x16Lib::EntryFromFloat32x4Bits(RecyclableObject* function, CallInfo callInfo, ...)
  51. {
  52. PROBE_STACK(function->GetScriptContext(), Js::Constants::MinStackDefault);
  53. ARGUMENTS(args, callInfo);
  54. ScriptContext* scriptContext = function->GetScriptContext();
  55. AssertMsg(args.Info.Count > 0, "Should always have implicit 'this'");
  56. Assert(!(callInfo.Flags & CallFlags_New));
  57. if (args.Info.Count >= 2 && JavascriptSIMDFloat32x4::Is(args[1]))
  58. {
  59. JavascriptSIMDFloat32x4 *instance = JavascriptSIMDFloat32x4::FromVar(args[1]);
  60. Assert(instance);
  61. return SIMDUtils::SIMDConvertTypeFromBits<JavascriptSIMDFloat32x4, JavascriptSIMDUint8x16>(*instance, *scriptContext);
  62. }
  63. JavascriptError::ThrowTypeError(scriptContext, JSERR_SimdUint8x16TypeMismatch, _u("fromFloat32x4Bits"));
  64. }
  65. Var SIMDUint8x16Lib::EntryFromInt32x4Bits(RecyclableObject* function, CallInfo callInfo, ...)
  66. {
  67. PROBE_STACK(function->GetScriptContext(), Js::Constants::MinStackDefault);
  68. ARGUMENTS(args, callInfo);
  69. ScriptContext* scriptContext = function->GetScriptContext();
  70. AssertMsg(args.Info.Count > 0, "Should always have implicit 'this'");
  71. Assert(!(callInfo.Flags & CallFlags_New));
  72. if (args.Info.Count >= 2 && JavascriptSIMDInt32x4::Is(args[1]))
  73. {
  74. JavascriptSIMDInt32x4 *instance = JavascriptSIMDInt32x4::FromVar(args[1]);
  75. Assert(instance);
  76. return SIMDUtils::SIMDConvertTypeFromBits<JavascriptSIMDInt32x4, JavascriptSIMDUint8x16>(*instance, *scriptContext);
  77. }
  78. JavascriptError::ThrowTypeError(scriptContext, JSERR_SimdUint8x16TypeMismatch, _u("fromInt32x4Bits"));
  79. }
  80. Var SIMDUint8x16Lib::EntryFromInt16x8Bits(RecyclableObject* function, CallInfo callInfo, ...)
  81. {
  82. PROBE_STACK(function->GetScriptContext(), Js::Constants::MinStackDefault);
  83. ARGUMENTS(args, callInfo);
  84. ScriptContext* scriptContext = function->GetScriptContext();
  85. AssertMsg(args.Info.Count > 0, "Should always have implicit 'this'");
  86. Assert(!(callInfo.Flags & CallFlags_New));
  87. if (args.Info.Count >= 2 && JavascriptSIMDInt16x8::Is(args[1]))
  88. {
  89. JavascriptSIMDInt16x8 *instance = JavascriptSIMDInt16x8::FromVar(args[1]);
  90. Assert(instance);
  91. return SIMDUtils::SIMDConvertTypeFromBits<JavascriptSIMDInt16x8, JavascriptSIMDUint8x16>(*instance, *scriptContext);
  92. }
  93. JavascriptError::ThrowTypeError(scriptContext, JSERR_SimdUint8x16TypeMismatch, _u("fromInt16x8Bits"));
  94. }
  95. Var SIMDUint8x16Lib::EntryFromInt8x16Bits(RecyclableObject* function, CallInfo callInfo, ...)
  96. {
  97. PROBE_STACK(function->GetScriptContext(), Js::Constants::MinStackDefault);
  98. ARGUMENTS(args, callInfo);
  99. ScriptContext* scriptContext = function->GetScriptContext();
  100. AssertMsg(args.Info.Count > 0, "Should always have implicit 'this'");
  101. Assert(!(callInfo.Flags & CallFlags_New));
  102. if (args.Info.Count >= 2 && JavascriptSIMDInt8x16::Is(args[1]))
  103. {
  104. JavascriptSIMDInt8x16 *instance = JavascriptSIMDInt8x16::FromVar(args[1]);
  105. Assert(instance);
  106. return SIMDUtils::SIMDConvertTypeFromBits<JavascriptSIMDInt8x16, JavascriptSIMDUint8x16>(*instance, *scriptContext);
  107. }
  108. JavascriptError::ThrowTypeError(scriptContext, JSERR_SimdUint8x16TypeMismatch, _u("fromInt8x16Bits"));
  109. }
  110. Var SIMDUint8x16Lib::EntryFromUint32x4Bits(RecyclableObject* function, CallInfo callInfo, ...)
  111. {
  112. PROBE_STACK(function->GetScriptContext(), Js::Constants::MinStackDefault);
  113. ARGUMENTS(args, callInfo);
  114. ScriptContext* scriptContext = function->GetScriptContext();
  115. AssertMsg(args.Info.Count > 0, "Should always have implicit 'this'");
  116. Assert(!(callInfo.Flags & CallFlags_New));
  117. if (args.Info.Count >= 2 && JavascriptSIMDUint32x4::Is(args[1]))
  118. {
  119. JavascriptSIMDUint32x4 *instance = JavascriptSIMDUint32x4::FromVar(args[1]);
  120. Assert(instance);
  121. return SIMDUtils::SIMDConvertTypeFromBits<JavascriptSIMDUint32x4, JavascriptSIMDUint8x16>(*instance, *scriptContext);
  122. }
  123. JavascriptError::ThrowTypeError(scriptContext, JSERR_SimdUint8x16TypeMismatch, _u("fromUint32x4Bits"));
  124. }
  125. Var SIMDUint8x16Lib::EntryFromUint16x8Bits(RecyclableObject* function, CallInfo callInfo, ...)
  126. {
  127. PROBE_STACK(function->GetScriptContext(), Js::Constants::MinStackDefault);
  128. ARGUMENTS(args, callInfo);
  129. ScriptContext* scriptContext = function->GetScriptContext();
  130. AssertMsg(args.Info.Count > 0, "Should always have implicit 'this'");
  131. Assert(!(callInfo.Flags & CallFlags_New));
  132. if (args.Info.Count >= 2 && JavascriptSIMDUint16x8::Is(args[1]))
  133. {
  134. JavascriptSIMDUint16x8 *instance = JavascriptSIMDUint16x8::FromVar(args[1]);
  135. Assert(instance);
  136. return SIMDUtils::SIMDConvertTypeFromBits<JavascriptSIMDUint16x8, JavascriptSIMDUint8x16>(*instance, *scriptContext);
  137. }
  138. JavascriptError::ThrowTypeError(scriptContext, JSERR_SimdUint8x16TypeMismatch, _u("fromUint16x8Bits"));
  139. }
  140. Var SIMDUint8x16Lib::EntryMin(RecyclableObject* function, CallInfo callInfo, ...)
  141. {
  142. PROBE_STACK(function->GetScriptContext(), Js::Constants::MinStackDefault);
  143. ARGUMENTS(args, callInfo);
  144. ScriptContext* scriptContext = function->GetScriptContext();
  145. AssertMsg(args.Info.Count > 0, "Should always have implicit 'this'");
  146. Assert(!(callInfo.Flags & CallFlags_New));
  147. // If any of the args are missing, then it is Undefined type which causes TypeError exception.
  148. if (args.Info.Count >= 3 && JavascriptSIMDUint8x16::Is(args[1]) && JavascriptSIMDUint8x16::Is(args[2]))
  149. {
  150. JavascriptSIMDUint8x16 *a = JavascriptSIMDUint8x16::FromVar(args[1]);
  151. JavascriptSIMDUint8x16 *b = JavascriptSIMDUint8x16::FromVar(args[2]);
  152. Assert(a && b);
  153. SIMDValue result, aValue, bValue;
  154. aValue = a->GetValue();
  155. bValue = b->GetValue();
  156. result = SIMDUint8x16Operation::OpMin(aValue, bValue);
  157. return JavascriptSIMDUint8x16::New(&result, scriptContext);
  158. }
  159. JavascriptError::ThrowTypeError(scriptContext, JSERR_SimdUint8x16TypeMismatch, _u("min"));
  160. }
  161. Var SIMDUint8x16Lib::EntryMax(RecyclableObject* function, CallInfo callInfo, ...)
  162. {
  163. PROBE_STACK(function->GetScriptContext(), Js::Constants::MinStackDefault);
  164. ARGUMENTS(args, callInfo);
  165. ScriptContext* scriptContext = function->GetScriptContext();
  166. AssertMsg(args.Info.Count > 0, "Should always have implicit 'this'");
  167. Assert(!(callInfo.Flags & CallFlags_New));
  168. // If any of the args are missing, then it is Undefined type which causes TypeError exception.
  169. if (args.Info.Count >= 3 && JavascriptSIMDUint8x16::Is(args[1]) && JavascriptSIMDUint8x16::Is(args[2]))
  170. {
  171. JavascriptSIMDUint8x16 *a = JavascriptSIMDUint8x16::FromVar(args[1]);
  172. JavascriptSIMDUint8x16 *b = JavascriptSIMDUint8x16::FromVar(args[2]);
  173. Assert(a && b);
  174. SIMDValue result, aValue, bValue;
  175. aValue = a->GetValue();
  176. bValue = b->GetValue();
  177. result = SIMDUint8x16Operation::OpMax(aValue, bValue);
  178. return JavascriptSIMDUint8x16::New(&result, scriptContext);
  179. }
  180. JavascriptError::ThrowTypeError(scriptContext, JSERR_SimdUint8x16TypeMismatch, _u("max"));
  181. }
  182. Var SIMDUint8x16Lib::EntryLoad(RecyclableObject* function, CallInfo callInfo, ...)
  183. {
  184. PROBE_STACK(function->GetScriptContext(), Js::Constants::MinStackDefault);
  185. ARGUMENTS(args, callInfo);
  186. ScriptContext* scriptContext = function->GetScriptContext();
  187. AssertMsg(args.Info.Count > 0, "Should always have implicit 'this'");
  188. Assert(!(callInfo.Flags & CallFlags_New));
  189. return SIMDUtils::SIMD128TypedArrayLoad<JavascriptSIMDUint8x16>(args[1], args[2], 16 * INT8_SIZE, scriptContext);
  190. }
  191. Var SIMDUint8x16Lib::EntryStore(RecyclableObject* function, CallInfo callInfo, ...)
  192. {
  193. PROBE_STACK(function->GetScriptContext(), Js::Constants::MinStackDefault);
  194. ARGUMENTS(args, callInfo);
  195. ScriptContext* scriptContext = function->GetScriptContext();
  196. AssertMsg(args.Info.Count > 0, "Should always have implicit 'this'");
  197. Assert(!(callInfo.Flags & CallFlags_New));
  198. if (args.Info.Count >= 4 && JavascriptSIMDUint8x16::Is(args[3]))
  199. {
  200. SIMDUtils::SIMD128TypedArrayStore<JavascriptSIMDUint8x16>(args[1], args[2], args[3], 16 * INT8_SIZE, scriptContext);
  201. return JavascriptSIMDUint8x16::FromVar(args[3]);
  202. }
  203. JavascriptError::ThrowTypeError(scriptContext, JSERR_SimdInvalidArgType, _u("SIMD.Uint8x16.store"));
  204. }
  205. Var SIMDUint8x16Lib::EntryNeg(RecyclableObject* function, CallInfo callInfo, ...)
  206. {
  207. PROBE_STACK(function->GetScriptContext(), Js::Constants::MinStackDefault);
  208. ARGUMENTS(args, callInfo);
  209. ScriptContext* scriptContext = function->GetScriptContext();
  210. AssertMsg(args.Info.Count > 0, "Should always have implicit 'this'");
  211. Assert(!(callInfo.Flags & CallFlags_New));
  212. if (args.Info.Count >= 2 && JavascriptSIMDUint8x16::Is(args[1]))
  213. {
  214. JavascriptSIMDUint8x16 *a = JavascriptSIMDUint8x16::FromVar(args[1]);
  215. Assert(a);
  216. SIMDValue value, result;
  217. value = a->GetValue();
  218. result = SIMDInt8x16Operation::OpNeg(value);
  219. return JavascriptSIMDUint8x16::New(&result, scriptContext);
  220. }
  221. JavascriptError::ThrowTypeError(scriptContext, JSERR_SimdInt8x16TypeMismatch, L"neg");
  222. }
  223. Var SIMDUint8x16Lib::EntryNot(RecyclableObject* function, CallInfo callInfo, ...)
  224. {
  225. PROBE_STACK(function->GetScriptContext(), Js::Constants::MinStackDefault);
  226. ARGUMENTS(args, callInfo);
  227. ScriptContext* scriptContext = function->GetScriptContext();
  228. AssertMsg(args.Info.Count > 0, "Should always have implicit 'this'");
  229. Assert(!(callInfo.Flags & CallFlags_New));
  230. if (args.Info.Count >= 2 && JavascriptSIMDUint8x16::Is(args[1]))
  231. {
  232. JavascriptSIMDUint8x16 *a = JavascriptSIMDUint8x16::FromVar(args[1]);
  233. Assert(a);
  234. SIMDValue value, result;
  235. value = a->GetValue();
  236. result = SIMDInt32x4Operation::OpNot(value);
  237. return JavascriptSIMDUint8x16::New(&result, scriptContext);
  238. }
  239. JavascriptError::ThrowTypeError(scriptContext, JSERR_SimdUint8x16TypeMismatch, _u("not"));
  240. }
  241. Var SIMDUint8x16Lib::EntryAdd(RecyclableObject* function, CallInfo callInfo, ...)
  242. {
  243. PROBE_STACK(function->GetScriptContext(), Js::Constants::MinStackDefault);
  244. ARGUMENTS(args, callInfo);
  245. ScriptContext* scriptContext = function->GetScriptContext();
  246. AssertMsg(args.Info.Count > 0, "Should always have implicit 'this'");
  247. Assert(!(callInfo.Flags & CallFlags_New));
  248. // If any of the args are missing, then it is Undefined type which causes TypeError exception.
  249. if (args.Info.Count >= 3 && JavascriptSIMDUint8x16::Is(args[1]) && JavascriptSIMDUint8x16::Is(args[2]))
  250. {
  251. JavascriptSIMDUint8x16 *a = JavascriptSIMDUint8x16::FromVar(args[1]);
  252. JavascriptSIMDUint8x16 *b = JavascriptSIMDUint8x16::FromVar(args[2]);
  253. Assert(a && b);
  254. SIMDValue result, aValue, bValue;
  255. aValue = a->GetValue();
  256. bValue = b->GetValue();
  257. result = SIMDInt8x16Operation::OpAdd(aValue, bValue);
  258. return JavascriptSIMDUint8x16::New(&result, scriptContext);
  259. }
  260. JavascriptError::ThrowTypeError(scriptContext, JSERR_SimdUint8x16TypeMismatch, _u("add"));
  261. }
  262. Var SIMDUint8x16Lib::EntrySub(RecyclableObject* function, CallInfo callInfo, ...)
  263. {
  264. PROBE_STACK(function->GetScriptContext(), Js::Constants::MinStackDefault);
  265. ARGUMENTS(args, callInfo);
  266. ScriptContext* scriptContext = function->GetScriptContext();
  267. AssertMsg(args.Info.Count > 0, "Should always have implicit 'this'");
  268. Assert(!(callInfo.Flags & CallFlags_New));
  269. // If any of the args are missing, then it is Undefined type which causes TypeError exception.
  270. if (args.Info.Count >= 3 && JavascriptSIMDUint8x16::Is(args[1]) && JavascriptSIMDUint8x16::Is(args[2]))
  271. {
  272. JavascriptSIMDUint8x16 *a = JavascriptSIMDUint8x16::FromVar(args[1]);
  273. JavascriptSIMDUint8x16 *b = JavascriptSIMDUint8x16::FromVar(args[2]);
  274. Assert(a && b);
  275. SIMDValue result, aValue, bValue;
  276. aValue = a->GetValue();
  277. bValue = b->GetValue();
  278. result = SIMDInt8x16Operation::OpSub(aValue, bValue);
  279. return JavascriptSIMDUint8x16::New(&result, scriptContext);
  280. }
  281. JavascriptError::ThrowTypeError(scriptContext, JSERR_SimdUint8x16TypeMismatch, _u("sub"));
  282. }
  283. Var SIMDUint8x16Lib::EntryMul(RecyclableObject* function, CallInfo callInfo, ...)
  284. {
  285. PROBE_STACK(function->GetScriptContext(), Js::Constants::MinStackDefault);
  286. ARGUMENTS(args, callInfo);
  287. ScriptContext* scriptContext = function->GetScriptContext();
  288. AssertMsg(args.Info.Count > 0, "Should always have implicit 'this'");
  289. Assert(!(callInfo.Flags & CallFlags_New));
  290. // If any of the args are missing, then it is Undefined type which causes TypeError exception.
  291. if (args.Info.Count >= 3 && JavascriptSIMDUint8x16::Is(args[1]) && JavascriptSIMDUint8x16::Is(args[2]))
  292. {
  293. JavascriptSIMDUint8x16 *a = JavascriptSIMDUint8x16::FromVar(args[1]);
  294. JavascriptSIMDUint8x16 *b = JavascriptSIMDUint8x16::FromVar(args[2]);
  295. Assert(a && b);
  296. SIMDValue result, aValue, bValue;
  297. aValue = a->GetValue();
  298. bValue = b->GetValue();
  299. result = SIMDInt8x16Operation::OpMul(aValue, bValue);
  300. return JavascriptSIMDUint8x16::New(&result, scriptContext);
  301. }
  302. JavascriptError::ThrowTypeError(scriptContext, JSERR_SimdUint8x16TypeMismatch, _u("mul"));
  303. }
  304. Var SIMDUint8x16Lib::EntryAnd(RecyclableObject* function, CallInfo callInfo, ...)
  305. {
  306. PROBE_STACK(function->GetScriptContext(), Js::Constants::MinStackDefault);
  307. ARGUMENTS(args, callInfo);
  308. ScriptContext* scriptContext = function->GetScriptContext();
  309. AssertMsg(args.Info.Count > 0, "Should always have implicit 'this'");
  310. Assert(!(callInfo.Flags & CallFlags_New));
  311. // If any of the args are missing, then it is Undefined type which causes TypeError exception.
  312. if (args.Info.Count >= 3 && JavascriptSIMDUint8x16::Is(args[1]) && JavascriptSIMDUint8x16::Is(args[2]))
  313. {
  314. JavascriptSIMDUint8x16 *a = JavascriptSIMDUint8x16::FromVar(args[1]);
  315. JavascriptSIMDUint8x16 *b = JavascriptSIMDUint8x16::FromVar(args[2]);
  316. Assert(a && b);
  317. SIMDValue result, aValue, bValue;
  318. aValue = a->GetValue();
  319. bValue = b->GetValue();
  320. result = SIMDInt32x4Operation::OpAnd(aValue, bValue);
  321. return JavascriptSIMDUint8x16::New(&result, scriptContext);
  322. }
  323. JavascriptError::ThrowTypeError(scriptContext, JSERR_SimdUint8x16TypeMismatch, _u("and"));
  324. }
  325. Var SIMDUint8x16Lib::EntryOr(RecyclableObject* function, CallInfo callInfo, ...)
  326. {
  327. PROBE_STACK(function->GetScriptContext(), Js::Constants::MinStackDefault);
  328. ARGUMENTS(args, callInfo);
  329. ScriptContext* scriptContext = function->GetScriptContext();
  330. AssertMsg(args.Info.Count > 0, "Should always have implicit 'this'");
  331. Assert(!(callInfo.Flags & CallFlags_New));
  332. // If any of the args are missing, then it is Undefined type which causes TypeError exception.
  333. if (args.Info.Count >= 3 && JavascriptSIMDUint8x16::Is(args[1]) && JavascriptSIMDUint8x16::Is(args[2]))
  334. {
  335. JavascriptSIMDUint8x16 *a = JavascriptSIMDUint8x16::FromVar(args[1]);
  336. JavascriptSIMDUint8x16 *b = JavascriptSIMDUint8x16::FromVar(args[2]);
  337. Assert(a && b);
  338. SIMDValue result, aValue, bValue;
  339. aValue = a->GetValue();
  340. bValue = b->GetValue();
  341. result = SIMDInt32x4Operation::OpOr(aValue, bValue);
  342. return JavascriptSIMDUint8x16::New(&result, scriptContext);
  343. }
  344. JavascriptError::ThrowTypeError(scriptContext, JSERR_SimdUint8x16TypeMismatch, _u("or"));
  345. }
  346. Var SIMDUint8x16Lib::EntryXor(RecyclableObject* function, CallInfo callInfo, ...)
  347. {
  348. PROBE_STACK(function->GetScriptContext(), Js::Constants::MinStackDefault);
  349. ARGUMENTS(args, callInfo);
  350. ScriptContext* scriptContext = function->GetScriptContext();
  351. AssertMsg(args.Info.Count > 0, "Should always have implicit 'this'");
  352. Assert(!(callInfo.Flags & CallFlags_New));
  353. // If any of the args are missing, then it is Undefined type which causes TypeError exception.
  354. if (args.Info.Count >= 3 && JavascriptSIMDUint8x16::Is(args[1]) && JavascriptSIMDUint8x16::Is(args[2]))
  355. {
  356. JavascriptSIMDUint8x16 *a = JavascriptSIMDUint8x16::FromVar(args[1]);
  357. JavascriptSIMDUint8x16 *b = JavascriptSIMDUint8x16::FromVar(args[2]);
  358. Assert(a && b);
  359. SIMDValue result, aValue, bValue;
  360. aValue = a->GetValue();
  361. bValue = b->GetValue();
  362. result = SIMDInt32x4Operation::OpXor(aValue, bValue);
  363. return JavascriptSIMDUint8x16::New(&result, scriptContext);
  364. }
  365. JavascriptError::ThrowTypeError(scriptContext, JSERR_SimdUint8x16TypeMismatch, _u("xor"));
  366. }
  367. Var SIMDUint8x16Lib::EntryLessThan(RecyclableObject* function, CallInfo callInfo, ...)
  368. {
  369. PROBE_STACK(function->GetScriptContext(), Js::Constants::MinStackDefault);
  370. ARGUMENTS(args, callInfo);
  371. ScriptContext* scriptContext = function->GetScriptContext();
  372. AssertMsg(args.Info.Count > 0, "Should always have implicit 'this'");
  373. Assert(!(callInfo.Flags & CallFlags_New));
  374. // If any of the args are missing, then it is Undefined type which causes TypeError exception.
  375. // strict type on both operands
  376. if (args.Info.Count >= 3 && JavascriptSIMDUint8x16::Is(args[1]) && JavascriptSIMDUint8x16::Is(args[2]))
  377. {
  378. JavascriptSIMDUint8x16 *a = JavascriptSIMDUint8x16::FromVar(args[1]);
  379. JavascriptSIMDUint8x16 *b = JavascriptSIMDUint8x16::FromVar(args[2]);
  380. Assert(a && b);
  381. SIMDValue result, aValue, bValue;
  382. aValue = a->GetValue();
  383. bValue = b->GetValue();
  384. result = SIMDUint8x16Operation::OpLessThan(aValue, bValue);
  385. return JavascriptSIMDBool8x16::New(&result, scriptContext);
  386. }
  387. JavascriptError::ThrowTypeError(scriptContext, JSERR_SimdUint8x16TypeMismatch, _u("lessThan"));
  388. }
  389. Var SIMDUint8x16Lib::EntryLessThanOrEqual(RecyclableObject* function, CallInfo callInfo, ...)
  390. {
  391. PROBE_STACK(function->GetScriptContext(), Js::Constants::MinStackDefault);
  392. ARGUMENTS(args, callInfo);
  393. ScriptContext* scriptContext = function->GetScriptContext();
  394. AssertMsg(args.Info.Count > 0, "Should always have implicit 'this'");
  395. Assert(!(callInfo.Flags & CallFlags_New));
  396. // If any of the args are missing, then it is Undefined type which causes TypeError exception.
  397. // strict type on both operands
  398. if (args.Info.Count >= 3 && JavascriptSIMDUint8x16::Is(args[1]) && JavascriptSIMDUint8x16::Is(args[2]))
  399. {
  400. JavascriptSIMDUint8x16 *a = JavascriptSIMDUint8x16::FromVar(args[1]);
  401. JavascriptSIMDUint8x16 *b = JavascriptSIMDUint8x16::FromVar(args[2]);
  402. Assert(a && b);
  403. SIMDValue result, aValue, bValue;
  404. aValue = a->GetValue();
  405. bValue = b->GetValue();
  406. result = SIMDUint8x16Operation::OpLessThanOrEqual(aValue, bValue);
  407. return JavascriptSIMDBool8x16::New(&result, scriptContext);
  408. }
  409. JavascriptError::ThrowTypeError(scriptContext, JSERR_SimdUint8x16TypeMismatch, _u("lessThanOrEqual"));
  410. }
  411. Var SIMDUint8x16Lib::EntryEqual(RecyclableObject* function, CallInfo callInfo, ...)
  412. {
  413. PROBE_STACK(function->GetScriptContext(), Js::Constants::MinStackDefault);
  414. ARGUMENTS(args, callInfo);
  415. ScriptContext* scriptContext = function->GetScriptContext();
  416. AssertMsg(args.Info.Count > 0, "Should always have implicit 'this'");
  417. Assert(!(callInfo.Flags & CallFlags_New));
  418. // If any of the args are missing, then it is Undefined type which causes TypeError exception.
  419. // strict type on both operands
  420. if (args.Info.Count >= 3 && JavascriptSIMDUint8x16::Is(args[1]) && JavascriptSIMDUint8x16::Is(args[2]))
  421. {
  422. JavascriptSIMDUint8x16 *a = JavascriptSIMDUint8x16::FromVar(args[1]);
  423. JavascriptSIMDUint8x16 *b = JavascriptSIMDUint8x16::FromVar(args[2]);
  424. Assert(a && b);
  425. SIMDValue result, aValue, bValue;
  426. aValue = a->GetValue();
  427. bValue = b->GetValue();
  428. result = SIMDInt8x16Operation::OpEqual(aValue, bValue);
  429. return JavascriptSIMDBool8x16::New(&result, scriptContext);
  430. }
  431. JavascriptError::ThrowTypeError(scriptContext, JSERR_SimdUint8x16TypeMismatch, _u("equal"));
  432. }
  433. Var SIMDUint8x16Lib::EntryNotEqual(RecyclableObject* function, CallInfo callInfo, ...)
  434. {
  435. PROBE_STACK(function->GetScriptContext(), Js::Constants::MinStackDefault);
  436. ARGUMENTS(args, callInfo);
  437. ScriptContext* scriptContext = function->GetScriptContext();
  438. AssertMsg(args.Info.Count > 0, "Should always have implicit 'this'");
  439. Assert(!(callInfo.Flags & CallFlags_New));
  440. // If any of the args are missing, then it is Undefined type which causes TypeError exception.
  441. // strict type on both operands
  442. if (args.Info.Count >= 3 && JavascriptSIMDUint8x16::Is(args[1]) && JavascriptSIMDUint8x16::Is(args[2]))
  443. {
  444. JavascriptSIMDUint8x16 *a = JavascriptSIMDUint8x16::FromVar(args[1]);
  445. JavascriptSIMDUint8x16 *b = JavascriptSIMDUint8x16::FromVar(args[2]);
  446. Assert(a && b);
  447. SIMDValue result, aValue, bValue;
  448. aValue = a->GetValue();
  449. bValue = b->GetValue();
  450. result = SIMDInt8x16Operation::OpNotEqual(aValue, bValue);
  451. return JavascriptSIMDBool8x16::New(&result, scriptContext);
  452. }
  453. JavascriptError::ThrowTypeError(scriptContext, JSERR_SimdUint8x16TypeMismatch, _u("notEqual"));
  454. }
  455. Var SIMDUint8x16Lib::EntryGreaterThan(RecyclableObject* function, CallInfo callInfo, ...)
  456. {
  457. PROBE_STACK(function->GetScriptContext(), Js::Constants::MinStackDefault);
  458. ARGUMENTS(args, callInfo);
  459. ScriptContext* scriptContext = function->GetScriptContext();
  460. AssertMsg(args.Info.Count > 0, "Should always have implicit 'this'");
  461. Assert(!(callInfo.Flags & CallFlags_New));
  462. // If any of the args are missing, then it is Undefined type which causes TypeError exception.
  463. // strict type on both operands
  464. if (args.Info.Count >= 3 && JavascriptSIMDUint8x16::Is(args[1]) && JavascriptSIMDUint8x16::Is(args[2]))
  465. {
  466. JavascriptSIMDUint8x16 *a = JavascriptSIMDUint8x16::FromVar(args[1]);
  467. JavascriptSIMDUint8x16 *b = JavascriptSIMDUint8x16::FromVar(args[2]);
  468. Assert(a && b);
  469. SIMDValue result, aValue, bValue;
  470. aValue = a->GetValue();
  471. bValue = b->GetValue();
  472. result = SIMDUint8x16Operation::OpGreaterThan(aValue, bValue);
  473. return JavascriptSIMDBool8x16::New(&result, scriptContext);
  474. }
  475. JavascriptError::ThrowTypeError(scriptContext, JSERR_SimdUint8x16TypeMismatch, _u("greaterThan"));
  476. }
  477. Var SIMDUint8x16Lib::EntryGreaterThanOrEqual(RecyclableObject* function, CallInfo callInfo, ...)
  478. {
  479. PROBE_STACK(function->GetScriptContext(), Js::Constants::MinStackDefault);
  480. ARGUMENTS(args, callInfo);
  481. ScriptContext* scriptContext = function->GetScriptContext();
  482. AssertMsg(args.Info.Count > 0, "Should always have implicit 'this'");
  483. Assert(!(callInfo.Flags & CallFlags_New));
  484. // If any of the args are missing, then it is Undefined type which causes TypeError exception.
  485. // strict type on both operands
  486. if (args.Info.Count >= 3 && JavascriptSIMDUint8x16::Is(args[1]) && JavascriptSIMDUint8x16::Is(args[2]))
  487. {
  488. JavascriptSIMDUint8x16 *a = JavascriptSIMDUint8x16::FromVar(args[1]);
  489. JavascriptSIMDUint8x16 *b = JavascriptSIMDUint8x16::FromVar(args[2]);
  490. Assert(a && b);
  491. SIMDValue result, aValue, bValue;
  492. aValue = a->GetValue();
  493. bValue = b->GetValue();
  494. result = SIMDUint8x16Operation::OpGreaterThanOrEqual(aValue, bValue);
  495. return JavascriptSIMDBool8x16::New(&result, scriptContext);
  496. }
  497. JavascriptError::ThrowTypeError(scriptContext, JSERR_SimdUint8x16TypeMismatch, _u("greaterThanOrEqual"));
  498. }
  499. Var SIMDUint8x16Lib::EntryShiftLeftByScalar(RecyclableObject* function, CallInfo callInfo, ...)
  500. {
  501. PROBE_STACK(function->GetScriptContext(), Js::Constants::MinStackDefault);
  502. ARGUMENTS(args, callInfo);
  503. ScriptContext* scriptContext = function->GetScriptContext();
  504. AssertMsg(args.Info.Count > 0, "Should always have implicit 'this'");
  505. Assert(!(callInfo.Flags & CallFlags_New));
  506. if (args.Info.Count >= 3 && JavascriptSIMDUint8x16::Is(args[1]))
  507. {
  508. JavascriptSIMDUint8x16 *a = JavascriptSIMDUint8x16::FromVar(args[1]);
  509. Assert(a);
  510. SIMDValue result, aValue;
  511. aValue = a->GetValue();
  512. Var countVar = args[2]; // {int} bits Bit count
  513. int8 count = JavascriptConversion::ToInt8(countVar, scriptContext);
  514. result = SIMDInt8x16Operation::OpShiftLeftByScalar(aValue, count);
  515. return JavascriptSIMDUint8x16::New(&result, scriptContext);
  516. }
  517. JavascriptError::ThrowTypeError(scriptContext, JSERR_SimdUint8x16TypeMismatch, _u("shiftLeft"));
  518. }
  519. Var SIMDUint8x16Lib::EntryShiftRightByScalar(RecyclableObject* function, CallInfo callInfo, ...)
  520. {
  521. PROBE_STACK(function->GetScriptContext(), Js::Constants::MinStackDefault);
  522. ARGUMENTS(args, callInfo);
  523. ScriptContext* scriptContext = function->GetScriptContext();
  524. AssertMsg(args.Info.Count > 0, "Should always have implicit 'this'");
  525. Assert(!(callInfo.Flags & CallFlags_New));
  526. if (args.Info.Count >= 3 && JavascriptSIMDUint8x16::Is(args[1]))
  527. {
  528. JavascriptSIMDUint8x16 *a = JavascriptSIMDUint8x16::FromVar(args[1]);
  529. Assert(a);
  530. SIMDValue result, aValue;
  531. aValue = a->GetValue();
  532. Var countVar = args[2]; // {int} bits Bit count
  533. int8 count = JavascriptConversion::ToInt8(countVar, scriptContext);
  534. result = SIMDUint8x16Operation::OpShiftRightByScalar(aValue, count);
  535. return JavascriptSIMDUint8x16::New(&result, scriptContext);
  536. }
  537. JavascriptError::ThrowTypeError(scriptContext, JSERR_SimdUint8x16TypeMismatch, _u("shiftRightByScalar"));
  538. }
  539. Var SIMDUint8x16Lib::EntrySwizzle(RecyclableObject* function, CallInfo callInfo, ...)
  540. {
  541. PROBE_STACK(function->GetScriptContext(), Js::Constants::MinStackDefault);
  542. ARGUMENTS(args, callInfo);
  543. ScriptContext* scriptContext = function->GetScriptContext();
  544. AssertMsg(args.Info.Count > 0, "Should always have implicit 'this'");
  545. Assert(!(callInfo.Flags & CallFlags_New));
  546. if (args.Info.Count >= 2 && JavascriptSIMDUint8x16::Is(args[1]))
  547. {
  548. // type check on lane indices
  549. if (args.Info.Count < 18)
  550. {
  551. // missing lane args
  552. JavascriptError::ThrowTypeError(scriptContext, JSERR_NeedNumber, _u("Lane index"));
  553. }
  554. Var lanes[16];
  555. for (uint i = 0; i < 16; ++i)
  556. {
  557. lanes[i] = args[i + 2];
  558. }
  559. return SIMDUtils::SIMD128SlowShuffle<JavascriptSIMDUint8x16, 16>(args[1], args[1], lanes, 16, scriptContext);
  560. }
  561. JavascriptError::ThrowTypeError(scriptContext, JSERR_SimdUint8x16TypeMismatch, _u("swizzle"));
  562. }
  563. Var SIMDUint8x16Lib::EntryShuffle(RecyclableObject* function, CallInfo callInfo, ...)
  564. {
  565. PROBE_STACK(function->GetScriptContext(), Js::Constants::MinStackDefault);
  566. ARGUMENTS(args, callInfo);
  567. ScriptContext* scriptContext = function->GetScriptContext();
  568. AssertMsg(args.Info.Count > 0, "Should always have implicit 'this'");
  569. Assert(!(callInfo.Flags & CallFlags_New));
  570. // If any of the args are missing, then it is Undefined type which causes TypeError exception.
  571. // strict type on both operands
  572. if (args.Info.Count >= 3 && JavascriptSIMDUint8x16::Is(args[1]) && JavascriptSIMDUint8x16::Is(args[2]))
  573. {
  574. // type check on lane indices
  575. if (args.Info.Count < 19)
  576. {
  577. // missing lane args
  578. JavascriptError::ThrowTypeError(scriptContext, JSERR_NeedNumber, _u("Lane index"));
  579. }
  580. Var lanes[16];
  581. for (uint i = 0; i < 16; ++i)
  582. {
  583. lanes[i] = args[i + 3];
  584. }
  585. return SIMDUtils::SIMD128SlowShuffle<JavascriptSIMDUint8x16, 16>(args[1], args[2], lanes, 32, scriptContext);
  586. }
  587. JavascriptError::ThrowTypeError(scriptContext, JSERR_SimdUint8x16TypeMismatch, _u("shuffle"));
  588. }
  589. //Lane Access
  590. Var SIMDUint8x16Lib::EntryExtractLane(RecyclableObject* function, CallInfo callInfo, ...)
  591. {
  592. PROBE_STACK(function->GetScriptContext(), Js::Constants::MinStackDefault);
  593. ARGUMENTS(args, callInfo);
  594. ScriptContext* scriptContext = function->GetScriptContext();
  595. AssertMsg(args.Info.Count > 0, "Should always have implicit 'this'");
  596. Assert(!(callInfo.Flags & CallFlags_New));
  597. // first arg has to be of type Uint8x16, so cannot be missing.
  598. if (args.Info.Count >= 3 && JavascriptSIMDUint8x16::Is(args[1]))
  599. {
  600. // if value arg is missing, then it is undefined.
  601. Var laneVar = args.Info.Count >= 3 ? args[2] : scriptContext->GetLibrary()->GetUndefined();
  602. uint8 result = SIMDUtils::SIMD128ExtractLane<JavascriptSIMDUint8x16, 16, uint8>(args[1], laneVar, scriptContext);
  603. return JavascriptNumber::ToVarNoCheck(result, scriptContext);
  604. }
  605. JavascriptError::ThrowTypeError(scriptContext, JSERR_SimdUint8x16TypeMismatch, _u("ExtractLane"));
  606. }
  607. Var SIMDUint8x16Lib::EntryReplaceLane(RecyclableObject* function, CallInfo callInfo, ...)
  608. {
  609. PROBE_STACK(function->GetScriptContext(), Js::Constants::MinStackDefault);
  610. ARGUMENTS(args, callInfo);
  611. ScriptContext* scriptContext = function->GetScriptContext();
  612. AssertMsg(args.Info.Count > 0, "Should always have implicit 'this'");
  613. Assert(!(callInfo.Flags & CallFlags_New));
  614. // first arg has to be of type Uint8x16, so cannot be missing.
  615. if (args.Info.Count >= 4 && JavascriptSIMDUint8x16::Is(args[1]))
  616. {
  617. // if value arg is missing, then it is undefined.
  618. Var laneVar = args.Info.Count >= 4 ? args[2] : scriptContext->GetLibrary()->GetUndefined();
  619. Var argVal = args.Info.Count >= 4 ? args[3] : scriptContext->GetLibrary()->GetUndefined();
  620. uint8 value = JavascriptConversion::ToInt8(argVal, scriptContext);
  621. SIMDValue result = SIMDUtils::SIMD128ReplaceLane<JavascriptSIMDUint8x16, 16, uint8>(args[1], laneVar, value, scriptContext);
  622. return JavascriptSIMDUint8x16::New(&result, scriptContext);
  623. }
  624. JavascriptError::ThrowTypeError(scriptContext, JSERR_SimdUint8x16TypeMismatch, _u("ReplaceLane"));
  625. }
  626. Var SIMDUint8x16Lib::EntryAddSaturate(RecyclableObject* function, CallInfo callInfo, ...)
  627. {
  628. PROBE_STACK(function->GetScriptContext(), Js::Constants::MinStackDefault);
  629. ARGUMENTS(args, callInfo);
  630. ScriptContext* scriptContext = function->GetScriptContext();
  631. AssertMsg(args.Info.Count > 0, "Should always have implicit 'this'");
  632. Assert(!(callInfo.Flags & CallFlags_New));
  633. // If any of the args are missing, then it is Undefined type which causes TypeError exception.
  634. // strict type on both operands
  635. if (args.Info.Count >= 3 && JavascriptSIMDUint8x16::Is(args[1]) && JavascriptSIMDUint8x16::Is(args[2]))
  636. {
  637. JavascriptSIMDUint8x16 *a = JavascriptSIMDUint8x16::FromVar(args[1]);
  638. JavascriptSIMDUint8x16 *b = JavascriptSIMDUint8x16::FromVar(args[2]);
  639. Assert(a && b);
  640. SIMDValue result, aValue, bValue;
  641. aValue = a->GetValue();
  642. bValue = b->GetValue();
  643. result = SIMDUint8x16Operation::OpAddSaturate(aValue, bValue);
  644. return JavascriptSIMDUint8x16::New(&result, scriptContext);
  645. }
  646. JavascriptError::ThrowTypeError(scriptContext, JSERR_SimdUint8x16TypeMismatch, _u("addSaturate"));
  647. }
  648. Var SIMDUint8x16Lib::EntrySubSaturate(RecyclableObject* function, CallInfo callInfo, ...)
  649. {
  650. PROBE_STACK(function->GetScriptContext(), Js::Constants::MinStackDefault);
  651. ARGUMENTS(args, callInfo);
  652. ScriptContext* scriptContext = function->GetScriptContext();
  653. AssertMsg(args.Info.Count > 0, "Should always have implicit 'this'");
  654. Assert(!(callInfo.Flags & CallFlags_New));
  655. // If any of the args are missing, then it is Undefined type which causes TypeError exception.
  656. // strict type on both operands
  657. if (args.Info.Count >= 3 && JavascriptSIMDUint8x16::Is(args[1]) && JavascriptSIMDUint8x16::Is(args[2]))
  658. {
  659. JavascriptSIMDUint8x16 *a = JavascriptSIMDUint8x16::FromVar(args[1]);
  660. JavascriptSIMDUint8x16 *b = JavascriptSIMDUint8x16::FromVar(args[2]);
  661. Assert(a && b);
  662. SIMDValue result, aValue, bValue;
  663. aValue = a->GetValue();
  664. bValue = b->GetValue();
  665. result = SIMDUint8x16Operation::OpSubSaturate(aValue, bValue);
  666. return JavascriptSIMDUint8x16::New(&result, scriptContext);
  667. }
  668. JavascriptError::ThrowTypeError(scriptContext, JSERR_SimdUint8x16TypeMismatch, _u("subSaturate"));
  669. }
  670. Var SIMDUint8x16Lib::EntrySelect(RecyclableObject* function, CallInfo callInfo, ...)
  671. {
  672. PROBE_STACK(function->GetScriptContext(), Js::Constants::MinStackDefault);
  673. ARGUMENTS(args, callInfo);
  674. ScriptContext* scriptContext = function->GetScriptContext();
  675. AssertMsg(args.Info.Count > 0, "Should always have implicit 'this'");
  676. Assert(!(callInfo.Flags & CallFlags_New));
  677. if (args.Info.Count >= 4 && JavascriptSIMDBool8x16::Is(args[1]) &&
  678. JavascriptSIMDUint8x16::Is(args[2]) && JavascriptSIMDUint8x16::Is(args[3]))
  679. {
  680. JavascriptSIMDBool8x16 *m = JavascriptSIMDBool8x16::FromVar(args[1]);
  681. JavascriptSIMDUint8x16 *t = JavascriptSIMDUint8x16::FromVar(args[2]);
  682. JavascriptSIMDUint8x16 *f = JavascriptSIMDUint8x16::FromVar(args[3]);
  683. Assert(m && t && f);
  684. SIMDValue result, maskValue, trueValue, falseValue;
  685. maskValue = m->GetValue();
  686. trueValue = t->GetValue();
  687. falseValue = f->GetValue();
  688. result = SIMDInt32x4Operation::OpSelect(maskValue, trueValue, falseValue);
  689. return JavascriptSIMDUint8x16::New(&result, scriptContext);
  690. }
  691. JavascriptError::ThrowTypeError(scriptContext, JSERR_SimdUint8x16TypeMismatch, _u("select"));
  692. }
  693. }