EvalFunctionInstance.cs 4.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112
  1. using Esprima;
  2. using Jint.Runtime;
  3. using Jint.Runtime.Descriptors;
  4. using Jint.Runtime.Environments;
  5. using Jint.Runtime.Interpreter.Statements;
  6. namespace Jint.Native.Function
  7. {
  8. public sealed class EvalFunctionInstance : FunctionInstance
  9. {
  10. private static readonly ParserOptions ParserOptions = new ParserOptions { AdaptRegexp = true, Tolerant = false };
  11. public EvalFunctionInstance(Engine engine, string[] parameters, LexicalEnvironment scope, bool strict)
  12. : base(engine, "eval", parameters, scope, strict)
  13. {
  14. Prototype = Engine.Function.PrototypeObject;
  15. SetOwnProperty("length", new PropertyDescriptor(1, PropertyFlag.AllForbidden));
  16. }
  17. public override JsValue Call(JsValue thisObject, JsValue[] arguments)
  18. {
  19. return Call(thisObject, arguments, false);
  20. }
  21. public JsValue Call(JsValue thisObject, JsValue[] arguments, bool directCall)
  22. {
  23. var arg = arguments.At(0);
  24. if (arg.Type != Types.String)
  25. {
  26. return arg;
  27. }
  28. var code = TypeConverter.ToString(arg);
  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. var lexicalEnvironment = _engine.ExecutionContext.LexicalEnvironment;
  45. if (StrictModeScope.IsStrictModeCode)
  46. {
  47. strictVarEnv = LexicalEnvironment.NewDeclarativeEnvironment(Engine, lexicalEnvironment);
  48. Engine.EnterExecutionContext(strictVarEnv, strictVarEnv, Engine.ExecutionContext.ThisBinding);
  49. }
  50. bool argumentInstanceRented = Engine.DeclarationBindingInstantiation(
  51. DeclarationBindingType.EvalCode,
  52. program.HoistingScope,
  53. functionInstance: this,
  54. arguments);
  55. var statement = JintStatement.Build(_engine, program);
  56. var result = statement.Execute();
  57. var value = result.GetValueOrDefault();
  58. if (argumentInstanceRented)
  59. {
  60. lexicalEnvironment?._record?.FunctionWasCalled();
  61. _engine.ExecutionContext.VariableEnvironment?._record?.FunctionWasCalled();
  62. }
  63. if (result.Type == CompletionType.Throw)
  64. {
  65. var ex = new JavaScriptException(value).SetCallstack(_engine, result.Location);
  66. throw ex;
  67. }
  68. else
  69. {
  70. return value;
  71. }
  72. }
  73. finally
  74. {
  75. if (strictVarEnv != null)
  76. {
  77. Engine.LeaveExecutionContext();
  78. }
  79. if (!directCall)
  80. {
  81. Engine.LeaveExecutionContext();
  82. }
  83. }
  84. }
  85. }
  86. }
  87. catch (ParserException e)
  88. {
  89. if (e.Description == Messages.InvalidLHSInAssignment)
  90. {
  91. ExceptionHelper.ThrowReferenceError(_engine, (string) null);
  92. }
  93. ExceptionHelper.ThrowSyntaxError(_engine);
  94. return null;
  95. }
  96. }
  97. }
  98. }