JintArrayExpression.cs 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133
  1. using Esprima.Ast;
  2. using Jint.Native;
  3. using Jint.Native.Iterator;
  4. namespace Jint.Runtime.Interpreter.Expressions
  5. {
  6. internal sealed class JintArrayExpression : JintExpression
  7. {
  8. private JintExpression?[] _expressions = Array.Empty<JintExpression?>();
  9. private bool _hasSpreads;
  10. private JintArrayExpression(ArrayExpression expression) : base(expression)
  11. {
  12. _initialized = false;
  13. }
  14. public static JintExpression Build(ArrayExpression expression)
  15. {
  16. return expression.Elements.Count == 0
  17. ? JintEmptyArrayExpression.Instance
  18. : new JintArrayExpression(expression);
  19. }
  20. protected override void Initialize(EvaluationContext context)
  21. {
  22. ref readonly var elements = ref ((ArrayExpression) _expression).Elements;
  23. var expressions = _expressions = new JintExpression[((ArrayExpression) _expression).Elements.Count];
  24. for (var n = 0; n < expressions.Length; n++)
  25. {
  26. var expr = elements[n];
  27. if (expr != null)
  28. {
  29. var expression = Build(expr);
  30. expressions[n] = expression;
  31. _hasSpreads |= expr.Type == Nodes.SpreadElement;
  32. }
  33. }
  34. // we get called from nested spread expansion in call
  35. _initialized = true;
  36. }
  37. protected override object EvaluateInternal(EvaluationContext context)
  38. {
  39. var engine = context.Engine;
  40. var a = engine.Realm.Intrinsics.Array.ArrayCreate(_hasSpreads ? 0 : (uint) _expressions.Length);
  41. uint arrayIndexCounter = 0;
  42. foreach (var expr in _expressions)
  43. {
  44. if (expr == null)
  45. {
  46. a.SetIndexValue(arrayIndexCounter++, null, updateLength: false);
  47. }
  48. else if (_hasSpreads && expr is JintSpreadExpression jse)
  49. {
  50. jse.GetValueAndCheckIterator(context, out var objectInstance, out var iterator);
  51. // optimize for array
  52. if (objectInstance is JsArray ai)
  53. {
  54. var length = ai.GetLength();
  55. var newLength = arrayIndexCounter + length;
  56. a.EnsureCapacity(newLength);
  57. a.CopyValues(ai, sourceStartIndex: 0, targetStartIndex: arrayIndexCounter, length);
  58. arrayIndexCounter += length;
  59. a.SetLength(newLength);
  60. }
  61. else
  62. {
  63. var protocol = new ArraySpreadProtocol(engine, a, iterator!, arrayIndexCounter);
  64. protocol.Execute();
  65. arrayIndexCounter += protocol._addedCount;
  66. }
  67. }
  68. else
  69. {
  70. var value = expr.GetValue(context);
  71. a.SetIndexValue(arrayIndexCounter++, value, updateLength: false);
  72. }
  73. }
  74. if (_hasSpreads)
  75. {
  76. a.SetLength(arrayIndexCounter);
  77. }
  78. return a;
  79. }
  80. private sealed class ArraySpreadProtocol : IteratorProtocol
  81. {
  82. private readonly JsArray _instance;
  83. private long _index;
  84. internal uint _addedCount;
  85. public ArraySpreadProtocol(
  86. Engine engine,
  87. JsArray instance,
  88. IteratorInstance iterator,
  89. long startIndex) : base(engine, iterator, 0)
  90. {
  91. _instance = instance;
  92. _index = startIndex - 1;
  93. }
  94. protected override void ProcessItem(JsValue[] args, JsValue currentValue)
  95. {
  96. _index++;
  97. _addedCount++;
  98. _instance.SetIndexValue((uint) _index, currentValue, updateLength: false);
  99. }
  100. }
  101. internal sealed class JintEmptyArrayExpression : JintExpression
  102. {
  103. public static JintEmptyArrayExpression Instance = new(new ArrayExpression(NodeList.Create(System.Linq.Enumerable.Empty<Expression?>())));
  104. private JintEmptyArrayExpression(Expression expression) : base(expression)
  105. {
  106. }
  107. protected override object EvaluateInternal(EvaluationContext context)
  108. {
  109. return new JsArray(context.Engine, Array.Empty<JsValue>());
  110. }
  111. public override JsValue GetValue(EvaluationContext context)
  112. {
  113. return new JsArray(context.Engine, Array.Empty<JsValue>());
  114. }
  115. }
  116. }
  117. }