JintTryStatement.cs 2.8 KB

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