EvalFunctionInstance.cs 4.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108
  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. private static readonly JsString _functionName = new JsString("eval");
  12. public EvalFunctionInstance(Engine engine, string[] parameters, LexicalEnvironment scope, bool strict)
  13. : base(engine, _functionName, parameters, scope, strict)
  14. {
  15. _prototype = Engine.Function.PrototypeObject;
  16. _length = PropertyDescriptor.AllForbiddenDescriptor.NumberOne;
  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. var arg = arguments.At(0);
  25. if (arg.Type != Types.String)
  26. {
  27. return arg;
  28. }
  29. var code = TypeConverter.ToString(arg);
  30. try
  31. {
  32. var parser = new JavaScriptParser(code, ParserOptions);
  33. var program = parser.ParseScript(StrictModeScope.IsStrictModeCode);
  34. using (new StrictModeScope(program.Strict))
  35. {
  36. using (new EvalCodeScope())
  37. {
  38. LexicalEnvironment strictVarEnv = null;
  39. try
  40. {
  41. if (!directCall)
  42. {
  43. Engine.EnterExecutionContext(Engine.GlobalEnvironment, Engine.GlobalEnvironment, Engine.Global);
  44. }
  45. var lexicalEnvironment = _engine.ExecutionContext.LexicalEnvironment;
  46. if (StrictModeScope.IsStrictModeCode)
  47. {
  48. strictVarEnv = LexicalEnvironment.NewDeclarativeEnvironment(Engine, lexicalEnvironment);
  49. Engine.EnterExecutionContext(strictVarEnv, strictVarEnv, Engine.ExecutionContext.ThisBinding);
  50. }
  51. var argumentsInstance = Engine.DeclarationBindingInstantiation(
  52. DeclarationBindingType.EvalCode,
  53. program.HoistingScope,
  54. functionInstance: this,
  55. arguments);
  56. var statement = JintStatement.Build(_engine, program);
  57. var result = statement.Execute();
  58. var value = result.GetValueOrDefault();
  59. argumentsInstance?.FunctionWasCalled();
  60. if (result.Type == CompletionType.Throw)
  61. {
  62. var ex = new JavaScriptException(value).SetCallstack(_engine, result.Location);
  63. throw ex;
  64. }
  65. else
  66. {
  67. return value;
  68. }
  69. }
  70. finally
  71. {
  72. if (strictVarEnv != null)
  73. {
  74. Engine.LeaveExecutionContext();
  75. }
  76. if (!directCall)
  77. {
  78. Engine.LeaveExecutionContext();
  79. }
  80. }
  81. }
  82. }
  83. }
  84. catch (ParserException e)
  85. {
  86. if (e.Description == Messages.InvalidLHSInAssignment)
  87. {
  88. ExceptionHelper.ThrowReferenceError(_engine, (string) null);
  89. }
  90. return ExceptionHelper.ThrowSyntaxError<JsValue>(_engine);
  91. }
  92. }
  93. }
  94. }