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