electron/shell/common/node_bindings.h

170 lines
4.7 KiB
C
Raw Normal View History

// 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.
2019-06-19 20:56:58 +00:00
#ifndef SHELL_COMMON_NODE_BINDINGS_H_
#define SHELL_COMMON_NODE_BINDINGS_H_
2013-04-13 10:39:09 +00:00
#include <type_traits>
#include "base/files/file_path.h"
2016-03-08 04:38:49 +00:00
#include "base/macros.h"
#include "base/memory/weak_ptr.h"
2017-01-24 07:45:26 +00:00
#include "base/single_thread_task_runner.h"
#include "uv.h" // NOLINT(build/include_directory)
#include "v8/include/v8.h"
2013-04-13 10:39:09 +00:00
namespace base {
class MessageLoop;
}
namespace node {
class Environment;
class MultiIsolatePlatform;
class IsolateData;
2018-04-18 01:44:10 +00:00
} // namespace node
namespace electron {
2013-04-13 10:39:09 +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:
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);
}
T* t_ = {};
};
2013-04-13 10:39:09 +00:00
class NodeBindings {
public:
enum class BrowserEnvironment { BROWSER, RENDERER, WORKER };
2017-03-08 08:33:44 +00:00
static NodeBindings* Create(BrowserEnvironment browser_env);
static void RegisterBuiltinModules();
static bool IsInitialized();
2013-04-13 10:39:09 +00:00
virtual ~NodeBindings();
// Setup V8, libuv.
2015-01-24 05:05:32 +00:00
void Initialize();
// Create the environment and load node.js.
node::Environment* CreateEnvironment(v8::Handle<v8::Context> context,
node::MultiIsolatePlatform* platform);
// Load node.js in the environment.
void LoadEnvironment(node::Environment* env);
// Prepare for message loop integration.
2015-01-24 05:05:32 +00:00
void PrepareMessageLoop();
// Do message loop integration.
virtual void RunMessageLoop();
node::IsolateData* isolate_data() const { return isolate_data_; }
// 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_; }
uv_loop_t* uv_loop() const { return uv_loop_; }
bool in_worker_loop() const { return uv_loop_ == &worker_loop_; }
protected:
2017-03-08 08:33:44 +00:00
explicit NodeBindings(BrowserEnvironment browser_env);
2013-04-13 10:39:09 +00:00
// Called to poll events in new thread.
virtual void PollEvents() = 0;
// Run the libuv loop for once.
void UvRunOnce();
// Make the main thread run libuv loop.
void WakeupMainThread();
// Interrupt the PollEvents.
void WakeupEmbedThread();
2017-03-08 08:33:44 +00:00
// Which environment we are running.
const BrowserEnvironment browser_env_;
2013-04-13 13:10:41 +00:00
// Current thread's MessageLoop.
2016-11-30 07:30:03 +00:00
scoped_refptr<base::SingleThreadTaskRunner> task_runner_;
// Current thread's libuv loop.
uv_loop_t* uv_loop_;
2013-07-22 07:25:39 +00:00
private:
// 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
// Whether the libuv loop has ended.
bool embed_closed_ = false;
2013-07-22 07:25:39 +00:00
// Loop used when constructed in WORKER mode
uv_loop_t worker_loop_;
// Dummy handle to make uv's loop not quit.
UvHandle<uv_async_t> dummy_uv_handle_;
// Thread for polling events.
uv_thread_t embed_thread_;
// Semaphore to wait for main loop in the embed thread.
uv_sem_t embed_sem_;
// Environment that to wrap the uv loop.
node::Environment* uv_env_ = nullptr;
// Isolate data used in creating the environment
node::IsolateData* isolate_data_ = nullptr;
base::WeakPtrFactory<NodeBindings> weak_factory_;
2013-04-13 10:39:09 +00:00
DISALLOW_COPY_AND_ASSIGN(NodeBindings);
};
} // namespace electron
2013-04-13 10:39:09 +00:00
2019-06-19 20:56:58 +00:00
#endif // SHELL_COMMON_NODE_BINDINGS_H_