JintCallExpression.cs 9.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260
  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. thisValue = value;
  150. }
  151. else
  152. {
  153. var refEnv = (EnvironmentRecord) baseValue;
  154. thisValue = refEnv.WithBaseObject();
  155. }
  156. }
  157. }
  158. else
  159. {
  160. thisValue = Undefined.Instance;
  161. }
  162. var argList = ArgumentListEvaluation(context);
  163. if (!func.IsObject() && !engine._referenceResolver.TryGetCallable(engine, reference, out func))
  164. {
  165. var message = referenceRecord == null
  166. ? reference + " is not a function"
  167. : $"Property '{referenceRecord.GetReferencedName()}' of object is not a function";
  168. ExceptionHelper.ThrowTypeError(engine.Realm, message);
  169. }
  170. var callable = func as ICallable;
  171. if (callable is null)
  172. {
  173. var message = $"{referenceRecord?.GetReferencedName() ?? reference} is not a function";
  174. ExceptionHelper.ThrowTypeError(engine.Realm, message);
  175. }
  176. if (tailPosition)
  177. {
  178. // TODO tail call
  179. // PrepareForTailCall();
  180. }
  181. var result = engine.Call(callable, thisValue, argList, _calleeExpression);
  182. if (!_cached && argList.Length > 0)
  183. {
  184. engine._jsValueArrayPool.ReturnArray(argList);
  185. }
  186. engine._referencePool.Return(referenceRecord);
  187. return result;
  188. }
  189. /// <summary>
  190. /// https://tc39.es/ecma262/#sec-isintailposition
  191. /// </summary>
  192. private static bool IsInTailPosition(CallExpression call)
  193. {
  194. // TODO tail calls
  195. return false;
  196. }
  197. private JsValue[] ArgumentListEvaluation(EvaluationContext context)
  198. {
  199. var cachedArguments = _cachedArguments;
  200. var arguments = System.Array.Empty<JsValue>();
  201. if (_cached)
  202. {
  203. arguments = cachedArguments.CachedArguments;
  204. }
  205. else
  206. {
  207. if (cachedArguments.JintArguments.Length > 0)
  208. {
  209. if (_hasSpreads)
  210. {
  211. arguments = BuildArgumentsWithSpreads(context, cachedArguments.JintArguments);
  212. }
  213. else
  214. {
  215. arguments = context.Engine._jsValueArrayPool.RentArray(cachedArguments.JintArguments.Length);
  216. BuildArguments(context, cachedArguments.JintArguments, arguments);
  217. }
  218. }
  219. }
  220. return arguments;
  221. }
  222. private class CachedArgumentsHolder
  223. {
  224. internal JintExpression[] JintArguments;
  225. internal JsValue[] CachedArguments;
  226. }
  227. }
  228. }