From 7d01b6a9c51b29c88a4aa7a488848d8d53566dbc Mon Sep 17 00:00:00 2001 From: Chengzhong Wu Date: Fri, 20 Sep 2024 20:59:05 +0100 Subject: [PATCH] src: cleanup per env handles directly without a list Environment handles can be cleaned up directly without saving the references in a list and iterate the list. PR-URL: https://github.com/nodejs/node/pull/54993 Reviewed-By: Yagiz Nizipli Reviewed-By: Joyee Cheung --- src/env-inl.h | 6 ------ src/env.cc | 40 ++++++++++++++++------------------------ src/env.h | 19 +++---------------- 3 files changed, 19 insertions(+), 46 deletions(-) diff --git a/src/env-inl.h b/src/env-inl.h index d98a32d6ec3114..852c82fd7f50c0 100644 --- a/src/env-inl.h +++ b/src/env-inl.h @@ -260,12 +260,6 @@ inline uv_idle_t* Environment::immediate_idle_handle() { return &immediate_idle_handle_; } -inline void Environment::RegisterHandleCleanup(uv_handle_t* handle, - HandleCleanupCb cb, - void* arg) { - handle_cleanup_queue_.push_back(HandleCleanup{handle, cb, arg}); -} - template inline void Environment::CloseHandle(T* handle, OnCloseCallback callback) { handle_cleanup_waiting_++; diff --git a/src/env.cc b/src/env.cc index 38802b1e9acf9b..ca75e0360bbf6a 100644 --- a/src/env.cc +++ b/src/env.cc @@ -1109,13 +1109,8 @@ void Environment::InitializeLibuv() { } } - // Register clean-up cb to be called to clean up the handles - // when the environment is freed, note that they are not cleaned in - // the one environment per process setup, but will be called in - // FreeEnvironment. - RegisterHandleCleanups(); - StartProfilerIdleNotifier(); + env_handle_initialized_ = true; } void Environment::ExitEnv(StopFlags::Flags flags) { @@ -1136,27 +1131,27 @@ void Environment::ExitEnv(StopFlags::Flags flags) { }); } -void Environment::RegisterHandleCleanups() { - HandleCleanupCb close_and_finish = [](Environment* env, uv_handle_t* handle, - void* arg) { - handle->data = env; +void Environment::ClosePerEnvHandles() { + // If LoadEnvironment and InitializeLibuv are not called, like when building + // snapshots, skip closing the per environment handles. + if (!env_handle_initialized_) { + return; + } - env->CloseHandle(handle, [](uv_handle_t* handle) { + auto close_and_finish = [&](uv_handle_t* handle) { + CloseHandle(handle, [](uv_handle_t* handle) { #ifdef DEBUG memset(handle, 0xab, uv_handle_size(handle->type)); #endif }); }; - auto register_handle = [&](uv_handle_t* handle) { - RegisterHandleCleanup(handle, close_and_finish, nullptr); - }; - register_handle(reinterpret_cast(timer_handle())); - register_handle(reinterpret_cast(immediate_check_handle())); - register_handle(reinterpret_cast(immediate_idle_handle())); - register_handle(reinterpret_cast(&idle_prepare_handle_)); - register_handle(reinterpret_cast(&idle_check_handle_)); - register_handle(reinterpret_cast(&task_queues_async_)); + close_and_finish(reinterpret_cast(timer_handle())); + close_and_finish(reinterpret_cast(immediate_check_handle())); + close_and_finish(reinterpret_cast(immediate_idle_handle())); + close_and_finish(reinterpret_cast(&idle_prepare_handle_)); + close_and_finish(reinterpret_cast(&idle_check_handle_)); + close_and_finish(reinterpret_cast(&task_queues_async_)); } void Environment::CleanupHandles() { @@ -1176,10 +1171,6 @@ void Environment::CleanupHandles() { for (HandleWrap* handle : handle_wrap_queue_) handle->Close(); - for (HandleCleanup& hc : handle_cleanup_queue_) - hc.cb_(this, hc.handle_, hc.arg_); - handle_cleanup_queue_.clear(); - while (handle_cleanup_waiting_ != 0 || request_waiting_ != 0 || !handle_wrap_queue_.IsEmpty()) { @@ -1233,6 +1224,7 @@ MaybeLocal Environment::RunSnapshotDeserializeMain() const { void Environment::RunCleanup() { started_cleanup_ = true; TRACE_EVENT0(TRACING_CATEGORY_NODE1(environment), "RunCleanup"); + ClosePerEnvHandles(); // Only BaseObject's cleanups are registered as per-realm cleanup hooks now. // Defer the BaseObject cleanup after handles are cleaned up. CleanupHandles(); diff --git a/src/env.h b/src/env.h index 30561ab7a24c73..2ec0a56e05ff87 100644 --- a/src/env.h +++ b/src/env.h @@ -683,24 +683,10 @@ class Environment : public MemoryRetainer { inline const std::vector& argv(); const std::string& exec_path() const; - typedef void (*HandleCleanupCb)(Environment* env, - uv_handle_t* handle, - void* arg); - struct HandleCleanup { - uv_handle_t* handle_; - HandleCleanupCb cb_; - void* arg_; - }; - - void RegisterHandleCleanups(); void CleanupHandles(); void Exit(ExitCode code); void ExitEnv(StopFlags::Flags flags); - - // Register clean-up cb to be called on environment destruction. - inline void RegisterHandleCleanup(uv_handle_t* handle, - HandleCleanupCb cb, - void* arg); + void ClosePerEnvHandles(); template inline void CloseHandle(T* handle, OnCloseCallback callback); @@ -1090,6 +1076,8 @@ class Environment : public MemoryRetainer { std::list loaded_addons_; v8::Isolate* const isolate_; IsolateData* const isolate_data_; + + bool env_handle_initialized_ = false; uv_timer_t timer_handle_; uv_check_t immediate_check_handle_; uv_idle_t immediate_idle_handle_; @@ -1201,7 +1189,6 @@ class Environment : public MemoryRetainer { CleanableQueue cleanable_queue_; HandleWrapQueue handle_wrap_queue_; ReqWrapQueue req_wrap_queue_; - std::list handle_cleanup_queue_; int handle_cleanup_waiting_ = 0; int request_waiting_ = 0;