web_worker_observer.cc 3.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111
  1. // Copyright (c) 2017 GitHub, Inc.
  2. // Use of this source code is governed by the MIT license that can be
  3. // found in the LICENSE file.
  4. #include "shell/renderer/web_worker_observer.h"
  5. #include <utility>
  6. #include "base/containers/cxx20_erase_set.h"
  7. #include "base/no_destructor.h"
  8. #include "base/ranges/algorithm.h"
  9. #include "base/threading/thread_local.h"
  10. #include "shell/common/api/electron_bindings.h"
  11. #include "shell/common/gin_helper/event_emitter_caller.h"
  12. #include "shell/common/node_bindings.h"
  13. #include "shell/common/node_includes.h"
  14. namespace electron {
  15. namespace {
  16. static base::NoDestructor<base::ThreadLocalOwnedPointer<WebWorkerObserver>>
  17. lazy_tls;
  18. } // namespace
  19. // static
  20. WebWorkerObserver* WebWorkerObserver::GetCurrent() {
  21. return lazy_tls->Get();
  22. }
  23. // static
  24. WebWorkerObserver* WebWorkerObserver::Create() {
  25. auto obs = std::make_unique<WebWorkerObserver>();
  26. auto* obs_raw = obs.get();
  27. lazy_tls->Set(std::move(obs));
  28. return obs_raw;
  29. }
  30. WebWorkerObserver::WebWorkerObserver()
  31. : node_bindings_(
  32. NodeBindings::Create(NodeBindings::BrowserEnvironment::kWorker)),
  33. electron_bindings_(
  34. std::make_unique<ElectronBindings>(node_bindings_->uv_loop())) {}
  35. WebWorkerObserver::~WebWorkerObserver() = default;
  36. void WebWorkerObserver::WorkerScriptReadyForEvaluation(
  37. v8::Local<v8::Context> worker_context) {
  38. v8::Context::Scope context_scope(worker_context);
  39. auto* isolate = worker_context->GetIsolate();
  40. v8::MicrotasksScope microtasks_scope(
  41. isolate, worker_context->GetMicrotaskQueue(),
  42. v8::MicrotasksScope::kDoNotRunMicrotasks);
  43. // Start the embed thread.
  44. node_bindings_->PrepareEmbedThread();
  45. // Setup node tracing controller.
  46. if (!node::tracing::TraceEventHelper::GetAgent())
  47. node::tracing::TraceEventHelper::SetAgent(node::CreateAgent());
  48. // Setup node environment for each window.
  49. v8::Maybe<bool> initialized = node::InitializeContext(worker_context);
  50. CHECK(!initialized.IsNothing() && initialized.FromJust());
  51. std::shared_ptr<node::Environment> env =
  52. node_bindings_->CreateEnvironment(worker_context, nullptr);
  53. // Add Electron extended APIs.
  54. electron_bindings_->BindTo(env->isolate(), env->process_object());
  55. // Load everything.
  56. node_bindings_->LoadEnvironment(env.get());
  57. // Make uv loop being wrapped by window context.
  58. node_bindings_->set_uv_env(env.get());
  59. // Give the node loop a run to make sure everything is ready.
  60. node_bindings_->StartPolling();
  61. // Keep the environment alive until we free it in ContextWillDestroy()
  62. environments_.insert(std::move(env));
  63. }
  64. void WebWorkerObserver::ContextWillDestroy(v8::Local<v8::Context> context) {
  65. node::Environment* env = node::Environment::GetCurrent(context);
  66. if (env)
  67. gin_helper::EmitEvent(env->isolate(), env->process_object(), "exit");
  68. // Destroying the node environment will also run the uv loop,
  69. // Node.js expects `kExplicit` microtasks policy and will run microtasks
  70. // checkpoints after every call into JavaScript. Since we use a different
  71. // policy in the renderer - switch to `kExplicit`
  72. v8::MicrotaskQueue* microtask_queue = context->GetMicrotaskQueue();
  73. auto old_policy = microtask_queue->microtasks_policy();
  74. DCHECK_EQ(microtask_queue->GetMicrotasksScopeDepth(), 0);
  75. microtask_queue->set_microtasks_policy(v8::MicrotasksPolicy::kExplicit);
  76. base::EraseIf(environments_,
  77. [env](auto const& item) { return item.get() == env; });
  78. microtask_queue->set_microtasks_policy(old_policy);
  79. // ElectronBindings is tracking node environments.
  80. electron_bindings_->EnvironmentDestroyed(env);
  81. if (lazy_tls->Get())
  82. lazy_tls->Set(nullptr);
  83. }
  84. } // namespace electron