2014-10-31 18:17:05 +00:00
|
|
|
// Copyright (c) 2013 GitHub, Inc.
|
2014-04-25 09:49:37 +00:00
|
|
|
// Use of this source code is governed by the MIT license that can be
|
2013-04-13 10:39:09 +00:00
|
|
|
// found in the LICENSE file.
|
|
|
|
|
2021-11-22 07:34:31 +00:00
|
|
|
#ifndef ELECTRON_SHELL_COMMON_NODE_BINDINGS_H_
|
|
|
|
#define ELECTRON_SHELL_COMMON_NODE_BINDINGS_H_
|
2013-04-13 10:39:09 +00:00
|
|
|
|
2023-08-23 13:56:16 +00:00
|
|
|
#include <memory>
|
2024-01-10 22:23:35 +00:00
|
|
|
#include <optional>
|
2022-10-20 05:49:49 +00:00
|
|
|
#include <string>
|
2020-09-14 00:53:50 +00:00
|
|
|
#include <type_traits>
|
2022-10-20 05:49:49 +00:00
|
|
|
#include <vector>
|
2020-09-14 00:53:50 +00:00
|
|
|
|
2018-08-21 18:05:45 +00:00
|
|
|
#include "base/files/file_path.h"
|
2023-08-31 00:38:07 +00:00
|
|
|
#include "base/functional/callback.h"
|
2023-05-11 20:07:39 +00:00
|
|
|
#include "base/memory/raw_ptr.h"
|
|
|
|
#include "base/memory/raw_ptr_exclusion.h"
|
2014-01-08 03:55:54 +00:00
|
|
|
#include "base/memory/weak_ptr.h"
|
2024-09-06 16:18:52 +00:00
|
|
|
#include "base/types/to_address.h"
|
2023-07-18 08:41:50 +00:00
|
|
|
#include "gin/public/context_holder.h"
|
|
|
|
#include "gin/public/gin_embedders.h"
|
|
|
|
#include "shell/common/node_includes.h"
|
2020-04-24 19:57:41 +00:00
|
|
|
#include "uv.h" // NOLINT(build/include_directory)
|
2013-12-15 06:20:28 +00:00
|
|
|
#include "v8/include/v8.h"
|
2013-04-13 10:39:09 +00:00
|
|
|
|
2013-07-22 06:58:25 +00:00
|
|
|
namespace base {
|
2021-07-02 00:51:37 +00:00
|
|
|
class SingleThreadTaskRunner;
|
2013-07-22 06:58:25 +00:00
|
|
|
}
|
|
|
|
|
2019-06-19 21:23:04 +00:00
|
|
|
namespace electron {
|
2013-04-13 10:39:09 +00:00
|
|
|
|
2020-09-14 00:53:50 +00: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:
|
2024-09-06 16:18:52 +00:00
|
|
|
UvHandle() : t_{new T} {}
|
2020-09-14 00:53:50 +00:00
|
|
|
~UvHandle() { reset(); }
|
2024-09-06 16:18:52 +00:00
|
|
|
|
2024-09-09 16:52:57 +00:00
|
|
|
explicit UvHandle(UvHandle&& that) {
|
|
|
|
t_ = that.t_;
|
|
|
|
that.t_ = nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
UvHandle& operator=(UvHandle&& that) {
|
|
|
|
reset();
|
|
|
|
t_ = that.t_;
|
|
|
|
that.t_ = nullptr;
|
|
|
|
return *this;
|
|
|
|
}
|
2024-09-06 16:18:52 +00:00
|
|
|
|
|
|
|
UvHandle(const UvHandle&) = delete;
|
|
|
|
UvHandle& operator=(const UvHandle&) = delete;
|
|
|
|
|
2020-09-14 00:53:50 +00:00
|
|
|
T* get() { return t_; }
|
2024-09-06 16:18:52 +00:00
|
|
|
T* operator->() { return t_; }
|
|
|
|
const T* get() const { return t_; }
|
|
|
|
const T* operator->() const { return t_; }
|
|
|
|
|
2020-09-14 00:53:50 +00:00
|
|
|
uv_handle_t* handle() { return reinterpret_cast<uv_handle_t*>(t_); }
|
|
|
|
|
2024-09-06 16:18:52 +00:00
|
|
|
// compare by handle pointer address
|
|
|
|
auto operator<=>(const UvHandle& that) const = default;
|
|
|
|
|
2020-09-14 00:53:50 +00:00
|
|
|
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 20:07:39 +00:00
|
|
|
RAW_PTR_EXCLUSION T* t_ = {};
|
2020-09-14 00:53:50 +00:00
|
|
|
};
|
|
|
|
|
2024-09-06 16:18:52 +00:00
|
|
|
// Helper for comparing UvHandles and raw uv pointers, e.g. as map keys
|
|
|
|
struct UvHandleCompare {
|
|
|
|
using is_transparent = void;
|
|
|
|
|
|
|
|
template <typename U, typename V>
|
|
|
|
bool operator()(U const& u, V const& v) const {
|
|
|
|
return base::to_address(u) < base::to_address(v);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2013-04-13 10:39:09 +00:00
|
|
|
class NodeBindings {
|
|
|
|
public:
|
2022-10-20 05:49:49 +00:00
|
|
|
enum class BrowserEnvironment { kBrowser, kRenderer, kUtility, kWorker };
|
2017-03-08 08:33:44 +00:00
|
|
|
|
2024-08-21 14:56:59 +00:00
|
|
|
static std::unique_ptr<NodeBindings> Create(BrowserEnvironment browser_env);
|
2023-02-09 01:31:38 +00:00
|
|
|
static void RegisterBuiltinBindings();
|
2018-06-13 07:38:31 +00:00
|
|
|
static bool IsInitialized();
|
2013-04-13 15:05:13 +00:00
|
|
|
|
2013-04-13 10:39:09 +00:00
|
|
|
virtual ~NodeBindings();
|
|
|
|
|
2013-12-15 06:20:28 +00:00
|
|
|
// Setup V8, libuv.
|
2023-02-22 09:03:46 +00:00
|
|
|
void Initialize(v8::Local<v8::Context> context);
|
2013-04-13 15:05:13 +00:00
|
|
|
|
2023-08-15 18:49:21 +00:00
|
|
|
std::vector<std::string> ParseNodeCliFlags();
|
2022-11-10 21:31:20 +00:00
|
|
|
|
2013-12-15 06:20:28 +00:00
|
|
|
// Create the environment and load node.js.
|
2023-08-23 13:56:16 +00:00
|
|
|
std::shared_ptr<node::Environment> CreateEnvironment(
|
2024-07-25 09:00:05 +00:00
|
|
|
v8::Local<v8::Context> context,
|
2023-08-23 13:56:16 +00:00
|
|
|
node::MultiIsolatePlatform* platform,
|
|
|
|
std::vector<std::string> args,
|
2023-08-31 00:38:07 +00:00
|
|
|
std::vector<std::string> exec_args,
|
2024-01-10 22:23:35 +00:00
|
|
|
std::optional<base::RepeatingCallback<void()>> on_app_code_ready =
|
|
|
|
std::nullopt);
|
2023-08-23 13:56:16 +00:00
|
|
|
|
|
|
|
std::shared_ptr<node::Environment> CreateEnvironment(
|
2024-07-25 09:00:05 +00:00
|
|
|
v8::Local<v8::Context> context,
|
2023-08-31 00:38:07 +00:00
|
|
|
node::MultiIsolatePlatform* platform,
|
2024-01-10 22:23:35 +00:00
|
|
|
std::optional<base::RepeatingCallback<void()>> on_app_code_ready =
|
|
|
|
std::nullopt);
|
2013-04-20 03:13:06 +00:00
|
|
|
|
2015-01-21 22:00:19 +00:00
|
|
|
// Load node.js in the environment.
|
|
|
|
void LoadEnvironment(node::Environment* env);
|
|
|
|
|
2022-03-30 03:09:42 +00:00
|
|
|
// Prepare embed thread for message loop integration.
|
|
|
|
void PrepareEmbedThread();
|
2013-04-13 15:05:13 +00:00
|
|
|
|
2022-03-30 03:09:42 +00:00
|
|
|
// Notify embed thread to start polling after environment is loaded.
|
|
|
|
void StartPolling();
|
2013-04-13 15:05:13 +00:00
|
|
|
|
2023-07-18 08:41:50 +00: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 07:42:22 +00:00
|
|
|
}
|
2019-11-02 22:14:11 +00:00
|
|
|
|
2014-01-09 13:35:29 +00:00
|
|
|
// Gets/sets the environment to wrap uv loop.
|
|
|
|
void set_uv_env(node::Environment* env) { uv_env_ = env; }
|
2014-01-10 08:29:38 +00:00
|
|
|
node::Environment* uv_env() const { return uv_env_; }
|
2014-01-09 13:35:29 +00:00
|
|
|
|
2023-09-07 23:25:17 +00:00
|
|
|
[[nodiscard]] constexpr uv_loop_t* uv_loop() { return uv_loop_; }
|
2020-07-08 18:00:43 +00:00
|
|
|
|
2021-11-03 11:41:45 +00:00
|
|
|
// disable copy
|
|
|
|
NodeBindings(const NodeBindings&) = delete;
|
|
|
|
NodeBindings& operator=(const NodeBindings&) = delete;
|
|
|
|
|
2023-08-31 00:38:07 +00: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 15:05:13 +00:00
|
|
|
protected:
|
2017-03-08 08:33:44 +00:00
|
|
|
explicit NodeBindings(BrowserEnvironment browser_env);
|
2013-04-13 10:39:09 +00:00
|
|
|
|
2013-07-22 06:58:25 +00:00
|
|
|
// Called to poll events in new thread.
|
|
|
|
virtual void PollEvents() = 0;
|
|
|
|
|
|
|
|
// Make the main thread run libuv loop.
|
|
|
|
void WakeupMainThread();
|
|
|
|
|
2013-07-23 05:08:40 +00:00
|
|
|
// Interrupt the PollEvents.
|
|
|
|
void WakeupEmbedThread();
|
|
|
|
|
2023-09-07 23:25:17 +00: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 08:33:44 +00:00
|
|
|
// Which environment we are running.
|
2019-05-03 18:11:41 +00:00
|
|
|
const BrowserEnvironment browser_env_;
|
2013-04-13 13:10:41 +00:00
|
|
|
|
2023-09-07 23:25:17 +00:00
|
|
|
// Loop used when constructed in WORKER mode
|
|
|
|
uv_loop_t worker_loop_;
|
2013-07-22 06:58:25 +00:00
|
|
|
|
2017-03-10 08:07:51 +00:00
|
|
|
// Current thread's libuv loop.
|
2023-09-07 23:25:17 +00: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 06:58:25 +00:00
|
|
|
|
2023-08-23 13:56:16 +00: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 07:25:39 +00:00
|
|
|
// Thread to poll uv events.
|
2018-04-18 01:44:10 +00:00
|
|
|
static void EmbedThreadRunner(void* arg);
|
2013-07-22 07:25:39 +00:00
|
|
|
|
2023-08-31 00:38:07 +00: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-30 03:09:42 +00:00
|
|
|
// Indicates whether polling thread has been created.
|
|
|
|
bool initialized_ = false;
|
|
|
|
|
2023-08-31 00:38:07 +00: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 07:25:39 +00:00
|
|
|
// Whether the libuv loop has ended.
|
2018-05-21 22:18:38 +00:00
|
|
|
bool embed_closed_ = false;
|
2013-07-22 07:25:39 +00:00
|
|
|
|
2013-07-22 06:58:25 +00:00
|
|
|
// Dummy handle to make uv's loop not quit.
|
2020-09-14 00:53:50 +00:00
|
|
|
UvHandle<uv_async_t> dummy_uv_handle_;
|
2013-07-22 06:58:25 +00: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 13:35:29 +00:00
|
|
|
// Environment that to wrap the uv loop.
|
2023-05-11 20:07:39 +00:00
|
|
|
raw_ptr<node::Environment> uv_env_ = nullptr;
|
2014-01-09 13:35:29 +00:00
|
|
|
|
2019-11-02 22:14:11 +00:00
|
|
|
// Isolate data used in creating the environment
|
2023-05-11 20:07:39 +00:00
|
|
|
raw_ptr<node::IsolateData> isolate_data_ = nullptr;
|
2019-11-02 22:14:11 +00:00
|
|
|
|
2021-01-26 18:16:21 +00:00
|
|
|
base::WeakPtrFactory<NodeBindings> weak_factory_{this};
|
2013-04-13 10:39:09 +00:00
|
|
|
};
|
|
|
|
|
2024-01-24 15:54:32 +00: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 21:23:04 +00:00
|
|
|
} // namespace electron
|
2013-04-13 10:39:09 +00:00
|
|
|
|
2021-11-22 07:34:31 +00:00
|
|
|
#endif // ELECTRON_SHELL_COMMON_NODE_BINDINGS_H_
|