ArrayPrototype.cs 34 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066
  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. throw new JavaScriptException(Engine.TypeError);
  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. throw new JavaScriptException(Engine.TypeError);
  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. throw new JavaScriptException(Engine.TypeError, "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 => throw new JavaScriptException(Engine.TypeError, "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").TryCast<ICallable>(x => throw new JavaScriptException(Engine.TypeError));
  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").TryCast<ICallable>(x => throw new JavaScriptException(Engine.TypeError));
  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 => { func = Engine.Object.PrototypeObject.Get("toString").TryCast<ICallable>(y => throw new ArgumentException()); });
  716. return func.Call(array, Arguments.Empty);
  717. }
  718. private JsValue ReduceRight(JsValue thisObj, JsValue[] arguments)
  719. {
  720. var callbackfn = arguments.At(0);
  721. var initialValue = arguments.At(1);
  722. var o = TypeConverter.ToObject(Engine, thisObj);
  723. var lenValue = o.Get("length");
  724. var len = TypeConverter.ToUint32(lenValue);
  725. var callable = GetCallable(callbackfn);
  726. if (len == 0 && arguments.Length < 2)
  727. {
  728. throw new JavaScriptException(Engine.TypeError);
  729. }
  730. int k = (int) len - 1;
  731. JsValue accumulator = Undefined;
  732. if (arguments.Length > 1)
  733. {
  734. accumulator = initialValue;
  735. }
  736. else
  737. {
  738. var kPresent = false;
  739. while (kPresent == false && k >= 0)
  740. {
  741. var pk = TypeConverter.ToString(k);
  742. kPresent = o.HasProperty(pk);
  743. if (kPresent)
  744. {
  745. accumulator = o.Get(pk);
  746. }
  747. k--;
  748. }
  749. if (kPresent == false)
  750. {
  751. throw new JavaScriptException(Engine.TypeError);
  752. }
  753. }
  754. for (; k >= 0; k--)
  755. {
  756. var pk = TypeConverter.ToString(k);
  757. var kPresent = o.HasProperty(pk);
  758. if (kPresent)
  759. {
  760. var kvalue = o.Get(pk);
  761. accumulator = callable.Call(Undefined, new[] {accumulator, kvalue, k, o});
  762. }
  763. }
  764. return accumulator;
  765. }
  766. public JsValue Push(JsValue thisObject, JsValue[] arguments)
  767. {
  768. if (thisObject is ArrayInstance arrayInstance)
  769. {
  770. return arrayInstance.Push(arguments);
  771. }
  772. var o = TypeConverter.ToObject(Engine, thisObject);
  773. var lenVal = TypeConverter.ToNumber(o.Get("length"));
  774. // cast to double as we need to prevent an overflow
  775. double n = TypeConverter.ToUint32(lenVal);
  776. for (var i = 0; i < arguments.Length; i++)
  777. {
  778. o.Put(TypeConverter.ToString(n), arguments[i], true);
  779. n++;
  780. }
  781. o.Put("length", n, true);
  782. return n;
  783. }
  784. public JsValue Pop(JsValue thisObject, JsValue[] arguments)
  785. {
  786. var o = ArrayOperations.For(Engine, thisObject);
  787. var lenVal = TypeConverter.ToNumber(o.Target.Get("length"));
  788. uint len = TypeConverter.ToUint32(lenVal);
  789. if (len == 0)
  790. {
  791. o.SetLength(0);
  792. return Undefined;
  793. }
  794. len = len - 1;
  795. string indx = TypeConverter.ToString(len);
  796. JsValue element = o.Target.Get(indx);
  797. o.Target.Delete(indx, true);
  798. o.Target.Put("length", len, true);
  799. return element;
  800. }
  801. /// <summary>
  802. /// Adapter to use optimized array operations when possible.
  803. /// Gaps the difference between ArgumensInstance and ArrayInstance.
  804. /// </summary>
  805. private abstract class ArrayOperations
  806. {
  807. public abstract ObjectInstance Target { get; }
  808. public virtual uint GetSmallestIndex() => 0;
  809. public abstract uint GetLength();
  810. public abstract void SetLength(uint length);
  811. public virtual void EnsureCapacity(uint capacity)
  812. {
  813. }
  814. public abstract JsValue Get(uint index);
  815. public abstract bool TryGetValue(uint index, out JsValue value);
  816. public abstract void Put(uint index, JsValue value, bool throwOnError);
  817. public abstract void DeleteAt(uint index);
  818. public static ArrayOperations For(Engine engine, JsValue thisObj)
  819. {
  820. var instance = TypeConverter.ToObject(engine, thisObj);
  821. return For(instance);
  822. }
  823. public static ArrayOperations For(ObjectInstance instance)
  824. {
  825. if (instance is ArrayInstance arrayInstance)
  826. {
  827. return new ArrayInstanceOperations(arrayInstance);
  828. }
  829. return new ObjectInstanceOperations(instance);
  830. }
  831. private class ObjectInstanceOperations : ArrayOperations
  832. {
  833. private readonly ObjectInstance _instance;
  834. public ObjectInstanceOperations(ObjectInstance instance)
  835. {
  836. _instance = instance;
  837. }
  838. public override ObjectInstance Target => _instance;
  839. public override uint GetLength()
  840. {
  841. var desc = _instance.GetProperty("length");
  842. var descValue = desc.Value;
  843. if (desc.IsDataDescriptor() && !ReferenceEquals(descValue, null))
  844. {
  845. return TypeConverter.ToUint32(descValue);
  846. }
  847. var getter = desc.Get ?? Undefined;
  848. if (getter.IsUndefined())
  849. {
  850. return 0;
  851. }
  852. // if getter is not undefined it must be ICallable
  853. var callable = (ICallable) getter;
  854. var value = callable.Call(_instance, Arguments.Empty);
  855. return TypeConverter.ToUint32(value);
  856. }
  857. public override void SetLength(uint length) => _instance.Put("length", length, true);
  858. public override JsValue Get(uint index) => _instance.Get(TypeConverter.ToString(index));
  859. public override bool TryGetValue(uint index, out JsValue value)
  860. {
  861. var property = TypeConverter.ToString(index);
  862. var kPresent = _instance.HasProperty(property);
  863. value = kPresent ? _instance.Get(property) : Undefined;
  864. return kPresent;
  865. }
  866. public override void Put(uint index, JsValue value, bool throwOnError) => _instance.Put(TypeConverter.ToString(index), value, throwOnError);
  867. public override void DeleteAt(uint index) => _instance.Delete(TypeConverter.ToString(index), true);
  868. }
  869. private class ArrayInstanceOperations : ArrayOperations
  870. {
  871. private readonly ArrayInstance _array;
  872. public ArrayInstanceOperations(ArrayInstance array)
  873. {
  874. _array = array;
  875. }
  876. public override ObjectInstance Target => _array;
  877. public override uint GetSmallestIndex() => _array.GetSmallestIndex();
  878. public override uint GetLength() => _array.GetLength();
  879. public override void SetLength(uint length) => _array.Put("length", length, true);
  880. public override void EnsureCapacity(uint capacity)
  881. {
  882. _array.EnsureCapacity(capacity);
  883. }
  884. public override bool TryGetValue(uint index, out JsValue value)
  885. {
  886. return _array.TryGetValue(index, out value);
  887. }
  888. public override JsValue Get(uint index) => _array.Get(TypeConverter.ToString(index));
  889. public override void DeleteAt(uint index) => _array.DeleteAt(index);
  890. public override void Put(uint index, JsValue value, bool throwOnError) => _array.SetIndexValue(index, value, throwOnError);
  891. }
  892. }
  893. }
  894. }