JintTryStatement.cs 3.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100
  1. using System.Collections.Generic;
  2. using Esprima.Ast;
  3. using Jint.Native;
  4. using Jint.Runtime.Environments;
  5. namespace Jint.Runtime.Interpreter.Statements
  6. {
  7. /// <summary>
  8. /// https://tc39.es/ecma262/#sec-try-statement
  9. /// </summary>
  10. internal sealed class JintTryStatement : JintStatement<TryStatement>
  11. {
  12. private JintStatement _block;
  13. private JintStatement _catch;
  14. private JintStatement _finalizer;
  15. public JintTryStatement(TryStatement statement) : base(statement)
  16. {
  17. }
  18. protected override void Initialize(EvaluationContext context)
  19. {
  20. _block = Build(_statement.Block);
  21. if (_statement.Finalizer != null)
  22. {
  23. _finalizer = Build(_statement.Finalizer);
  24. }
  25. }
  26. protected override bool SupportsResume => true;
  27. protected override Completion ExecuteInternal(EvaluationContext context)
  28. {
  29. var engine = context.Engine;
  30. int callStackSizeBeforeExecution = engine.CallStack.Count;
  31. var b = _block.Execute(context);
  32. if (b.Type == CompletionType.Throw)
  33. {
  34. // initialize lazily
  35. if (_statement.Handler is not null && _catch is null)
  36. {
  37. _catch = Build(_statement.Handler.Body);
  38. }
  39. // execute catch
  40. if (_statement.Handler is not null)
  41. {
  42. // Quick-patch for call stack not being unwinded when an exception is caught.
  43. // Ideally, this should instead be solved by always popping the stack when returning
  44. // from a call, regardless of whether it throws (i.e. CallStack.Pop() in finally clause
  45. // in Engine.Call/Engine.Construct - however, that method currently breaks stack traces
  46. // in error messages.
  47. while (callStackSizeBeforeExecution < engine.CallStack.Count)
  48. {
  49. engine.CallStack.Pop();
  50. }
  51. // https://tc39.es/ecma262/#sec-runtime-semantics-catchclauseevaluation
  52. var thrownValue = b.Value;
  53. var oldEnv = engine.ExecutionContext.LexicalEnvironment;
  54. var catchEnv = JintEnvironment.NewDeclarativeEnvironment(engine, oldEnv, catchEnvironment: true);
  55. var boundNames = new List<string>();
  56. _statement.Handler.Param.GetBoundNames(boundNames);
  57. foreach (var argName in boundNames)
  58. {
  59. catchEnv.CreateMutableBinding(argName, false);
  60. }
  61. engine.UpdateLexicalEnvironment(catchEnv);
  62. var catchParam = _statement.Handler?.Param;
  63. catchParam.BindingInitialization(context, thrownValue, catchEnv);
  64. b = _catch.Execute(context);
  65. engine.UpdateLexicalEnvironment(oldEnv);
  66. }
  67. }
  68. if (_finalizer != null)
  69. {
  70. var f = _finalizer.Execute(context);
  71. if (f.Type == CompletionType.Normal)
  72. {
  73. return b;
  74. }
  75. return f.UpdateEmpty(Undefined.Instance);
  76. }
  77. return b.UpdateEmpty(Undefined.Instance);
  78. }
  79. }
  80. }