JintUpdateExpression.cs 2.8 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889
  1. using Esprima.Ast;
  2. using Jint.Native;
  3. using Jint.Runtime.References;
  4. namespace Jint.Runtime.Interpreter.Expressions
  5. {
  6. internal sealed class JintUpdateExpression : JintExpression
  7. {
  8. private readonly JintExpression _argument;
  9. private readonly int _change;
  10. private readonly bool _prefix;
  11. private readonly JintIdentifierExpression _leftIdentifier;
  12. private readonly bool _evalOrArguments;
  13. public JintUpdateExpression(Engine engine, UpdateExpression expression) : base(engine, expression)
  14. {
  15. _prefix = expression.Prefix;
  16. _argument = Build(engine, expression.Argument);
  17. if (expression.Operator == UnaryOperator.Increment)
  18. {
  19. _change = 1;
  20. }
  21. else if (expression.Operator == UnaryOperator.Decrement)
  22. {
  23. _change = - 1;
  24. }
  25. else
  26. {
  27. ExceptionHelper.ThrowArgumentException();
  28. }
  29. _leftIdentifier = _argument as JintIdentifierExpression;
  30. _evalOrArguments = _leftIdentifier?._expressionName == "eval" || _leftIdentifier?._expressionName == "arguments";
  31. }
  32. protected override object EvaluateInternal()
  33. {
  34. var fastResult = _leftIdentifier != null
  35. ? UpdateIdentifier()
  36. : null;
  37. return fastResult ?? UpdateNonIdentifier();
  38. }
  39. private object UpdateNonIdentifier()
  40. {
  41. if (!(_argument.Evaluate() is Reference reference))
  42. {
  43. ExceptionHelper.ThrowError(_engine, "Invalid left-hand side expression");
  44. return null;
  45. }
  46. reference.AssertValid(_engine);
  47. var oldValue = TypeConverter.ToNumber(_engine.GetValue(reference, false));
  48. var newValue = oldValue + _change;
  49. _engine.PutValue(reference, newValue);
  50. _engine._referencePool.Return(reference);
  51. return JsNumber.Create(_prefix ? newValue : oldValue);
  52. }
  53. private JsValue UpdateIdentifier()
  54. {
  55. var strict = StrictModeScope.IsStrictModeCode;
  56. var name = _leftIdentifier._expressionName;
  57. if (TryGetIdentifierEnvironmentWithBindingValue(
  58. name,
  59. out var environmentRecord,
  60. out var value))
  61. {
  62. if (strict && _evalOrArguments)
  63. {
  64. ExceptionHelper.ThrowSyntaxError(_engine);
  65. }
  66. var oldValue = TypeConverter.ToNumber(value);
  67. var newValue = oldValue + _change;
  68. environmentRecord.SetMutableBinding(name, newValue, strict);
  69. return JsNumber.Create(_prefix ? newValue : oldValue);
  70. }
  71. return null;
  72. }
  73. }
  74. }