JintCallExpression.cs 9.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258
  1. using Esprima.Ast;
  2. using Jint.Native;
  3. using Jint.Native.Function;
  4. using Jint.Native.Object;
  5. using Jint.Runtime.Environments;
  6. using Jint.Runtime.References;
  7. namespace Jint.Runtime.Interpreter.Expressions
  8. {
  9. internal sealed class JintCallExpression : JintExpression
  10. {
  11. private CachedArgumentsHolder _cachedArguments;
  12. private bool _cached;
  13. private JintExpression _calleeExpression;
  14. private bool _hasSpreads;
  15. public JintCallExpression(CallExpression expression) : base(expression)
  16. {
  17. _initialized = false;
  18. }
  19. protected override void Initialize(EvaluationContext context)
  20. {
  21. var engine = context.Engine;
  22. var expression = (CallExpression) _expression;
  23. _calleeExpression = Build(engine, expression.Callee);
  24. var cachedArgumentsHolder = new CachedArgumentsHolder
  25. {
  26. JintArguments = new JintExpression[expression.Arguments.Count]
  27. };
  28. static bool CanSpread(Node e)
  29. {
  30. return e?.Type == Nodes.SpreadElement
  31. || e is AssignmentExpression ae && ae.Right?.Type == Nodes.SpreadElement;
  32. }
  33. bool cacheable = true;
  34. for (var i = 0; i < expression.Arguments.Count; i++)
  35. {
  36. var expressionArgument = expression.Arguments[i];
  37. cachedArgumentsHolder.JintArguments[i] = Build(engine, expressionArgument);
  38. cacheable &= expressionArgument.Type == Nodes.Literal;
  39. _hasSpreads |= CanSpread(expressionArgument);
  40. if (expressionArgument is ArrayExpression ae)
  41. {
  42. for (var elementIndex = 0; elementIndex < ae.Elements.Count; elementIndex++)
  43. {
  44. _hasSpreads |= CanSpread(ae.Elements[elementIndex]);
  45. }
  46. }
  47. }
  48. if (cacheable)
  49. {
  50. _cached = true;
  51. var arguments = System.Array.Empty<JsValue>();
  52. if (cachedArgumentsHolder.JintArguments.Length > 0)
  53. {
  54. arguments = new JsValue[cachedArgumentsHolder.JintArguments.Length];
  55. BuildArguments(context, cachedArgumentsHolder.JintArguments, arguments);
  56. }
  57. cachedArgumentsHolder.CachedArguments = arguments;
  58. }
  59. _cachedArguments = cachedArgumentsHolder;
  60. }
  61. protected override ExpressionResult EvaluateInternal(EvaluationContext context)
  62. {
  63. return NormalCompletion(_calleeExpression is JintSuperExpression
  64. ? SuperCall(context)
  65. : Call(context)
  66. );
  67. }
  68. private JsValue SuperCall(EvaluationContext context)
  69. {
  70. var engine = context.Engine;
  71. var thisEnvironment = (FunctionEnvironmentRecord) engine.ExecutionContext.GetThisEnvironment();
  72. var newTarget = engine.GetNewTarget(thisEnvironment);
  73. var func = GetSuperConstructor(thisEnvironment);
  74. if (!func.IsConstructor)
  75. {
  76. ExceptionHelper.ThrowTypeError(engine.Realm, "Not a constructor");
  77. }
  78. var argList = ArgumentListEvaluation(context);
  79. var result = ((IConstructor) func).Construct(argList, newTarget);
  80. var thisER = (FunctionEnvironmentRecord) engine.ExecutionContext.GetThisEnvironment();
  81. return thisER.BindThisValue(result);
  82. }
  83. /// <summary>
  84. /// https://tc39.es/ecma262/#sec-getsuperconstructor
  85. /// </summary>
  86. private static ObjectInstance GetSuperConstructor(FunctionEnvironmentRecord thisEnvironment)
  87. {
  88. var envRec = thisEnvironment;
  89. var activeFunction = envRec._functionObject;
  90. var superConstructor = activeFunction.GetPrototypeOf();
  91. return superConstructor;
  92. }
  93. /// <summary>
  94. /// https://tc39.es/ecma262/#sec-function-calls
  95. /// </summary>
  96. private JsValue Call(EvaluationContext context)
  97. {
  98. var reference = _calleeExpression.Evaluate(context).Value;
  99. if (ReferenceEquals(reference, Undefined.Instance))
  100. {
  101. return Undefined.Instance;
  102. }
  103. var engine = context.Engine;
  104. var func = engine.GetValue(reference, false);
  105. if (reference is Reference referenceRecord
  106. && !referenceRecord.IsPropertyReference()
  107. && referenceRecord.GetReferencedName() == CommonProperties.Eval
  108. && ReferenceEquals(func, engine.Realm.Intrinsics.Eval))
  109. {
  110. var argList = ArgumentListEvaluation(context);
  111. if (argList.Length == 0)
  112. {
  113. return Undefined.Instance;
  114. }
  115. var evalFunctionInstance = (EvalFunctionInstance) func;
  116. var evalArg = argList[0];
  117. var strictCaller = StrictModeScope.IsStrictModeCode;
  118. var evalRealm = evalFunctionInstance._realm;
  119. var direct = !((CallExpression) _expression).Optional;
  120. var value = evalFunctionInstance.PerformEval(evalArg, evalRealm, strictCaller, direct);
  121. engine._referencePool.Return(referenceRecord);
  122. return value;
  123. }
  124. var thisCall = (CallExpression) _expression;
  125. var tailCall = IsInTailPosition(thisCall);
  126. return EvaluateCall(context, func, reference, thisCall.Arguments, tailCall);
  127. }
  128. /// <summary>
  129. /// https://tc39.es/ecma262/#sec-evaluatecall
  130. /// </summary>
  131. private JsValue EvaluateCall(EvaluationContext context, JsValue func, object reference, in NodeList<Expression> arguments, bool tailPosition)
  132. {
  133. JsValue thisValue;
  134. var referenceRecord = reference as Reference;
  135. var engine = context.Engine;
  136. if (referenceRecord is not null)
  137. {
  138. if (referenceRecord.IsPropertyReference())
  139. {
  140. thisValue = referenceRecord.GetThisValue();
  141. }
  142. else
  143. {
  144. var baseValue = referenceRecord.GetBase();
  145. // deviation from the spec to support null-propagation helper
  146. if (baseValue.IsNullOrUndefined()
  147. && engine._referenceResolver.TryUnresolvableReference(engine, referenceRecord, out var value))
  148. {
  149. return value;
  150. }
  151. var refEnv = (EnvironmentRecord) baseValue;
  152. thisValue = refEnv.WithBaseObject();
  153. }
  154. }
  155. else
  156. {
  157. thisValue = Undefined.Instance;
  158. }
  159. var argList = ArgumentListEvaluation(context);
  160. if (!func.IsObject() && !engine._referenceResolver.TryGetCallable(engine, reference, out func))
  161. {
  162. var message = referenceRecord == null
  163. ? reference + " is not a function"
  164. : $"Property '{referenceRecord.GetReferencedName()}' of object is not a function";
  165. ExceptionHelper.ThrowTypeError(engine.Realm, message);
  166. }
  167. var callable = func as ICallable;
  168. if (callable is null)
  169. {
  170. var message = $"{referenceRecord?.GetReferencedName() ?? reference} is not a function";
  171. ExceptionHelper.ThrowTypeError(engine.Realm, message);
  172. }
  173. if (tailPosition)
  174. {
  175. // TODO tail call
  176. // PrepareForTailCall();
  177. }
  178. var result = engine.Call(callable, thisValue, argList, _calleeExpression);
  179. if (!_cached && argList.Length > 0)
  180. {
  181. engine._jsValueArrayPool.ReturnArray(argList);
  182. }
  183. engine._referencePool.Return(referenceRecord);
  184. return result;
  185. }
  186. /// <summary>
  187. /// https://tc39.es/ecma262/#sec-isintailposition
  188. /// </summary>
  189. private static bool IsInTailPosition(CallExpression call)
  190. {
  191. // TODO tail calls
  192. return false;
  193. }
  194. private JsValue[] ArgumentListEvaluation(EvaluationContext context)
  195. {
  196. var cachedArguments = _cachedArguments;
  197. var arguments = System.Array.Empty<JsValue>();
  198. if (_cached)
  199. {
  200. arguments = cachedArguments.CachedArguments;
  201. }
  202. else
  203. {
  204. if (cachedArguments.JintArguments.Length > 0)
  205. {
  206. if (_hasSpreads)
  207. {
  208. arguments = BuildArgumentsWithSpreads(context, cachedArguments.JintArguments);
  209. }
  210. else
  211. {
  212. arguments = context.Engine._jsValueArrayPool.RentArray(cachedArguments.JintArguments.Length);
  213. BuildArguments(context, cachedArguments.JintArguments, arguments);
  214. }
  215. }
  216. }
  217. return arguments;
  218. }
  219. private class CachedArgumentsHolder
  220. {
  221. internal JintExpression[] JintArguments;
  222. internal JsValue[] CachedArguments;
  223. }
  224. }
  225. }