JintTryStatement.cs 3.3 KB

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