ExecutionContext.cs 3.4 KB

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