2
0

ArrayPrototype.cs 34 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074
  1. using System;
  2. using System.Collections.Generic;
  3. using Jint.Native.Object;
  4. using Jint.Runtime;
  5. using Jint.Runtime.Descriptors;
  6. using Jint.Runtime.Interop;
  7. namespace Jint.Native.Array
  8. {
  9. /// <summary>
  10. /// http://www.ecma-international.org/ecma-262/5.1/#sec-15.4.4
  11. /// </summary>
  12. public sealed class ArrayPrototype : ArrayInstance
  13. {
  14. private ArrayPrototype(Engine engine) : base(engine)
  15. {
  16. }
  17. public static ArrayPrototype CreatePrototypeObject(Engine engine, ArrayConstructor arrayConstructor)
  18. {
  19. var obj = new ArrayPrototype(engine)
  20. {
  21. Extensible = true,
  22. Prototype = engine.Object.PrototypeObject
  23. };
  24. obj.FastAddProperty("length", 0, true, false, false);
  25. obj.SetOwnProperty("constructor", new PropertyDescriptor(arrayConstructor, PropertyFlag.NonEnumerable));
  26. return obj;
  27. }
  28. public void Configure()
  29. {
  30. FastAddProperty("toString", new ClrFunctionInstance(Engine, ToString, 0), true, false, true);
  31. FastAddProperty("toLocaleString", new ClrFunctionInstance(Engine, ToLocaleString), true, false, true);
  32. FastAddProperty("concat", new ClrFunctionInstance(Engine, Concat, 1), true, false, true);
  33. FastAddProperty("join", new ClrFunctionInstance(Engine, Join, 1), true, false, true);
  34. FastAddProperty("pop", new ClrFunctionInstance(Engine, Pop), true, false, true);
  35. FastAddProperty("push", new ClrFunctionInstance(Engine, Push, 1), true, false, true);
  36. FastAddProperty("reverse", new ClrFunctionInstance(Engine, Reverse), true, false, true);
  37. FastAddProperty("shift", new ClrFunctionInstance(Engine, Shift), true, false, true);
  38. FastAddProperty("slice", new ClrFunctionInstance(Engine, Slice, 2), true, false, true);
  39. FastAddProperty("sort", new ClrFunctionInstance(Engine, Sort, 1), true, false, true);
  40. FastAddProperty("splice", new ClrFunctionInstance(Engine, Splice, 2), true, false, true);
  41. FastAddProperty("unshift", new ClrFunctionInstance(Engine, Unshift, 1), true, false, true);
  42. FastAddProperty("indexOf", new ClrFunctionInstance(Engine, IndexOf, 1), true, false, true);
  43. FastAddProperty("lastIndexOf", new ClrFunctionInstance(Engine, LastIndexOf, 1), true, false, true);
  44. FastAddProperty("every", new ClrFunctionInstance(Engine, Every, 1), true, false, true);
  45. FastAddProperty("some", new ClrFunctionInstance(Engine, Some, 1), true, false, true);
  46. FastAddProperty("forEach", new ClrFunctionInstance(Engine, ForEach, 1), true, false, true);
  47. FastAddProperty("map", new ClrFunctionInstance(Engine, Map, 1), true, false, true);
  48. FastAddProperty("filter", new ClrFunctionInstance(Engine, Filter, 1), true, false, true);
  49. FastAddProperty("reduce", new ClrFunctionInstance(Engine, Reduce, 1), true, false, true);
  50. FastAddProperty("reduceRight", new ClrFunctionInstance(Engine, ReduceRight, 1), true, false, true);
  51. FastAddProperty("find", new ClrFunctionInstance(Engine, Find, 1), true, false, true);
  52. FastAddProperty("findIndex", new ClrFunctionInstance(Engine, FindIndex, 1), true, false, true);
  53. }
  54. private JsValue LastIndexOf(JsValue thisObj, JsValue[] arguments)
  55. {
  56. var o = ArrayOperations.For(Engine, thisObj);
  57. var len = o.GetLength();
  58. if (len == 0)
  59. {
  60. return -1;
  61. }
  62. var n = arguments.Length > 1 ? TypeConverter.ToInteger(arguments[1]) : len - 1;
  63. double k;
  64. if (n >= 0)
  65. {
  66. k = System.Math.Min(n, len - 1); // min
  67. }
  68. else
  69. {
  70. k = len - System.Math.Abs(n);
  71. }
  72. if (k < 0 || k > uint.MaxValue)
  73. {
  74. return -1;
  75. }
  76. var searchElement = arguments.At(0);
  77. var i = (uint) k;
  78. for (;; i--)
  79. {
  80. if (o.TryGetValue(i, out var value))
  81. {
  82. var same = ExpressionInterpreter.StrictlyEqual(value, searchElement);
  83. if (same)
  84. {
  85. return i;
  86. }
  87. }
  88. if (i == 0)
  89. {
  90. break;
  91. }
  92. }
  93. return -1;
  94. }
  95. private JsValue Reduce(JsValue thisObj, JsValue[] arguments)
  96. {
  97. var callbackfn = arguments.At(0);
  98. var initialValue = arguments.At(1);
  99. var o = ArrayOperations.For(Engine, thisObj);
  100. var len = o.GetLength();
  101. var callable = GetCallable(callbackfn);
  102. if (len == 0 && arguments.Length < 2)
  103. {
  104. ExceptionHelper.ThrowTypeError(Engine);
  105. }
  106. var k = 0;
  107. JsValue accumulator = Undefined;
  108. if (arguments.Length > 1)
  109. {
  110. accumulator = initialValue;
  111. }
  112. else
  113. {
  114. var kPresent = false;
  115. while (kPresent == false && k < len)
  116. {
  117. if (kPresent = o.TryGetValue((uint) k, out var temp))
  118. {
  119. accumulator = temp;
  120. }
  121. k++;
  122. }
  123. if (kPresent == false)
  124. {
  125. ExceptionHelper.ThrowTypeError(Engine);
  126. }
  127. }
  128. var args = new JsValue[4];
  129. while (k < len)
  130. {
  131. var i = (uint) k;
  132. if (o.TryGetValue(i, out var kvalue))
  133. {
  134. args[0] = accumulator;
  135. args[1] = kvalue;
  136. args[2] = i;
  137. args[3] = o.Target;
  138. accumulator = callable.Call(Undefined, args);
  139. }
  140. k++;
  141. }
  142. return accumulator;
  143. }
  144. private JsValue Filter(JsValue thisObj, JsValue[] arguments)
  145. {
  146. var callbackfn = arguments.At(0);
  147. var thisArg = arguments.At(1);
  148. var o = ArrayOperations.For(Engine, thisObj);
  149. var len = o.GetLength();
  150. var callable = GetCallable(callbackfn);
  151. var a = Engine.Array.ConstructFast(0);
  152. uint to = 0;
  153. var args = Engine.JsValueArrayPool.RentArray(3);
  154. for (uint k = 0; k < len; k++)
  155. {
  156. if (o.TryGetValue(k, out var kvalue))
  157. {
  158. args[0] = kvalue;
  159. args[1] = k;
  160. args[2] = o.Target;
  161. var selected = callable.Call(thisArg, args);
  162. if (TypeConverter.ToBoolean(selected))
  163. {
  164. a.SetIndexValue(to, kvalue, updateLength: false);
  165. to++;
  166. }
  167. }
  168. }
  169. a.SetLength(to);
  170. Engine.JsValueArrayPool.ReturnArray(args);
  171. return a;
  172. }
  173. private JsValue Map(JsValue thisObj, JsValue[] arguments)
  174. {
  175. if (thisObj is ArrayInstance arrayInstance)
  176. {
  177. return arrayInstance.Map(arguments);
  178. }
  179. var o = ArrayOperations.For(Engine, thisObj);
  180. var len = o.GetLength();
  181. var callbackfn = arguments.At(0);
  182. var thisArg = arguments.At(1);
  183. var callable = GetCallable(callbackfn);
  184. var jsValues = Engine.JsValueArrayPool.RentArray(1);
  185. jsValues[0] = len;
  186. var a = Engine.Array.Construct(jsValues, len);
  187. Engine.JsValueArrayPool.ReturnArray(jsValues);
  188. var args = Engine.JsValueArrayPool.RentArray(3);
  189. for (uint k = 0; k < len; k++)
  190. {
  191. if (o.TryGetValue(k, out var kvalue))
  192. {
  193. args[0] = kvalue;
  194. args[1] = k;
  195. args[2] = o.Target;
  196. var mappedValue = callable.Call(thisArg, args);
  197. a.SetIndexValue(k, mappedValue, updateLength: false);
  198. }
  199. }
  200. a.SetLength(len);
  201. Engine.JsValueArrayPool.ReturnArray(args);
  202. return a;
  203. }
  204. private JsValue ForEach(JsValue thisObj, JsValue[] arguments)
  205. {
  206. var callbackfn = arguments.At(0);
  207. var thisArg = arguments.At(1);
  208. var o = ArrayOperations.For(Engine, thisObj);
  209. var len = o.GetLength();
  210. var callable = GetCallable(callbackfn);
  211. var args = Engine.JsValueArrayPool.RentArray(3);
  212. for (uint k = 0; k < len; k++)
  213. {
  214. if (o.TryGetValue(k, out var kvalue))
  215. {
  216. args[0] = kvalue;
  217. args[1] = k;
  218. args[2] = o.Target;
  219. callable.Call(thisArg, args);
  220. }
  221. }
  222. Engine.JsValueArrayPool.ReturnArray(args);
  223. return Undefined;
  224. }
  225. private JsValue Some(JsValue thisObj, JsValue[] arguments)
  226. {
  227. var target = TypeConverter.ToObject(Engine, thisObj);
  228. return target.FindWithCallback(arguments, out _, out _);
  229. }
  230. private JsValue Every(JsValue thisObj, JsValue[] arguments)
  231. {
  232. var callbackfn = arguments.At(0);
  233. var thisArg = arguments.At(1);
  234. var o = ArrayOperations.For(Engine, thisObj);
  235. var len = o.GetLength();
  236. var callable = GetCallable(callbackfn);
  237. var args = Engine.JsValueArrayPool.RentArray(3);
  238. for (uint k = 0; k < len; k++)
  239. {
  240. if (o.TryGetValue(k, out var kvalue))
  241. {
  242. args[0] = kvalue;
  243. args[1] = k;
  244. args[2] = o.Target;
  245. var testResult = callable.Call(thisArg, args);
  246. if (false == TypeConverter.ToBoolean(testResult))
  247. {
  248. return JsBoolean.False;
  249. }
  250. }
  251. }
  252. Engine.JsValueArrayPool.ReturnArray(args);
  253. return JsBoolean.True;
  254. }
  255. private JsValue IndexOf(JsValue thisObj, JsValue[] arguments)
  256. {
  257. var o = ArrayOperations.For(Engine, thisObj);
  258. var len = o.GetLength();
  259. if (len == 0)
  260. {
  261. return -1;
  262. }
  263. var startIndex = arguments.Length > 1 ? TypeConverter.ToInteger(arguments[1]) : 0;
  264. if (startIndex > uint.MaxValue)
  265. {
  266. return -1;
  267. }
  268. uint k;
  269. if (startIndex < 0)
  270. {
  271. var abs = System.Math.Abs(startIndex);
  272. long temp = len - (uint) abs;
  273. if (abs > len || temp < 0)
  274. {
  275. temp = 0;
  276. }
  277. k = (uint) temp;
  278. }
  279. else
  280. {
  281. k = (uint) startIndex;
  282. }
  283. if (k >= len)
  284. {
  285. return -1;
  286. }
  287. uint smallestIndex = o.GetSmallestIndex();
  288. if (smallestIndex > k)
  289. {
  290. k = smallestIndex;
  291. }
  292. var searchElement = arguments.At(0);
  293. for (; k < len; k++)
  294. {
  295. if (o.TryGetValue(k, out var elementK))
  296. {
  297. var same = ExpressionInterpreter.StrictlyEqual(elementK, searchElement);
  298. if (same)
  299. {
  300. return k;
  301. }
  302. }
  303. }
  304. return -1;
  305. }
  306. private JsValue Find(JsValue thisObj, JsValue[] arguments)
  307. {
  308. var target = TypeConverter.ToObject(Engine, thisObj);
  309. target.FindWithCallback(arguments, out _, out var value);
  310. return value;
  311. }
  312. private JsValue FindIndex(JsValue thisObj, JsValue[] arguments)
  313. {
  314. var target = TypeConverter.ToObject(Engine, thisObj);
  315. if (target.FindWithCallback(arguments, out var index, out _))
  316. {
  317. return index;
  318. }
  319. return -1;
  320. }
  321. private JsValue Splice(JsValue thisObj, JsValue[] arguments)
  322. {
  323. var start = arguments.At(0);
  324. var deleteCount = arguments.At(1);
  325. var o = ArrayOperations.For(Engine, thisObj);
  326. var len = o.GetLength();
  327. var relativeStart = TypeConverter.ToInteger(start);
  328. uint actualStart;
  329. if (relativeStart < 0)
  330. {
  331. actualStart = (uint) System.Math.Max(len + relativeStart, 0);
  332. }
  333. else
  334. {
  335. actualStart = (uint) System.Math.Min(relativeStart, len);
  336. }
  337. var actualDeleteCount = (uint) System.Math.Min(System.Math.Max(TypeConverter.ToInteger(deleteCount), 0), len - actualStart);
  338. var a = Engine.Array.ConstructFast(actualDeleteCount);
  339. for (uint k = 0; k < actualDeleteCount; k++)
  340. {
  341. if (o.TryGetValue(actualStart + k, out var fromValue))
  342. {
  343. a.SetIndexValue(k, fromValue, updateLength: false);
  344. }
  345. }
  346. a.SetLength(actualDeleteCount);
  347. var items = System.Array.Empty<JsValue>();
  348. if (arguments.Length > 2)
  349. {
  350. items = new JsValue[arguments.Length - 2];
  351. System.Array.Copy(arguments, 2, items, 0, items.Length);
  352. }
  353. var length = len - actualDeleteCount + (uint) items.Length;
  354. o.EnsureCapacity(length);
  355. if (items.Length < actualDeleteCount)
  356. {
  357. for (uint k = actualStart; k < len - actualDeleteCount; k++)
  358. {
  359. var from = k + actualDeleteCount;
  360. var to = (uint) (k + items.Length);
  361. if (o.TryGetValue(from, out var fromValue))
  362. {
  363. o.Put(to, fromValue, true);
  364. }
  365. else
  366. {
  367. o.DeleteAt(to);
  368. }
  369. }
  370. for (var k = len; k > len - actualDeleteCount + items.Length; k--)
  371. {
  372. o.DeleteAt(k - 1);
  373. }
  374. }
  375. else if (items.Length > actualDeleteCount)
  376. {
  377. for (var k = len - actualDeleteCount; k > actualStart; k--)
  378. {
  379. var from = k + actualDeleteCount - 1;
  380. uint to = (uint) (k + items.Length - 1);
  381. if (o.TryGetValue(from, out var fromValue))
  382. {
  383. o.Put(to, fromValue, true);
  384. }
  385. else
  386. {
  387. o.DeleteAt(to);
  388. }
  389. }
  390. }
  391. for (uint k = 0; k < items.Length; k++)
  392. {
  393. var e = items[k];
  394. o.Put(k + actualStart, e, true);
  395. }
  396. o.SetLength(length);
  397. return a;
  398. }
  399. private JsValue Unshift(JsValue thisObj, JsValue[] arguments)
  400. {
  401. var o = ArrayOperations.For(Engine, thisObj);
  402. var len = o.GetLength();
  403. var argCount = (uint) arguments.Length;
  404. o.EnsureCapacity(len + argCount);
  405. for (var k = len; k > 0; k--)
  406. {
  407. var from = k - 1;
  408. var to = k + argCount - 1;
  409. if (o.TryGetValue(from, out var fromValue))
  410. {
  411. o.Put(to, fromValue, true);
  412. }
  413. else
  414. {
  415. o.DeleteAt(to);
  416. }
  417. }
  418. for (uint j = 0; j < argCount; j++)
  419. {
  420. o.Put(j, arguments[j], true);
  421. }
  422. o.SetLength(len + argCount);
  423. return len + argCount;
  424. }
  425. private JsValue Sort(JsValue thisObj, JsValue[] arguments)
  426. {
  427. if (!thisObj.IsObject())
  428. {
  429. ExceptionHelper.ThrowTypeError(_engine, "Array.prorotype.sort can only be applied on objects");
  430. }
  431. var obj = ArrayOperations.For(thisObj.AsObject());
  432. var len = obj.GetLength();
  433. if (len <= 1)
  434. {
  435. return obj.Target;
  436. }
  437. var compareArg = arguments.At(0);
  438. ICallable compareFn = null;
  439. if (!compareArg.IsUndefined())
  440. {
  441. compareFn = compareArg.TryCast<ICallable>(x => ExceptionHelper.ThrowTypeError(_engine, "The sort argument must be a function"));
  442. }
  443. int Comparer(JsValue x, JsValue y)
  444. {
  445. var xUndefined = x.IsUndefined();
  446. var yUndefined = y.IsUndefined();
  447. if (xUndefined && yUndefined)
  448. {
  449. return 0;
  450. }
  451. if (xUndefined)
  452. {
  453. return 1;
  454. }
  455. if (yUndefined)
  456. {
  457. return -1;
  458. }
  459. if (compareFn != null)
  460. {
  461. var s = TypeConverter.ToNumber(compareFn.Call(Undefined, new[] {x, y}));
  462. if (s < 0)
  463. {
  464. return -1;
  465. }
  466. if (s > 0)
  467. {
  468. return 1;
  469. }
  470. return 0;
  471. }
  472. var xString = TypeConverter.ToString(x);
  473. var yString = TypeConverter.ToString(y);
  474. var r = System.String.CompareOrdinal(xString, yString);
  475. return r;
  476. }
  477. var array = new JsValue[len];
  478. for (uint i = 0; i < len; ++i)
  479. {
  480. array[i] = obj.Get(i);
  481. }
  482. // don't eat inner exceptions
  483. try
  484. {
  485. System.Array.Sort(array, Comparer);
  486. }
  487. catch (InvalidOperationException e)
  488. {
  489. throw e.InnerException;
  490. }
  491. for (uint i = 0; i < len; ++i)
  492. {
  493. obj.Put(i, array[i], false);
  494. }
  495. return obj.Target;
  496. }
  497. private JsValue Slice(JsValue thisObj, JsValue[] arguments)
  498. {
  499. var start = arguments.At(0);
  500. var end = arguments.At(1);
  501. var o = ArrayOperations.For(Engine, thisObj);
  502. var len = o.GetLength();
  503. var relativeStart = TypeConverter.ToInteger(start);
  504. uint k;
  505. if (relativeStart < 0)
  506. {
  507. k = (uint) System.Math.Max(len + relativeStart, 0);
  508. }
  509. else
  510. {
  511. k = (uint) System.Math.Min(TypeConverter.ToInteger(start), len);
  512. }
  513. uint final;
  514. if (end.IsUndefined())
  515. {
  516. final = TypeConverter.ToUint32(len);
  517. }
  518. else
  519. {
  520. double relativeEnd = TypeConverter.ToInteger(end);
  521. if (relativeEnd < 0)
  522. {
  523. final = (uint) System.Math.Max(len + relativeEnd, 0);
  524. }
  525. else
  526. {
  527. final = (uint) System.Math.Min(TypeConverter.ToInteger(relativeEnd), len);
  528. }
  529. }
  530. var a = Engine.Array.Construct(final - k);
  531. uint n = 0;
  532. for (; k < final; k++)
  533. {
  534. if (o.TryGetValue(k, out var kValue))
  535. {
  536. a.SetIndexValue(n, kValue, updateLength: true);
  537. }
  538. n++;
  539. }
  540. return a;
  541. }
  542. private JsValue Shift(JsValue thisObj, JsValue[] arg2)
  543. {
  544. var o = ArrayOperations.For(Engine, thisObj);
  545. var len = o.GetLength();
  546. if (len == 0)
  547. {
  548. o.SetLength(0);
  549. return Undefined;
  550. }
  551. var first = o.Get(0);
  552. for (uint k = 1; k < len; k++)
  553. {
  554. var to = k - 1;
  555. if (o.TryGetValue(k, out var fromVal))
  556. {
  557. o.Put(to, fromVal, true);
  558. }
  559. else
  560. {
  561. o.DeleteAt(to);
  562. }
  563. }
  564. o.DeleteAt(len - 1);
  565. o.SetLength(len - 1);
  566. return first;
  567. }
  568. private JsValue Reverse(JsValue thisObj, JsValue[] arguments)
  569. {
  570. var o = ArrayOperations.For(Engine, thisObj);
  571. var len = o.GetLength();
  572. var middle = (uint) System.Math.Floor(len / 2.0);
  573. uint lower = 0;
  574. while (lower != middle)
  575. {
  576. var upper = len - lower - 1;
  577. var lowerExists = o.TryGetValue(lower, out var lowerValue);
  578. var upperExists = o.TryGetValue(upper, out var upperValue);
  579. if (lowerExists && upperExists)
  580. {
  581. o.Put(lower, upperValue, true);
  582. o.Put(upper, lowerValue, true);
  583. }
  584. if (!lowerExists && upperExists)
  585. {
  586. o.Put(lower, upperValue, true);
  587. o.DeleteAt(upper);
  588. }
  589. if (lowerExists && !upperExists)
  590. {
  591. o.DeleteAt(lower);
  592. o.Put(upper, lowerValue, true);
  593. }
  594. lower++;
  595. }
  596. return o.Target;
  597. }
  598. private JsValue Join(JsValue thisObj, JsValue[] arguments)
  599. {
  600. var separator = arguments.At(0);
  601. var o = ArrayOperations.For(Engine, thisObj);
  602. var len = o.GetLength();
  603. if (separator.IsUndefined())
  604. {
  605. separator = ",";
  606. }
  607. var sep = TypeConverter.ToString(separator);
  608. // as per the spec, this has to be called after ToString(separator)
  609. if (len == 0)
  610. {
  611. return "";
  612. }
  613. string StringFromJsValue(JsValue value)
  614. {
  615. return value.IsUndefined() || value.IsNull()
  616. ? ""
  617. : TypeConverter.ToString(value);
  618. }
  619. var s = StringFromJsValue(o.Get(0));
  620. if (len == 1)
  621. {
  622. return s;
  623. }
  624. var sb = ArrayExecutionContext.Current.StringBuilder;
  625. sb.Clear();
  626. sb.Append(s);
  627. for (uint k = 1; k < len; k++)
  628. {
  629. sb.Append(sep);
  630. sb.Append(StringFromJsValue(o.Get(k)));
  631. }
  632. return sb.ToString();
  633. }
  634. private JsValue ToLocaleString(JsValue thisObj, JsValue[] arguments)
  635. {
  636. var array = ArrayOperations.For(Engine, thisObj);
  637. var len = array.GetLength();
  638. const string separator = ",";
  639. if (len == 0)
  640. {
  641. return "";
  642. }
  643. JsValue r;
  644. if (!array.TryGetValue(0, out var firstElement) || firstElement.IsNull() || firstElement.IsUndefined())
  645. {
  646. r = "";
  647. }
  648. else
  649. {
  650. var elementObj = TypeConverter.ToObject(Engine, firstElement);
  651. var func = elementObj.Get("toLocaleString") as ICallable ?? ExceptionHelper.ThrowTypeError<ICallable>(_engine);
  652. r = func.Call(elementObj, Arguments.Empty);
  653. }
  654. for (uint k = 1; k < len; k++)
  655. {
  656. string s = r + separator;
  657. if (!array.TryGetValue(k, out var nextElement) || nextElement.IsNull())
  658. {
  659. r = "";
  660. }
  661. else
  662. {
  663. var elementObj = TypeConverter.ToObject(Engine, nextElement);
  664. var func = elementObj.Get("toLocaleString") as ICallable ?? ExceptionHelper.ThrowTypeError<ICallable>(_engine);
  665. r = func.Call(elementObj, Arguments.Empty);
  666. }
  667. r = s + r;
  668. }
  669. return r;
  670. }
  671. private JsValue Concat(JsValue thisObj, JsValue[] arguments)
  672. {
  673. var o = TypeConverter.ToObject(Engine, thisObj);
  674. uint n = 0;
  675. var items = new List<JsValue>(arguments.Length + 1) {o};
  676. items.AddRange(arguments);
  677. // try to find best capacity
  678. uint capacity = 0;
  679. for (var i = 0; i < items.Count; i++)
  680. {
  681. var eArray = items[i] as ArrayInstance;
  682. capacity += eArray?.GetLength() ?? (uint) 1;
  683. }
  684. var a = Engine.Array.ConstructFast(capacity);
  685. for (var i = 0; i < items.Count; i++)
  686. {
  687. var e = items[i];
  688. if (e is ArrayInstance eArray)
  689. {
  690. var len = eArray.GetLength();
  691. for (uint k = 0; k < len; k++)
  692. {
  693. if (eArray.TryGetValue(k, out var subElement))
  694. {
  695. a.SetIndexValue(n, subElement, updateLength: false);
  696. }
  697. n++;
  698. }
  699. }
  700. else
  701. {
  702. a.SetIndexValue(n, e, updateLength: false);
  703. n++;
  704. }
  705. }
  706. // this is not in the specs, but is necessary in case the last element of the last
  707. // array doesn't exist, and thus the length would not be incremented
  708. a.DefineOwnProperty("length", new PropertyDescriptor(n, PropertyFlag.None), false);
  709. return a;
  710. }
  711. private JsValue ToString(JsValue thisObj, JsValue[] arguments)
  712. {
  713. var array = TypeConverter.ToObject(Engine, thisObj);
  714. ICallable func;
  715. func = array.Get("join").TryCast<ICallable>(x =>
  716. {
  717. func = Engine.Object.PrototypeObject.Get("toString").TryCast<ICallable>(y => ExceptionHelper.ThrowArgumentException());
  718. });
  719. return func.Call(array, Arguments.Empty);
  720. }
  721. private JsValue ReduceRight(JsValue thisObj, JsValue[] arguments)
  722. {
  723. var callbackfn = arguments.At(0);
  724. var initialValue = arguments.At(1);
  725. var o = TypeConverter.ToObject(Engine, thisObj);
  726. var lenValue = o.Get("length");
  727. var len = TypeConverter.ToUint32(lenValue);
  728. var callable = GetCallable(callbackfn);
  729. if (len == 0 && arguments.Length < 2)
  730. {
  731. ExceptionHelper.ThrowTypeError(Engine);
  732. }
  733. int k = (int) len - 1;
  734. JsValue accumulator = Undefined;
  735. if (arguments.Length > 1)
  736. {
  737. accumulator = initialValue;
  738. }
  739. else
  740. {
  741. var kPresent = false;
  742. while (kPresent == false && k >= 0)
  743. {
  744. var pk = TypeConverter.ToString(k);
  745. kPresent = o.HasProperty(pk);
  746. if (kPresent)
  747. {
  748. accumulator = o.Get(pk);
  749. }
  750. k--;
  751. }
  752. if (kPresent == false)
  753. {
  754. ExceptionHelper.ThrowTypeError(Engine);
  755. }
  756. }
  757. var jsValues = new JsValue[4];
  758. for (; k >= 0; k--)
  759. {
  760. var pk = TypeConverter.ToString(k);
  761. var kPresent = o.HasProperty(pk);
  762. if (kPresent)
  763. {
  764. var kvalue = o.Get(pk);
  765. jsValues[0] = accumulator;
  766. jsValues[1] = kvalue;
  767. jsValues[2] = k;
  768. jsValues[3] = o;
  769. accumulator = callable.Call(Undefined, jsValues);
  770. }
  771. }
  772. return accumulator;
  773. }
  774. public JsValue Push(JsValue thisObject, JsValue[] arguments)
  775. {
  776. if (thisObject is ArrayInstance arrayInstance)
  777. {
  778. return arrayInstance.Push(arguments);
  779. }
  780. var o = TypeConverter.ToObject(Engine, thisObject);
  781. var lenVal = TypeConverter.ToNumber(o.Get("length"));
  782. // cast to double as we need to prevent an overflow
  783. double n = TypeConverter.ToUint32(lenVal);
  784. for (var i = 0; i < arguments.Length; i++)
  785. {
  786. o.Put(TypeConverter.ToString(n), arguments[i], true);
  787. n++;
  788. }
  789. o.Put("length", n, true);
  790. return n;
  791. }
  792. public JsValue Pop(JsValue thisObject, JsValue[] arguments)
  793. {
  794. var o = ArrayOperations.For(Engine, thisObject);
  795. var lenVal = TypeConverter.ToNumber(o.Target.Get("length"));
  796. uint len = TypeConverter.ToUint32(lenVal);
  797. if (len == 0)
  798. {
  799. o.SetLength(0);
  800. return Undefined;
  801. }
  802. len = len - 1;
  803. string indx = TypeConverter.ToString(len);
  804. JsValue element = o.Target.Get(indx);
  805. o.Target.Delete(indx, true);
  806. o.Target.Put("length", len, true);
  807. return element;
  808. }
  809. /// <summary>
  810. /// Adapter to use optimized array operations when possible.
  811. /// Gaps the difference between ArgumensInstance and ArrayInstance.
  812. /// </summary>
  813. private abstract class ArrayOperations
  814. {
  815. public abstract ObjectInstance Target { get; }
  816. public virtual uint GetSmallestIndex() => 0;
  817. public abstract uint GetLength();
  818. public abstract void SetLength(uint length);
  819. public virtual void EnsureCapacity(uint capacity)
  820. {
  821. }
  822. public abstract JsValue Get(uint index);
  823. public abstract bool TryGetValue(uint index, out JsValue value);
  824. public abstract void Put(uint index, JsValue value, bool throwOnError);
  825. public abstract void DeleteAt(uint index);
  826. public static ArrayOperations For(Engine engine, JsValue thisObj)
  827. {
  828. var instance = TypeConverter.ToObject(engine, thisObj);
  829. return For(instance);
  830. }
  831. public static ArrayOperations For(ObjectInstance instance)
  832. {
  833. if (instance is ArrayInstance arrayInstance)
  834. {
  835. return new ArrayInstanceOperations(arrayInstance);
  836. }
  837. return new ObjectInstanceOperations(instance);
  838. }
  839. private class ObjectInstanceOperations : ArrayOperations
  840. {
  841. private readonly ObjectInstance _instance;
  842. public ObjectInstanceOperations(ObjectInstance instance)
  843. {
  844. _instance = instance;
  845. }
  846. public override ObjectInstance Target => _instance;
  847. public override uint GetLength()
  848. {
  849. var desc = _instance.GetProperty("length");
  850. var descValue = desc.Value;
  851. if (desc.IsDataDescriptor() && !ReferenceEquals(descValue, null))
  852. {
  853. return TypeConverter.ToUint32(descValue);
  854. }
  855. var getter = desc.Get ?? Undefined;
  856. if (getter.IsUndefined())
  857. {
  858. return 0;
  859. }
  860. // if getter is not undefined it must be ICallable
  861. var callable = (ICallable) getter;
  862. var value = callable.Call(_instance, Arguments.Empty);
  863. return TypeConverter.ToUint32(value);
  864. }
  865. public override void SetLength(uint length) => _instance.Put("length", length, true);
  866. public override JsValue Get(uint index) => _instance.Get(TypeConverter.ToString(index));
  867. public override bool TryGetValue(uint index, out JsValue value)
  868. {
  869. var property = TypeConverter.ToString(index);
  870. var kPresent = _instance.HasProperty(property);
  871. value = kPresent ? _instance.Get(property) : Undefined;
  872. return kPresent;
  873. }
  874. public override void Put(uint index, JsValue value, bool throwOnError) => _instance.Put(TypeConverter.ToString(index), value, throwOnError);
  875. public override void DeleteAt(uint index) => _instance.Delete(TypeConverter.ToString(index), true);
  876. }
  877. private class ArrayInstanceOperations : ArrayOperations
  878. {
  879. private readonly ArrayInstance _array;
  880. public ArrayInstanceOperations(ArrayInstance array)
  881. {
  882. _array = array;
  883. }
  884. public override ObjectInstance Target => _array;
  885. public override uint GetSmallestIndex() => _array.GetSmallestIndex();
  886. public override uint GetLength() => _array.GetLength();
  887. public override void SetLength(uint length) => _array.Put("length", length, true);
  888. public override void EnsureCapacity(uint capacity)
  889. {
  890. _array.EnsureCapacity(capacity);
  891. }
  892. public override bool TryGetValue(uint index, out JsValue value)
  893. {
  894. return _array.TryGetValue(index, out value);
  895. }
  896. public override JsValue Get(uint index) => _array.Get(TypeConverter.ToString(index));
  897. public override void DeleteAt(uint index) => _array.DeleteAt(index);
  898. public override void Put(uint index, JsValue value, bool throwOnError) => _array.SetIndexValue(index, value, throwOnError);
  899. }
  900. }
  901. }
  902. }