2014-10-31 12:17:05 -06:00
|
|
|
// Copyright (c) 2013 GitHub, Inc.
|
2014-04-25 03:49:37 -06:00
|
|
|
// Use of this source code is governed by the MIT license that can be
|
2013-04-13 04:39:09 -06:00
|
|
|
// found in the LICENSE file.
|
|
|
|
|
2021-11-22 00:34:31 -07:00
|
|
|
#ifndef ELECTRON_SHELL_COMMON_NODE_BINDINGS_H_
|
|
|
|
#define ELECTRON_SHELL_COMMON_NODE_BINDINGS_H_
|
2013-04-13 04:39:09 -06:00
|
|
|
|
2023-08-23 07:56:16 -06:00
|
|
|
#include <memory>
|
2024-01-10 15:23:35 -07:00
|
|
|
#include <optional>
|
2022-10-19 23:49:49 -06:00
|
|
|
#include <string>
|
2020-09-13 18:53:50 -06:00
|
|
|
#include <type_traits>
|
2022-10-19 23:49:49 -06:00
|
|
|
#include <vector>
|
2020-09-13 18:53:50 -06:00
|
|
|
|
2018-08-21 12:05:45 -06:00
|
|
|
#include "base/files/file_path.h"
|
2023-08-30 18:38:07 -06:00
|
|
|
#include "base/functional/callback.h"
|
2023-05-11 14:07:39 -06:00
|
|
|
#include "base/memory/raw_ptr.h"
|
|
|
|
#include "base/memory/raw_ptr_exclusion.h"
|
2014-01-07 20:55:54 -07:00
|
|
|
#include "base/memory/weak_ptr.h"
|
2023-07-18 02:41:50 -06:00
|
|
|
#include "gin/public/context_holder.h"
|
|
|
|
#include "gin/public/gin_embedders.h"
|
|
|
|
#include "shell/common/node_includes.h"
|
2020-04-24 13:57:41 -06:00
|
|
|
#include "uv.h" // NOLINT(build/include_directory)
|
2013-12-14 23:20:28 -07:00
|
|
|
#include "v8/include/v8.h"
|
2013-04-13 04:39:09 -06:00
|
|
|
|
2013-07-22 00:58:25 -06:00
|
|
|
namespace base {
|
2021-07-01 18:51:37 -06:00
|
|
|
class SingleThreadTaskRunner;
|
2013-07-22 00:58:25 -06:00
|
|
|
}
|
|
|
|
|
2019-06-19 15:23:04 -06:00
|
|
|
namespace electron {
|
2013-04-13 04:39:09 -06:00
|
|
|
|
2020-09-13 18:53:50 -06:00
|
|
|
// A helper class to manage uv_handle_t types, e.g. uv_async_t.
|
|
|
|
//
|
|
|
|
// As per the uv docs: "uv_close() MUST be called on each handle before
|
|
|
|
// memory is released. Moreover, the memory can only be released in
|
|
|
|
// close_cb or after it has returned." This class encapsulates the work
|
|
|
|
// needed to follow those requirements.
|
|
|
|
template <typename T,
|
|
|
|
typename std::enable_if<
|
|
|
|
// these are the C-style 'subclasses' of uv_handle_t
|
|
|
|
std::is_same<T, uv_async_t>::value ||
|
|
|
|
std::is_same<T, uv_check_t>::value ||
|
|
|
|
std::is_same<T, uv_fs_event_t>::value ||
|
|
|
|
std::is_same<T, uv_fs_poll_t>::value ||
|
|
|
|
std::is_same<T, uv_idle_t>::value ||
|
|
|
|
std::is_same<T, uv_pipe_t>::value ||
|
|
|
|
std::is_same<T, uv_poll_t>::value ||
|
|
|
|
std::is_same<T, uv_prepare_t>::value ||
|
|
|
|
std::is_same<T, uv_process_t>::value ||
|
|
|
|
std::is_same<T, uv_signal_t>::value ||
|
|
|
|
std::is_same<T, uv_stream_t>::value ||
|
|
|
|
std::is_same<T, uv_tcp_t>::value ||
|
|
|
|
std::is_same<T, uv_timer_t>::value ||
|
|
|
|
std::is_same<T, uv_tty_t>::value ||
|
|
|
|
std::is_same<T, uv_udp_t>::value>::type* = nullptr>
|
|
|
|
class UvHandle {
|
|
|
|
public:
|
|
|
|
UvHandle() : t_(new T) {}
|
|
|
|
~UvHandle() { reset(); }
|
|
|
|
T* get() { return t_; }
|
|
|
|
uv_handle_t* handle() { return reinterpret_cast<uv_handle_t*>(t_); }
|
|
|
|
|
|
|
|
void reset() {
|
|
|
|
auto* h = handle();
|
|
|
|
if (h != nullptr) {
|
|
|
|
DCHECK_EQ(0, uv_is_closing(h));
|
|
|
|
uv_close(h, OnClosed);
|
|
|
|
t_ = nullptr;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
static void OnClosed(uv_handle_t* handle) {
|
|
|
|
delete reinterpret_cast<T*>(handle);
|
|
|
|
}
|
|
|
|
|
2023-05-11 14:07:39 -06:00
|
|
|
RAW_PTR_EXCLUSION T* t_ = {};
|
2020-09-13 18:53:50 -06:00
|
|
|
};
|
|
|
|
|
2013-04-13 04:39:09 -06:00
|
|
|
class NodeBindings {
|
|
|
|
public:
|
2022-10-19 23:49:49 -06:00
|
|
|
enum class BrowserEnvironment { kBrowser, kRenderer, kUtility, kWorker };
|
2017-03-08 01:33:44 -07:00
|
|
|
|
|
|
|
static NodeBindings* Create(BrowserEnvironment browser_env);
|
2023-02-08 18:31:38 -07:00
|
|
|
static void RegisterBuiltinBindings();
|
2018-06-13 01:38:31 -06:00
|
|
|
static bool IsInitialized();
|
2013-04-13 09:05:13 -06:00
|
|
|
|
2013-04-13 04:39:09 -06:00
|
|
|
virtual ~NodeBindings();
|
|
|
|
|
2013-12-14 23:20:28 -07:00
|
|
|
// Setup V8, libuv.
|
2023-02-22 02:03:46 -07:00
|
|
|
void Initialize(v8::Local<v8::Context> context);
|
2013-04-13 09:05:13 -06:00
|
|
|
|
2023-08-15 12:49:21 -06:00
|
|
|
std::vector<std::string> ParseNodeCliFlags();
|
2022-11-10 14:31:20 -07:00
|
|
|
|
2013-12-14 23:20:28 -07:00
|
|
|
// Create the environment and load node.js.
|
2023-08-23 07:56:16 -06:00
|
|
|
std::shared_ptr<node::Environment> CreateEnvironment(
|
|
|
|
v8::Handle<v8::Context> context,
|
|
|
|
node::MultiIsolatePlatform* platform,
|
|
|
|
std::vector<std::string> args,
|
2023-08-30 18:38:07 -06:00
|
|
|
std::vector<std::string> exec_args,
|
2024-01-10 15:23:35 -07:00
|
|
|
std::optional<base::RepeatingCallback<void()>> on_app_code_ready =
|
|
|
|
std::nullopt);
|
2023-08-23 07:56:16 -06:00
|
|
|
|
|
|
|
std::shared_ptr<node::Environment> CreateEnvironment(
|
|
|
|
v8::Handle<v8::Context> context,
|
2023-08-30 18:38:07 -06:00
|
|
|
node::MultiIsolatePlatform* platform,
|
2024-01-10 15:23:35 -07:00
|
|
|
std::optional<base::RepeatingCallback<void()>> on_app_code_ready =
|
|
|
|
std::nullopt);
|
2013-04-19 21:13:06 -06:00
|
|
|
|
2015-01-21 15:00:19 -07:00
|
|
|
// Load node.js in the environment.
|
|
|
|
void LoadEnvironment(node::Environment* env);
|
|
|
|
|
2022-03-29 21:09:42 -06:00
|
|
|
// Prepare embed thread for message loop integration.
|
|
|
|
void PrepareEmbedThread();
|
2013-04-13 09:05:13 -06:00
|
|
|
|
2022-03-29 21:09:42 -06:00
|
|
|
// Notify embed thread to start polling after environment is loaded.
|
|
|
|
void StartPolling();
|
2013-04-13 09:05:13 -06:00
|
|
|
|
2023-07-18 02:41:50 -06:00
|
|
|
node::IsolateData* isolate_data(v8::Local<v8::Context> context) const {
|
|
|
|
if (context->GetNumberOfEmbedderDataFields() <=
|
|
|
|
kElectronContextEmbedderDataIndex) {
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
auto* isolate_data = static_cast<node::IsolateData*>(
|
|
|
|
context->GetAlignedPointerFromEmbedderData(
|
|
|
|
kElectronContextEmbedderDataIndex));
|
|
|
|
CHECK(isolate_data);
|
|
|
|
CHECK(isolate_data->event_loop());
|
|
|
|
return isolate_data;
|
2022-03-21 01:42:22 -06:00
|
|
|
}
|
2019-11-02 16:14:11 -06:00
|
|
|
|
2014-01-09 06:35:29 -07:00
|
|
|
// Gets/sets the environment to wrap uv loop.
|
|
|
|
void set_uv_env(node::Environment* env) { uv_env_ = env; }
|
2014-01-10 01:29:38 -07:00
|
|
|
node::Environment* uv_env() const { return uv_env_; }
|
2014-01-09 06:35:29 -07:00
|
|
|
|
2023-09-07 17:25:17 -06:00
|
|
|
[[nodiscard]] constexpr uv_loop_t* uv_loop() { return uv_loop_; }
|
2020-07-08 12:00:43 -06:00
|
|
|
|
2021-11-03 05:41:45 -06:00
|
|
|
// disable copy
|
|
|
|
NodeBindings(const NodeBindings&) = delete;
|
|
|
|
NodeBindings& operator=(const NodeBindings&) = delete;
|
|
|
|
|
2023-08-30 18:38:07 -06:00
|
|
|
// Blocks until app code is signaled to be loaded via |SetAppCodeLoaded|.
|
|
|
|
// Only has an effect if called in the browser process
|
|
|
|
void JoinAppCode();
|
|
|
|
|
2013-04-13 09:05:13 -06:00
|
|
|
protected:
|
2017-03-08 01:33:44 -07:00
|
|
|
explicit NodeBindings(BrowserEnvironment browser_env);
|
2013-04-13 04:39:09 -06:00
|
|
|
|
2013-07-22 00:58:25 -06:00
|
|
|
// Called to poll events in new thread.
|
|
|
|
virtual void PollEvents() = 0;
|
|
|
|
|
|
|
|
// Make the main thread run libuv loop.
|
|
|
|
void WakeupMainThread();
|
|
|
|
|
2013-07-22 23:08:40 -06:00
|
|
|
// Interrupt the PollEvents.
|
|
|
|
void WakeupEmbedThread();
|
|
|
|
|
2023-09-07 17:25:17 -06:00
|
|
|
private:
|
|
|
|
static uv_loop_t* InitEventLoop(BrowserEnvironment browser_env,
|
|
|
|
uv_loop_t* worker_loop);
|
|
|
|
|
|
|
|
// Run the libuv loop for once.
|
|
|
|
void UvRunOnce();
|
|
|
|
|
|
|
|
[[nodiscard]] constexpr bool in_worker_loop() const {
|
|
|
|
return browser_env_ == BrowserEnvironment::kWorker;
|
|
|
|
}
|
|
|
|
|
2017-03-08 01:33:44 -07:00
|
|
|
// Which environment we are running.
|
2019-05-03 12:11:41 -06:00
|
|
|
const BrowserEnvironment browser_env_;
|
2013-04-13 07:10:41 -06:00
|
|
|
|
2023-09-07 17:25:17 -06:00
|
|
|
// Loop used when constructed in WORKER mode
|
|
|
|
uv_loop_t worker_loop_;
|
2013-07-22 00:58:25 -06:00
|
|
|
|
2017-03-10 01:07:51 -07:00
|
|
|
// Current thread's libuv loop.
|
2023-09-07 17:25:17 -06:00
|
|
|
// depends-on: worker_loop_
|
|
|
|
const raw_ptr<uv_loop_t> uv_loop_;
|
|
|
|
|
|
|
|
// Current thread's MessageLoop.
|
|
|
|
scoped_refptr<base::SingleThreadTaskRunner> task_runner_;
|
2013-07-22 00:58:25 -06:00
|
|
|
|
2023-08-23 07:56:16 -06:00
|
|
|
// Choose a reasonable unique index that's higher than any Blink uses
|
|
|
|
// and thus unlikely to collide with an existing index.
|
|
|
|
static constexpr int kElectronContextEmbedderDataIndex =
|
|
|
|
static_cast<int>(gin::kPerContextDataStartIndex) +
|
|
|
|
static_cast<int>(gin::kEmbedderElectron);
|
|
|
|
|
2013-07-22 01:25:39 -06:00
|
|
|
// Thread to poll uv events.
|
2018-04-17 19:44:10 -06:00
|
|
|
static void EmbedThreadRunner(void* arg);
|
2013-07-22 01:25:39 -06:00
|
|
|
|
2023-08-30 18:38:07 -06:00
|
|
|
// Default callback to indicate when the node environment has finished
|
|
|
|
// initializing and the primary import chain is fully resolved and executed
|
|
|
|
void SetAppCodeLoaded();
|
|
|
|
|
2022-03-29 21:09:42 -06:00
|
|
|
// Indicates whether polling thread has been created.
|
|
|
|
bool initialized_ = false;
|
|
|
|
|
2023-08-30 18:38:07 -06:00
|
|
|
// Indicates whether the app code has finished loading
|
|
|
|
// for ESM this is async after the module is loaded
|
|
|
|
bool app_code_loaded_ = false;
|
|
|
|
|
2013-07-22 01:25:39 -06:00
|
|
|
// Whether the libuv loop has ended.
|
2018-05-21 16:18:38 -06:00
|
|
|
bool embed_closed_ = false;
|
2013-07-22 01:25:39 -06:00
|
|
|
|
2013-07-22 00:58:25 -06:00
|
|
|
// Dummy handle to make uv's loop not quit.
|
2020-09-13 18:53:50 -06:00
|
|
|
UvHandle<uv_async_t> dummy_uv_handle_;
|
2013-07-22 00:58:25 -06:00
|
|
|
|
|
|
|
// Thread for polling events.
|
|
|
|
uv_thread_t embed_thread_;
|
|
|
|
|
|
|
|
// Semaphore to wait for main loop in the embed thread.
|
|
|
|
uv_sem_t embed_sem_;
|
|
|
|
|
2014-01-09 06:35:29 -07:00
|
|
|
// Environment that to wrap the uv loop.
|
2023-05-11 14:07:39 -06:00
|
|
|
raw_ptr<node::Environment> uv_env_ = nullptr;
|
2014-01-09 06:35:29 -07:00
|
|
|
|
2019-11-02 16:14:11 -06:00
|
|
|
// Isolate data used in creating the environment
|
2023-05-11 14:07:39 -06:00
|
|
|
raw_ptr<node::IsolateData> isolate_data_ = nullptr;
|
2019-11-02 16:14:11 -06:00
|
|
|
|
2021-01-26 11:16:21 -07:00
|
|
|
base::WeakPtrFactory<NodeBindings> weak_factory_{this};
|
2013-04-13 04:39:09 -06:00
|
|
|
};
|
|
|
|
|
2024-01-24 08:54:32 -07:00
|
|
|
// A thread-safe function responsible for loading preload script which runs for
|
|
|
|
// all node environments (including child processes and workers).
|
|
|
|
void OnNodePreload(node::Environment* env,
|
|
|
|
v8::Local<v8::Value> process,
|
|
|
|
v8::Local<v8::Value> require);
|
|
|
|
|
2019-06-19 15:23:04 -06:00
|
|
|
} // namespace electron
|
2013-04-13 04:39:09 -06:00
|
|
|
|
2021-11-22 00:34:31 -07:00
|
|
|
#endif // ELECTRON_SHELL_COMMON_NODE_BINDINGS_H_
|