EvalFunctionInstance.cs 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115
  1. using Esprima;
  2. using Jint.Native.Argument;
  3. using Jint.Runtime;
  4. using Jint.Runtime.Descriptors;
  5. using Jint.Runtime.Environments;
  6. namespace Jint.Native.Function
  7. {
  8. public class EvalFunctionInstance: FunctionInstance
  9. {
  10. private static readonly ParserOptions ParserOptions = new ParserOptions { AdaptRegexp = true, Tolerant = false };
  11. private readonly Engine _engine;
  12. public EvalFunctionInstance(Engine engine, string[] parameters, LexicalEnvironment scope, bool strict) : base(engine, parameters, scope, strict)
  13. {
  14. _engine = engine;
  15. Prototype = Engine.Function.PrototypeObject;
  16. SetOwnProperty("length", new PropertyDescriptor(1, PropertyFlag.AllForbidden));
  17. }
  18. public override JsValue Call(JsValue thisObject, JsValue[] arguments)
  19. {
  20. return Call(thisObject, arguments, false);
  21. }
  22. public JsValue Call(JsValue thisObject, JsValue[] arguments, bool directCall)
  23. {
  24. if (arguments.At(0).Type != Types.String)
  25. {
  26. return arguments.At(0);
  27. }
  28. var code = TypeConverter.ToString(arguments.At(0));
  29. try
  30. {
  31. var parser = new JavaScriptParser(code, ParserOptions);
  32. var program = parser.ParseProgram(StrictModeScope.IsStrictModeCode);
  33. using (new StrictModeScope(program.Strict))
  34. {
  35. using (new EvalCodeScope())
  36. {
  37. LexicalEnvironment strictVarEnv = null;
  38. try
  39. {
  40. if (!directCall)
  41. {
  42. Engine.EnterExecutionContext(Engine.GlobalEnvironment, Engine.GlobalEnvironment, Engine.Global);
  43. }
  44. if (StrictModeScope.IsStrictModeCode)
  45. {
  46. strictVarEnv = LexicalEnvironment.NewDeclarativeEnvironment(Engine, Engine.ExecutionContext.LexicalEnvironment);
  47. Engine.EnterExecutionContext(strictVarEnv, strictVarEnv, Engine.ExecutionContext.ThisBinding);
  48. }
  49. bool argumentInstanceRented = Engine.DeclarationBindingInstantiation(
  50. DeclarationBindingType.EvalCode,
  51. program.HoistingScope.FunctionDeclarations,
  52. program.HoistingScope.VariableDeclarations,
  53. this,
  54. arguments);
  55. var result = _engine.ExecuteStatement(program);
  56. var value = result.GetValueOrDefault();
  57. // we can safely release arguments if they don't escape the scope
  58. if (argumentInstanceRented
  59. && Engine.ExecutionContext.LexicalEnvironment?.Record is DeclarativeEnvironmentRecord der
  60. && !(result.Value is ArgumentsInstance))
  61. {
  62. der.ReleaseArguments();
  63. }
  64. if (result.Type == Completion.Throw)
  65. {
  66. var ex = new JavaScriptException(value).SetCallstack(_engine, result.Location);
  67. _engine.CompletionPool.Return(result);
  68. throw ex;
  69. }
  70. else
  71. {
  72. _engine.CompletionPool.Return(result);
  73. return value;
  74. }
  75. }
  76. finally
  77. {
  78. if (strictVarEnv != null)
  79. {
  80. Engine.LeaveExecutionContext();
  81. }
  82. if (!directCall)
  83. {
  84. Engine.LeaveExecutionContext();
  85. }
  86. }
  87. }
  88. }
  89. }
  90. catch (ParserException e)
  91. {
  92. if (e.Description == Messages.InvalidLHSInAssignment)
  93. {
  94. throw new JavaScriptException(Engine.ReferenceError);
  95. }
  96. throw new JavaScriptException(Engine.SyntaxError);
  97. }
  98. }
  99. }
  100. }