JintUpdateExpression.cs 3.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103
  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 JintExpression _argument;
  9. private int _change;
  10. private bool _prefix;
  11. private JintIdentifierExpression _leftIdentifier;
  12. private bool _evalOrArguments;
  13. public JintUpdateExpression(Engine engine, UpdateExpression expression) : base(engine, expression)
  14. {
  15. _initialized = false;
  16. }
  17. protected override void Initialize()
  18. {
  19. var expression = (UpdateExpression) _expression;
  20. _prefix = expression.Prefix;
  21. _argument = Build(_engine, expression.Argument);
  22. if (expression.Operator == UnaryOperator.Increment)
  23. {
  24. _change = 1;
  25. }
  26. else if (expression.Operator == UnaryOperator.Decrement)
  27. {
  28. _change = - 1;
  29. }
  30. else
  31. {
  32. ExceptionHelper.ThrowArgumentException();
  33. }
  34. _leftIdentifier = _argument as JintIdentifierExpression;
  35. _evalOrArguments = _leftIdentifier?.HasEvalOrArguments == true;
  36. }
  37. protected override object EvaluateInternal()
  38. {
  39. var fastResult = _leftIdentifier != null
  40. ? UpdateIdentifier()
  41. : null;
  42. return fastResult ?? UpdateNonIdentifier();
  43. }
  44. private object UpdateNonIdentifier()
  45. {
  46. if (!(_argument.Evaluate() is Reference reference))
  47. {
  48. return ExceptionHelper.ThrowTypeError<object>(_engine, "Invalid left-hand side expression");
  49. }
  50. reference.AssertValid(_engine);
  51. var value = _engine.GetValue(reference, false);
  52. var isInteger = value._type == InternalTypes.Integer;
  53. var newValue = isInteger
  54. ? JsNumber.Create(value.AsInteger() + _change)
  55. : JsNumber.Create(TypeConverter.ToNumber(value) + _change);
  56. _engine.PutValue(reference, newValue);
  57. _engine._referencePool.Return(reference);
  58. return _prefix
  59. ? newValue
  60. : (isInteger ? value : JsNumber.Create(TypeConverter.ToNumber(value)));
  61. }
  62. private JsValue UpdateIdentifier()
  63. {
  64. var strict = StrictModeScope.IsStrictModeCode;
  65. var name = _leftIdentifier._expressionName;
  66. if (TryGetIdentifierEnvironmentWithBindingValue(
  67. name,
  68. out var environmentRecord,
  69. out var value))
  70. {
  71. if (strict && _evalOrArguments)
  72. {
  73. ExceptionHelper.ThrowSyntaxError(_engine);
  74. }
  75. var isInteger = value._type == InternalTypes.Integer;
  76. var newValue = isInteger
  77. ? JsNumber.Create(value.AsInteger() + _change)
  78. : JsNumber.Create(TypeConverter.ToNumber(value) + _change);
  79. environmentRecord.SetMutableBinding(name.Key.Name, newValue, strict);
  80. return _prefix
  81. ? newValue
  82. : (isInteger ? value : JsNumber.Create(TypeConverter.ToNumber(value)));
  83. }
  84. return null;
  85. }
  86. }
  87. }