JintCallExpression.cs 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338
  1. using System.Diagnostics.CodeAnalysis;
  2. using System.Runtime.CompilerServices;
  3. using Esprima.Ast;
  4. using Jint.Native;
  5. using Jint.Native.Function;
  6. using Jint.Native.Object;
  7. using Jint.Runtime.Environments;
  8. using Jint.Runtime.References;
  9. namespace Jint.Runtime.Interpreter.Expressions
  10. {
  11. internal sealed class JintCallExpression : JintExpression
  12. {
  13. private CachedArgumentsHolder _cachedArguments = null!;
  14. private bool _cached;
  15. private JintExpression _calleeExpression = null!;
  16. private bool _hasSpreads;
  17. public JintCallExpression(CallExpression expression) : base(expression)
  18. {
  19. _initialized = false;
  20. }
  21. protected override void Initialize(EvaluationContext context)
  22. {
  23. var expression = (CallExpression) _expression;
  24. ref readonly var expressionArguments = ref expression.Arguments;
  25. _calleeExpression = Build(expression.Callee);
  26. var cachedArgumentsHolder = new CachedArgumentsHolder
  27. {
  28. JintArguments = new JintExpression[expressionArguments.Count]
  29. };
  30. static bool CanSpread(Node? e)
  31. {
  32. if (e is null)
  33. {
  34. return false;
  35. }
  36. return e.Type == Nodes.SpreadElement || e is AssignmentExpression { Right.Type: Nodes.SpreadElement };
  37. }
  38. var cacheable = true;
  39. for (var i = 0; i < expressionArguments.Count; i++)
  40. {
  41. var expressionArgument = expressionArguments[i];
  42. cachedArgumentsHolder.JintArguments[i] = Build(expressionArgument);
  43. cacheable &= expressionArgument.Type == Nodes.Literal;
  44. _hasSpreads |= CanSpread(expressionArgument);
  45. if (expressionArgument is ArrayExpression ae)
  46. {
  47. ref readonly var elements = ref ae.Elements;
  48. for (var elementIndex = 0; elementIndex < elements.Count; elementIndex++)
  49. {
  50. _hasSpreads |= CanSpread(elements[elementIndex]);
  51. }
  52. }
  53. }
  54. if (cacheable)
  55. {
  56. _cached = true;
  57. var arguments = Array.Empty<JsValue>();
  58. if (cachedArgumentsHolder.JintArguments.Length > 0)
  59. {
  60. arguments = new JsValue[cachedArgumentsHolder.JintArguments.Length];
  61. BuildArguments(context, cachedArgumentsHolder.JintArguments, arguments);
  62. }
  63. cachedArgumentsHolder.CachedArguments = arguments;
  64. }
  65. _cachedArguments = cachedArgumentsHolder;
  66. }
  67. protected override object EvaluateInternal(EvaluationContext context)
  68. {
  69. if (_calleeExpression._expression.Type == Nodes.Super)
  70. {
  71. return SuperCall(context);
  72. }
  73. // https://tc39.es/ecma262/#sec-function-calls
  74. var reference = _calleeExpression.Evaluate(context);
  75. if (ReferenceEquals(reference, JsValue.Undefined))
  76. {
  77. return JsValue.Undefined;
  78. }
  79. var engine = context.Engine;
  80. var func = engine.GetValue(reference, false);
  81. if (func.IsNullOrUndefined() && _expression.IsOptional())
  82. {
  83. return JsValue.Undefined;
  84. }
  85. var referenceRecord = reference as Reference;
  86. if (ReferenceEquals(func, engine.Realm.Intrinsics.Eval)
  87. && referenceRecord != null
  88. && !referenceRecord.IsPropertyReference
  89. && referenceRecord.ReferencedName == CommonProperties.Eval)
  90. {
  91. return HandleEval(context, func, engine, referenceRecord);
  92. }
  93. var thisCall = (CallExpression) _expression;
  94. var tailCall = IsInTailPosition(thisCall);
  95. // https://tc39.es/ecma262/#sec-evaluatecall
  96. JsValue thisObject;
  97. if (referenceRecord is not null)
  98. {
  99. if (referenceRecord.IsPropertyReference)
  100. {
  101. thisObject = referenceRecord.ThisValue;
  102. }
  103. else
  104. {
  105. var baseValue = referenceRecord.Base;
  106. // deviation from the spec to support null-propagation helper
  107. if (baseValue.IsNullOrUndefined()
  108. && engine._referenceResolver.TryUnresolvableReference(engine, referenceRecord, out var value))
  109. {
  110. thisObject = value;
  111. }
  112. else
  113. {
  114. var refEnv = (EnvironmentRecord) baseValue;
  115. thisObject = refEnv.WithBaseObject();
  116. }
  117. }
  118. }
  119. else
  120. {
  121. thisObject = JsValue.Undefined;
  122. }
  123. var arguments = ArgumentListEvaluation(context);
  124. if (!func.IsObject() && !engine._referenceResolver.TryGetCallable(engine, reference, out func))
  125. {
  126. ThrowMemberIsNotFunction(referenceRecord, reference, engine);
  127. }
  128. var callable = func as ICallable;
  129. if (callable is null)
  130. {
  131. ThrowReferenceNotFunction(referenceRecord, reference, engine);
  132. }
  133. if (tailCall)
  134. {
  135. // TODO tail call
  136. // PrepareForTailCall();
  137. }
  138. // ensure logic is in sync between Call, Construct and JintCallExpression!
  139. JsValue result;
  140. if (callable is FunctionInstance functionInstance)
  141. {
  142. var callStack = engine.CallStack;
  143. var recursionDepth = callStack.Push(functionInstance, _calleeExpression, engine.ExecutionContext);
  144. if (recursionDepth > engine.Options.Constraints.MaxRecursionDepth)
  145. {
  146. // automatically pops the current element as it was never reached
  147. ExceptionHelper.ThrowRecursionDepthOverflowException(callStack);
  148. }
  149. try
  150. {
  151. result = functionInstance.Call(thisObject, arguments);
  152. }
  153. finally
  154. {
  155. // if call stack was reset due to recursive call to engine or similar, we might not have it anymore
  156. if (callStack.Count > 0)
  157. {
  158. callStack.Pop();
  159. }
  160. }
  161. }
  162. else
  163. {
  164. result = callable.Call(thisObject, arguments);
  165. }
  166. if (!_cached && arguments.Length > 0)
  167. {
  168. engine._jsValueArrayPool.ReturnArray(arguments);
  169. }
  170. engine._referencePool.Return(referenceRecord);
  171. return result;
  172. }
  173. [DoesNotReturn]
  174. [MethodImpl(MethodImplOptions.NoInlining)]
  175. private static void ThrowReferenceNotFunction(Reference? referenceRecord1, object reference, Engine engine)
  176. {
  177. var message = $"{referenceRecord1?.ReferencedName ?? reference} is not a function";
  178. ExceptionHelper.ThrowTypeError(engine.Realm, message);
  179. }
  180. [DoesNotReturn]
  181. [MethodImpl(MethodImplOptions.NoInlining)]
  182. private static void ThrowMemberIsNotFunction(Reference? referenceRecord1, object reference, Engine engine)
  183. {
  184. var message = referenceRecord1 == null
  185. ? reference + " is not a function"
  186. : $"Property '{referenceRecord1.ReferencedName}' of object is not a function";
  187. ExceptionHelper.ThrowTypeError(engine.Realm, message);
  188. }
  189. private JsValue HandleEval(EvaluationContext context, JsValue func, Engine engine, Reference referenceRecord)
  190. {
  191. var argList = ArgumentListEvaluation(context);
  192. if (argList.Length == 0)
  193. {
  194. return JsValue.Undefined;
  195. }
  196. var evalFunctionInstance = (EvalFunctionInstance) func;
  197. var evalArg = argList[0];
  198. var strictCaller = StrictModeScope.IsStrictModeCode;
  199. var evalRealm = evalFunctionInstance._realm;
  200. var direct = !_expression.IsOptional();
  201. var value = evalFunctionInstance.PerformEval(evalArg, evalRealm, strictCaller, direct);
  202. engine._referencePool.Return(referenceRecord);
  203. return value;
  204. }
  205. private JsValue SuperCall(EvaluationContext context)
  206. {
  207. var engine = context.Engine;
  208. var thisEnvironment = (FunctionEnvironmentRecord) engine.ExecutionContext.GetThisEnvironment();
  209. var newTarget = engine.GetNewTarget(thisEnvironment);
  210. var func = GetSuperConstructor(thisEnvironment);
  211. if (func is null || !func.IsConstructor)
  212. {
  213. ExceptionHelper.ThrowTypeError(engine.Realm, "Not a constructor");
  214. }
  215. var defaultSuperCall = ReferenceEquals(_expression, ClassDefinition._defaultSuperCall);
  216. var argList = defaultSuperCall ? DefaultSuperCallArgumentListEvaluation(context) : ArgumentListEvaluation(context);
  217. var result = ((IConstructor) func).Construct(argList, newTarget);
  218. var thisER = (FunctionEnvironmentRecord) engine.ExecutionContext.GetThisEnvironment();
  219. thisER.BindThisValue(result);
  220. var F = thisER._functionObject;
  221. result.InitializeInstanceElements((ScriptFunctionInstance) F);
  222. return result;
  223. }
  224. /// <summary>
  225. /// https://tc39.es/ecma262/#sec-getsuperconstructor
  226. /// </summary>
  227. private static ObjectInstance? GetSuperConstructor(FunctionEnvironmentRecord thisEnvironment)
  228. {
  229. var envRec = thisEnvironment;
  230. var activeFunction = envRec._functionObject;
  231. var superConstructor = activeFunction.GetPrototypeOf();
  232. return superConstructor;
  233. }
  234. /// <summary>
  235. /// https://tc39.es/ecma262/#sec-isintailposition
  236. /// </summary>
  237. private static bool IsInTailPosition(CallExpression call)
  238. {
  239. // TODO tail calls
  240. return false;
  241. }
  242. private JsValue[] ArgumentListEvaluation(EvaluationContext context)
  243. {
  244. var cachedArguments = _cachedArguments;
  245. var arguments = Array.Empty<JsValue>();
  246. if (_cached)
  247. {
  248. arguments = cachedArguments.CachedArguments;
  249. }
  250. else
  251. {
  252. if (cachedArguments.JintArguments.Length > 0)
  253. {
  254. if (_hasSpreads)
  255. {
  256. arguments = BuildArgumentsWithSpreads(context, cachedArguments.JintArguments);
  257. }
  258. else
  259. {
  260. arguments = context.Engine._jsValueArrayPool.RentArray(cachedArguments.JintArguments.Length);
  261. BuildArguments(context, cachedArguments.JintArguments, arguments);
  262. }
  263. }
  264. }
  265. return arguments;
  266. }
  267. private JsValue[] DefaultSuperCallArgumentListEvaluation(EvaluationContext context)
  268. {
  269. // This branch behaves similarly to constructor(...args) { super(...args); }.
  270. // The most notable distinction is that while the aforementioned ECMAScript source text observably calls
  271. // the @@iterator method on %Array.prototype%, this function does not.
  272. var spreadExpression = (JintSpreadExpression) _cachedArguments.JintArguments[0];
  273. var array = (JsArray) spreadExpression._argument.GetValue(context);
  274. var length = array.GetLength();
  275. var args = new List<JsValue>((int) length);
  276. for (uint j = 0; j < length; ++j)
  277. {
  278. array.TryGetValue(j, out var value);
  279. args.Add(value);
  280. }
  281. return args.ToArray();
  282. }
  283. private sealed class CachedArgumentsHolder
  284. {
  285. internal JintExpression[] JintArguments = Array.Empty<JintExpression>();
  286. internal JsValue[] CachedArguments = Array.Empty<JsValue>();
  287. }
  288. }
  289. }