JintTryStatement.cs 2.8 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495
  1. using Esprima.Ast;
  2. using Jint.Native;
  3. using Jint.Runtime.Environments;
  4. namespace Jint.Runtime.Interpreter.Statements
  5. {
  6. /// <summary>
  7. /// https://tc39.es/ecma262/#sec-try-statement
  8. /// </summary>
  9. internal sealed class JintTryStatement : JintStatement<TryStatement>
  10. {
  11. private JintBlockStatement _block = null!;
  12. private JintBlockStatement? _catch;
  13. private JintBlockStatement? _finalizer;
  14. public JintTryStatement(TryStatement statement) : base(statement)
  15. {
  16. }
  17. protected override void Initialize(EvaluationContext context)
  18. {
  19. _block = new JintBlockStatement(_statement.Block);
  20. if (_statement.Finalizer != null)
  21. {
  22. _finalizer = new JintBlockStatement(_statement.Finalizer);
  23. }
  24. }
  25. internal override bool SupportsResume => true;
  26. protected override Completion ExecuteInternal(EvaluationContext context)
  27. {
  28. var engine = context.Engine;
  29. var b = _block.Execute(context);
  30. if (b.Type == CompletionType.Throw)
  31. {
  32. b = ExecuteCatch(context, b, engine);
  33. }
  34. if (_finalizer != null)
  35. {
  36. var f = _finalizer.Execute(context);
  37. if (f.Type == CompletionType.Normal)
  38. {
  39. return b;
  40. }
  41. return f.UpdateEmpty(JsValue.Undefined);
  42. }
  43. return b.UpdateEmpty(JsValue.Undefined);
  44. }
  45. private Completion ExecuteCatch(EvaluationContext context, Completion b, Engine engine)
  46. {
  47. // execute catch
  48. if (_statement.Handler is not null)
  49. {
  50. // initialize lazily
  51. if (_catch is null)
  52. {
  53. _catch = new JintBlockStatement(_statement.Handler.Body);
  54. }
  55. // https://tc39.es/ecma262/#sec-runtime-semantics-catchclauseevaluation
  56. var thrownValue = b.Value;
  57. var oldEnv = engine.ExecutionContext.LexicalEnvironment;
  58. var catchEnv = JintEnvironment.NewDeclarativeEnvironment(engine, oldEnv, catchEnvironment: true);
  59. var boundNames = new List<string>();
  60. _statement.Handler.Param.GetBoundNames(boundNames);
  61. foreach (var argName in boundNames)
  62. {
  63. catchEnv.CreateMutableBinding(argName, false);
  64. }
  65. engine.UpdateLexicalEnvironment(catchEnv);
  66. var catchParam = _statement.Handler?.Param;
  67. catchParam.BindingInitialization(context, thrownValue, catchEnv);
  68. b = _catch.Execute(context);
  69. engine.UpdateLexicalEnvironment(oldEnv);
  70. }
  71. return b;
  72. }
  73. }
  74. }