ExecutionContext.cs 3.1 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394
  1. using Jint.Native.Function;
  2. using Jint.Native.Generator;
  3. namespace Jint.Runtime.Environments;
  4. internal readonly struct ExecutionContext
  5. {
  6. internal ExecutionContext(
  7. IScriptOrModule? scriptOrModule,
  8. Environment lexicalEnvironment,
  9. Environment variableEnvironment,
  10. PrivateEnvironment? privateEnvironment,
  11. Realm realm,
  12. GeneratorInstance? generator = null,
  13. Function? function = null,
  14. ParserOptions? parserOptions = null)
  15. {
  16. ScriptOrModule = scriptOrModule;
  17. LexicalEnvironment = lexicalEnvironment;
  18. VariableEnvironment = variableEnvironment;
  19. PrivateEnvironment = privateEnvironment;
  20. Realm = realm;
  21. Function = function;
  22. Generator = generator;
  23. ParserOptions = parserOptions;
  24. }
  25. public readonly IScriptOrModule? ScriptOrModule;
  26. public readonly Environment LexicalEnvironment;
  27. public readonly Environment VariableEnvironment;
  28. public readonly PrivateEnvironment? PrivateEnvironment;
  29. public readonly Realm Realm;
  30. public readonly Function? Function;
  31. public readonly GeneratorInstance? Generator;
  32. public readonly ParserOptions? ParserOptions;
  33. public bool Suspended => Generator?._generatorState == GeneratorState.SuspendedYield;
  34. public ExecutionContext UpdateLexicalEnvironment(Environment lexicalEnvironment)
  35. {
  36. return new ExecutionContext(ScriptOrModule, lexicalEnvironment, VariableEnvironment, PrivateEnvironment, Realm, Generator, Function);
  37. }
  38. public ExecutionContext UpdateVariableEnvironment(Environment variableEnvironment)
  39. {
  40. return new ExecutionContext(ScriptOrModule, LexicalEnvironment, variableEnvironment, PrivateEnvironment, Realm, Generator, Function);
  41. }
  42. public ExecutionContext UpdatePrivateEnvironment(PrivateEnvironment? privateEnvironment)
  43. {
  44. return new ExecutionContext(ScriptOrModule, LexicalEnvironment, VariableEnvironment, privateEnvironment, Realm, Generator, Function);
  45. }
  46. public ExecutionContext UpdateGenerator(GeneratorInstance generator)
  47. {
  48. return new ExecutionContext(ScriptOrModule, LexicalEnvironment, VariableEnvironment, PrivateEnvironment, Realm, generator, Function);
  49. }
  50. /// <summary>
  51. /// https://tc39.es/ecma262/#sec-getthisenvironment
  52. /// </summary>
  53. internal Environment GetThisEnvironment()
  54. {
  55. // The loop will always terminate because the list of environments always
  56. // ends with the global environment which has a this binding.
  57. var lex = LexicalEnvironment;
  58. while (true)
  59. {
  60. if (lex is not null)
  61. {
  62. if (lex.HasThisBinding())
  63. {
  64. return lex;
  65. }
  66. lex = lex._outerEnv;
  67. }
  68. }
  69. }
  70. internal GeneratorKind GetGeneratorKind()
  71. {
  72. if (Generator is null)
  73. {
  74. return GeneratorKind.NonGenerator;
  75. }
  76. // TODO If generator has an [[AsyncGeneratorState]] internal slot, return async.
  77. return GeneratorKind.Sync;
  78. }
  79. }