JintCallExpression.cs 5.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160
  1. using Esprima.Ast;
  2. using Jint.Native;
  3. using Jint.Native.Function;
  4. using Jint.Runtime.Environments;
  5. using Jint.Runtime.References;
  6. namespace Jint.Runtime.Interpreter.Expressions
  7. {
  8. internal sealed class JintCallExpression : JintExpression
  9. {
  10. private CachedArgumentsHolder _cachedArguments;
  11. private bool _cached;
  12. private readonly JintExpression _calleeExpression;
  13. private bool _hasSpreads;
  14. public JintCallExpression(Engine engine, CallExpression expression) : base(engine, expression)
  15. {
  16. _initialized = false;
  17. _calleeExpression = Build(engine, expression.Callee);
  18. }
  19. protected override void Initialize()
  20. {
  21. var expression = (CallExpression) _expression;
  22. var cachedArgumentsHolder = new CachedArgumentsHolder
  23. {
  24. JintArguments = new JintExpression[expression.Arguments.Count]
  25. };
  26. bool CanSpread(Node e)
  27. {
  28. return e?.Type == Nodes.SpreadElement
  29. || e is AssignmentExpression ae && ae.Right?.Type == Nodes.SpreadElement;
  30. }
  31. bool cacheable = true;
  32. for (var i = 0; i < expression.Arguments.Count; i++)
  33. {
  34. var expressionArgument = expression.Arguments[i];
  35. cachedArgumentsHolder.JintArguments[i] = Build(_engine, expressionArgument);
  36. cacheable &= expressionArgument.Type == Nodes.Literal;
  37. _hasSpreads |= CanSpread(expressionArgument);
  38. if (expressionArgument is ArrayExpression ae)
  39. {
  40. for (var elementIndex = 0; elementIndex < ae.Elements.Count; elementIndex++)
  41. {
  42. _hasSpreads |= CanSpread(ae.Elements[elementIndex]);
  43. }
  44. }
  45. }
  46. if (cacheable)
  47. {
  48. _cached = true;
  49. var arguments = System.Array.Empty<JsValue>();
  50. if (cachedArgumentsHolder.JintArguments.Length > 0)
  51. {
  52. arguments = new JsValue[cachedArgumentsHolder.JintArguments.Length];
  53. BuildArguments(cachedArgumentsHolder.JintArguments, arguments);
  54. }
  55. cachedArgumentsHolder.CachedArguments = arguments;
  56. }
  57. _cachedArguments = cachedArgumentsHolder;
  58. }
  59. protected override object EvaluateInternal()
  60. {
  61. var callee = _calleeExpression.Evaluate();
  62. var expression = (CallExpression) _expression;
  63. // todo: implement as in http://www.ecma-international.org/ecma-262/5.1/#sec-11.2.4
  64. var cachedArguments = _cachedArguments;
  65. var arguments = System.Array.Empty<JsValue>();
  66. if (_cached)
  67. {
  68. arguments = cachedArguments.CachedArguments;
  69. }
  70. else
  71. {
  72. if (cachedArguments.JintArguments.Length > 0)
  73. {
  74. if (_hasSpreads)
  75. {
  76. arguments = BuildArgumentsWithSpreads(cachedArguments.JintArguments);
  77. }
  78. else
  79. {
  80. arguments = _engine._jsValueArrayPool.RentArray(cachedArguments.JintArguments.Length);
  81. BuildArguments(cachedArguments.JintArguments, arguments);
  82. }
  83. }
  84. }
  85. var func = _engine.GetValue(callee, false);
  86. var r = callee as Reference;
  87. if (func._type == InternalTypes.Undefined)
  88. {
  89. ExceptionHelper.ThrowTypeError(_engine, r == null ? "" : $"Object has no method '{r.GetReferencedName()}'");
  90. }
  91. if (!func.IsObject())
  92. {
  93. if (!_engine._referenceResolver.TryGetCallable(_engine, callee, out func))
  94. {
  95. ExceptionHelper.ThrowTypeError(_engine, r == null ? "" : $"Property '{r.GetReferencedName()}' of object is not a function");
  96. }
  97. }
  98. if (!(func is ICallable callable))
  99. {
  100. var message = $"{r?.GetReferencedName() ?? ""} is not a function";
  101. return ExceptionHelper.ThrowTypeError<object>(_engine, message);
  102. }
  103. var thisObject = Undefined.Instance;
  104. if (r != null)
  105. {
  106. var baseValue = r.GetBase();
  107. if ((baseValue._type & InternalTypes.ObjectEnvironmentRecord) == 0)
  108. {
  109. thisObject = r.GetThisValue();
  110. }
  111. else
  112. {
  113. var env = (EnvironmentRecord) baseValue;
  114. thisObject = env.ImplicitThisValue();
  115. }
  116. // is it a direct call to eval ? http://www.ecma-international.org/ecma-262/5.1/#sec-15.1.2.1.1
  117. if (r.GetReferencedName() == CommonProperties.Eval && callable is EvalFunctionInstance instance)
  118. {
  119. var value = instance.Call(thisObject, arguments, true);
  120. _engine._referencePool.Return(r);
  121. return value;
  122. }
  123. }
  124. var result = _engine.Call(callable, thisObject, arguments, expression.Location);
  125. if (!_cached && arguments.Length > 0)
  126. {
  127. _engine._jsValueArrayPool.ReturnArray(arguments);
  128. }
  129. _engine._referencePool.Return(r);
  130. return result;
  131. }
  132. private class CachedArgumentsHolder
  133. {
  134. internal JintExpression[] JintArguments;
  135. internal JsValue[] CachedArguments;
  136. }
  137. }
  138. }