JintTryStatement.cs 2.5 KB

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