JintArrayExpression.cs 4.3 KB

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