JintUpdateExpression.cs 2.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990
  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 == KnownKeys.Eval
  31. || _leftIdentifier?.ExpressionName == KnownKeys.Arguments;
  32. }
  33. protected override object EvaluateInternal()
  34. {
  35. var fastResult = _leftIdentifier != null
  36. ? UpdateIdentifier()
  37. : null;
  38. return fastResult ?? UpdateNonIdentifier();
  39. }
  40. private object UpdateNonIdentifier()
  41. {
  42. if (!(_argument.Evaluate() is Reference reference))
  43. {
  44. ExceptionHelper.ThrowError(_engine, "Invalid left-hand side expression");
  45. return null;
  46. }
  47. reference.AssertValid(_engine);
  48. var oldValue = TypeConverter.ToNumber(_engine.GetValue(reference, false));
  49. var newValue = oldValue + _change;
  50. _engine.PutValue(reference, newValue);
  51. _engine._referencePool.Return(reference);
  52. return JsNumber.Create(_prefix ? newValue : oldValue);
  53. }
  54. private JsValue UpdateIdentifier()
  55. {
  56. var strict = StrictModeScope.IsStrictModeCode;
  57. ref readonly var name = ref _leftIdentifier.ExpressionName;
  58. if (TryGetIdentifierEnvironmentWithBindingValue(
  59. name,
  60. out var environmentRecord,
  61. out var value))
  62. {
  63. if (strict && _evalOrArguments)
  64. {
  65. ExceptionHelper.ThrowSyntaxError(_engine);
  66. }
  67. var oldValue = TypeConverter.ToNumber(value);
  68. var newValue = oldValue + _change;
  69. environmentRecord.SetMutableBinding(name, newValue, strict);
  70. return JsNumber.Create(_prefix ? newValue : oldValue);
  71. }
  72. return null;
  73. }
  74. }
  75. }