From 92a77aa64d30b49ea1c8677bed1174847530414e Mon Sep 17 00:00:00 2001 From: Joyee Cheung Date: Thu, 9 Feb 2023 00:34:07 +0100 Subject: [PATCH] timers: use V8 fast API calls --- lib/internal/timers.js | 21 ++-- lib/timers.js | 6 +- node.gyp | 1 + src/env.cc | 8 +- src/env.h | 2 + src/node_external_reference.h | 9 ++ src/node_snapshotable.cc | 1 + src/node_snapshotable.h | 1 + src/timers.cc | 167 +++++++++++++++++++++----- src/timers.h | 70 +++++++++++ test/parallel/test-timers-now.js | 4 +- test/parallel/test-timers-ordering.js | 4 +- 12 files changed, 245 insertions(+), 49 deletions(-) create mode 100644 src/timers.h diff --git a/lib/internal/timers.js b/lib/internal/timers.js index dac5938eabd7df..8e4daa706acd4b 100644 --- a/lib/internal/timers.js +++ b/lib/internal/timers.js @@ -81,14 +81,11 @@ const { Symbol, } = primordials; +const binding = internalBinding('timers'); const { - scheduleTimer, - toggleTimerRef, - getLibuvNow, immediateInfo, timeoutInfo, - toggleImmediateRef -} = internalBinding('timers'); +} = binding; const { getDefaultTriggerAsyncId, @@ -307,12 +304,12 @@ const immediateQueue = new ImmediateList(); function incRefCount() { if (timeoutInfo[0]++ === 0) - toggleTimerRef(true); + binding.toggleTimerRef(true); } function decRefCount() { if (--timeoutInfo[0] === 0) - toggleTimerRef(false); + binding.toggleTimerRef(false); } // Schedule or re-schedule a timer. @@ -356,7 +353,7 @@ function insertGuarded(item, refed, start) { item[kRefed] = refed; } -function insert(item, msecs, start = getLibuvNow()) { +function insert(item, msecs, start = binding.getLibuvNow()) { // Truncate so that accuracy of sub-millisecond timers is not assumed. msecs = MathTrunc(msecs); item._idleStart = start; @@ -370,7 +367,7 @@ function insert(item, msecs, start = getLibuvNow()) { timerListQueue.insert(list); if (nextExpiry > expiry) { - scheduleTimer(msecs); + binding.scheduleTimer(msecs); nextExpiry = expiry; } } @@ -560,7 +557,7 @@ function getTimerCallbacks(runNextTicks) { let start; if (timer._repeat) - start = getLibuvNow(); + start = binding.getLibuvNow(); try { const args = timer._timerArgs; @@ -628,7 +625,7 @@ class Immediate { if (this[kRefed] === false) { this[kRefed] = true; if (immediateInfo[kRefCount]++ === 0) - toggleImmediateRef(true); + binding.toggleImmediateRef(true); } return this; } @@ -637,7 +634,7 @@ class Immediate { if (this[kRefed] === true) { this[kRefed] = false; if (--immediateInfo[kRefCount] === 0) - toggleImmediateRef(false); + binding.toggleImmediateRef(false); } return this; } diff --git a/lib/timers.js b/lib/timers.js index 4e3e0dc360f859..fdb661f0ec00b9 100644 --- a/lib/timers.js +++ b/lib/timers.js @@ -27,10 +27,10 @@ const { SymbolToPrimitive } = primordials; +const binding = internalBinding('timers'); const { immediateInfo, - toggleImmediateRef -} = internalBinding('timers'); +} = binding; const L = require('internal/linkedlist'); const { async_id_symbol, @@ -324,7 +324,7 @@ function clearImmediate(immediate) { immediate._destroyed = true; if (immediate[kRefed] && --immediateInfo[kRefCount] === 0) - toggleImmediateRef(false); + binding.toggleImmediateRef(false); immediate[kRefed] = null; if (destroyHooksExist() && immediate[async_id_symbol] !== undefined) { diff --git a/node.gyp b/node.gyp index 605bc811936a6f..7a62f3d5a6deae 100644 --- a/node.gyp +++ b/node.gyp @@ -661,6 +661,7 @@ 'src/string_decoder-inl.h', 'src/string_search.h', 'src/tcp_wrap.h', + 'src/timers.h', 'src/tracing/agent.h', 'src/tracing/node_trace_buffer.h', 'src/tracing/node_trace_writer.h', diff --git a/src/env.cc b/src/env.cc index 692a344703a196..8f778f83abded4 100644 --- a/src/env.cc +++ b/src/env.cc @@ -1273,12 +1273,16 @@ void Environment::ToggleImmediateRef(bool ref) { } } - -Local Environment::GetNow() { +uint64_t Environment::GetNowUint64() { uv_update_time(event_loop()); uint64_t now = uv_now(event_loop()); CHECK_GE(now, timer_base()); now -= timer_base(); + return now; +} + +Local Environment::GetNow() { + uint64_t now = GetNowUint64(); if (now <= 0xffffffff) return Integer::NewFromUnsigned(isolate(), static_cast(now)); else diff --git a/src/env.h b/src/env.h index c2eb764e740400..c6e853b5e9df1c 100644 --- a/src/env.h +++ b/src/env.h @@ -908,6 +908,8 @@ class Environment : public MemoryRetainer { static inline Environment* ForAsyncHooks(AsyncHooks* hooks); v8::Local GetNow(); + uint64_t GetNowUint64(); + void ScheduleTimer(int64_t duration); void ToggleTimerRef(bool ref); diff --git a/src/node_external_reference.h b/src/node_external_reference.h index c3ab57c0bb0f98..b28c41a16c85fc 100644 --- a/src/node_external_reference.h +++ b/src/node_external_reference.h @@ -11,6 +11,12 @@ namespace node { using CFunctionCallback = void (*)(v8::Local receiver); +using CFunctionCallbackReturnDouble = + double (*)(v8::Local receiver); +using CFunctionCallbackWithInt64 = void (*)(v8::Local receiver, + int64_t); +using CFunctionCallbackWithBool = void (*)(v8::Local receiver, + bool); // This class manages the external references from the V8 heap // to the C++ addresses in Node.js. @@ -20,6 +26,9 @@ class ExternalReferenceRegistry { #define ALLOWED_EXTERNAL_REFERENCE_TYPES(V) \ V(CFunctionCallback) \ + V(CFunctionCallbackReturnDouble) \ + V(CFunctionCallbackWithInt64) \ + V(CFunctionCallbackWithBool) \ V(const v8::CFunctionInfo*) \ V(v8::FunctionCallback) \ V(v8::AccessorGetterCallback) \ diff --git a/src/node_snapshotable.cc b/src/node_snapshotable.cc index d4f45bf31019ea..07cecb5f793201 100644 --- a/src/node_snapshotable.cc +++ b/src/node_snapshotable.cc @@ -20,6 +20,7 @@ #include "node_util.h" #include "node_v8.h" #include "node_v8_platform-inl.h" +#include "timers.h" #if HAVE_INSPECTOR #include "inspector/worker_inspector.h" // ParentInspectorHandle diff --git a/src/node_snapshotable.h b/src/node_snapshotable.h index a825350806bfb0..538e8a2134131a 100644 --- a/src/node_snapshotable.h +++ b/src/node_snapshotable.h @@ -27,6 +27,7 @@ struct PropInfo { V(v8_binding_data, v8_utils::BindingData) \ V(blob_binding_data, BlobBindingData) \ V(process_binding_data, process::BindingData) \ + V(timers_binding_data, timers::BindingData) \ V(util_weak_reference, util::WeakReference) enum class EmbedderObjectType : uint8_t { diff --git a/src/timers.cc b/src/timers.cc index 39bb749c0724f1..2cb8a1cc72cdb0 100644 --- a/src/timers.cc +++ b/src/timers.cc @@ -1,3 +1,4 @@ +#include "timers.h" #include "env-inl.h" #include "node_external_reference.h" #include "util-inl.h" @@ -6,16 +7,17 @@ #include namespace node { -namespace { +namespace timers { using v8::Context; using v8::Function; using v8::FunctionCallbackInfo; using v8::Local; +using v8::Number; using v8::Object; using v8::Value; -void SetupTimers(const FunctionCallbackInfo& args) { +void BindingData::SetupTimers(const FunctionCallbackInfo& args) { CHECK(args[0]->IsFunction()); CHECK(args[1]->IsFunction()); auto env = Environment::GetCurrent(args); @@ -24,36 +26,128 @@ void SetupTimers(const FunctionCallbackInfo& args) { env->set_timers_callback_function(args[1].As()); } -void GetLibuvNow(const FunctionCallbackInfo& args) { - Environment* env = Environment::GetCurrent(args); - args.GetReturnValue().Set(env->GetNow()); +void BindingData::SlowGetLibuvNow(const FunctionCallbackInfo& args) { + double now = GetLibuvNowImpl(Environment::GetBindingData(args)); + args.GetReturnValue().Set(Number::New(args.GetIsolate(), now)); } -void ScheduleTimer(const FunctionCallbackInfo& args) { - auto env = Environment::GetCurrent(args); - env->ScheduleTimer(args[0]->IntegerValue(env->context()).FromJust()); +double BindingData::FastGetLibuvNow(Local receiver) { + return GetLibuvNowImpl(FromJSObject(receiver)); +} + +double BindingData::GetLibuvNowImpl(BindingData* data) { + return static_cast(data->env()->GetNowUint64()); +} + +void BindingData::SlowScheduleTimers(const FunctionCallbackInfo& args) { + int64_t duration = + args[0]->IntegerValue(args.GetIsolate()->GetCurrentContext()).FromJust(); + ScheduleTimersImpl(Environment::GetBindingData(args), duration); +} + +void BindingData::FastScheduleTimers(Local receiver, int64_t duration) { + ScheduleTimersImpl(FromJSObject(receiver), duration); +} + +void BindingData::ScheduleTimersImpl(BindingData* data, int64_t duration) { + data->env()->ScheduleTimer(duration); +} + +void BindingData::SlowToggleTimerRef( + const v8::FunctionCallbackInfo& args) { + ToggleTimerRefImpl(Environment::GetBindingData(args), + args[0]->IsTrue()); +} + +void BindingData::FastToggleTimerRef(Local receiver, bool ref) { + ToggleTimerRefImpl(FromJSObject(receiver), ref); +} + +void BindingData::ToggleTimerRefImpl(BindingData* data, bool ref) { + data->env()->ToggleTimerRef(ref); +} + +void BindingData::SlowToggleImmediateRef( + const v8::FunctionCallbackInfo& args) { + ToggleImmediateRefImpl(Environment::GetBindingData(args), + args[0]->IsTrue()); +} + +void BindingData::FastToggleImmediateRef(Local receiver, bool ref) { + ToggleImmediateRefImpl(FromJSObject(receiver), ref); +} + +void BindingData::ToggleImmediateRefImpl(BindingData* data, bool ref) { + data->env()->ToggleImmediateRef(ref); +} + +BindingData::BindingData(Environment* env, Local object) + : SnapshotableObject(env, object, type_int) {} + +bool BindingData::PrepareForSerialization(Local context, + v8::SnapshotCreator* creator) { + // Return true because we need to maintain the reference to the binding from + // JS land. + return true; } -void ToggleTimerRef(const FunctionCallbackInfo& args) { - Environment::GetCurrent(args)->ToggleTimerRef(args[0]->IsTrue()); +InternalFieldInfoBase* BindingData::Serialize(int index) { + DCHECK_EQ(index, BaseObject::kEmbedderType); + InternalFieldInfo* info = + InternalFieldInfoBase::New(type()); + return info; } -void ToggleImmediateRef(const FunctionCallbackInfo& args) { - Environment::GetCurrent(args)->ToggleImmediateRef(args[0]->IsTrue()); +void BindingData::Deserialize(Local context, + Local holder, + int index, + InternalFieldInfoBase* info) { + DCHECK_EQ(index, BaseObject::kEmbedderType); + v8::HandleScope scope(context->GetIsolate()); + Environment* env = Environment::GetCurrent(context); + // Recreate the buffer in the constructor. + BindingData* binding = env->AddBindingData(context, holder); + CHECK_NOT_NULL(binding); } -void Initialize(Local target, - Local unused, - Local context, - void* priv) { +v8::CFunction BindingData::fast_get_libuv_now_( + v8::CFunction::Make(FastGetLibuvNow)); +v8::CFunction BindingData::fast_schedule_timers_( + v8::CFunction::Make(FastScheduleTimers)); +v8::CFunction BindingData::fast_toggle_timer_ref_( + v8::CFunction::Make(FastToggleTimerRef)); +v8::CFunction BindingData::fast_toggle_immediate_ref_( + v8::CFunction::Make(FastToggleImmediateRef)); + +void BindingData::Initialize(Local target, + Local unused, + Local context, + void* priv) { Environment* env = Environment::GetCurrent(context); + BindingData* const binding_data = + env->AddBindingData(context, target); + if (binding_data == nullptr) return; - SetMethod(context, target, "getLibuvNow", GetLibuvNow); SetMethod(context, target, "setupTimers", SetupTimers); - SetMethod(context, target, "scheduleTimer", ScheduleTimer); - SetMethod(context, target, "toggleTimerRef", ToggleTimerRef); - SetMethod(context, target, "toggleImmediateRef", ToggleImmediateRef); + SetFastMethod( + context, target, "getLibuvNow", SlowGetLibuvNow, &fast_get_libuv_now_); + SetFastMethod(context, + target, + "scheduleTimer", + SlowScheduleTimers, + &fast_schedule_timers_); + SetFastMethod(context, + target, + "toggleTimerRef", + SlowToggleTimerRef, + &fast_toggle_timer_ref_); + SetFastMethod(context, + target, + "toggleImmediateRef", + SlowToggleImmediateRef, + &fast_toggle_immediate_ref_); + // TODO(joyeecheung): move these into BindingData. target ->Set(context, FIXED_ONE_BYTE_STRING(env->isolate(), "immediateInfo"), @@ -66,16 +160,33 @@ void Initialize(Local target, env->timeout_info().GetJSArray()) .Check(); } -} // anonymous namespace -void RegisterTimerExternalReferences(ExternalReferenceRegistry* registry) { - registry->Register(GetLibuvNow); + +void BindingData::RegisterTimerExternalReferences( + ExternalReferenceRegistry* registry) { registry->Register(SetupTimers); - registry->Register(ScheduleTimer); - registry->Register(ToggleTimerRef); - registry->Register(ToggleImmediateRef); + + registry->Register(SlowGetLibuvNow); + registry->Register(FastGetLibuvNow); + registry->Register(fast_get_libuv_now_.GetTypeInfo()); + + registry->Register(SlowScheduleTimers); + registry->Register(FastScheduleTimers); + registry->Register(fast_schedule_timers_.GetTypeInfo()); + + registry->Register(SlowToggleTimerRef); + registry->Register(FastToggleTimerRef); + registry->Register(fast_toggle_timer_ref_.GetTypeInfo()); + + registry->Register(SlowToggleImmediateRef); + registry->Register(FastToggleImmediateRef); + registry->Register(fast_toggle_immediate_ref_.GetTypeInfo()); } +} // namespace timers + } // namespace node -NODE_BINDING_CONTEXT_AWARE_INTERNAL(timers, node::Initialize) -NODE_BINDING_EXTERNAL_REFERENCE(timers, node::RegisterTimerExternalReferences) +NODE_BINDING_CONTEXT_AWARE_INTERNAL(timers, + node::timers::BindingData::Initialize) +NODE_BINDING_EXTERNAL_REFERENCE( + timers, node::timers::BindingData::RegisterTimerExternalReferences) diff --git a/src/timers.h b/src/timers.h new file mode 100644 index 00000000000000..148ab53ea60e93 --- /dev/null +++ b/src/timers.h @@ -0,0 +1,70 @@ +#ifndef SRC_NODE_TIMERS_H_ +#define SRC_NODE_TIMERS_H_ + +#if defined(NODE_WANT_INTERNALS) && NODE_WANT_INTERNALS + +#include +#include "node_snapshotable.h" + +namespace node { +class ExternalReferenceRegistry; + +namespace timers { +class BindingData : public SnapshotableObject { + public: + BindingData(Environment* env, v8::Local obj); + + using InternalFieldInfo = InternalFieldInfoBase; + + SERIALIZABLE_OBJECT_METHODS() + static constexpr FastStringKey type_name{"node::timers::BindingData"}; + static constexpr EmbedderObjectType type_int = + EmbedderObjectType::k_timers_binding_data; + + SET_NO_MEMORY_INFO() + SET_SELF_SIZE(BindingData) + SET_MEMORY_INFO_NAME(BindingData) + + static void SetupTimers(const v8::FunctionCallbackInfo& args); + + static void SlowGetLibuvNow(const v8::FunctionCallbackInfo& args); + static double FastGetLibuvNow(v8::Local receiver); + static double GetLibuvNowImpl(BindingData* data); + + static void SlowScheduleTimers( + const v8::FunctionCallbackInfo& args); + static void FastScheduleTimers(v8::Local receiver, + int64_t duration); + static void ScheduleTimersImpl(BindingData* data, int64_t duration); + + static void SlowToggleTimerRef( + const v8::FunctionCallbackInfo& args); + static void FastToggleTimerRef(v8::Local receiver, bool ref); + static void ToggleTimerRefImpl(BindingData* data, bool ref); + + static void SlowToggleImmediateRef( + const v8::FunctionCallbackInfo& args); + static void FastToggleImmediateRef(v8::Local receiver, bool ref); + static void ToggleImmediateRefImpl(BindingData* data, bool ref); + + static void Initialize(v8::Local target, + v8::Local unused, + v8::Local context, + void* priv); + static void RegisterTimerExternalReferences( + ExternalReferenceRegistry* registry); + + private: + static v8::CFunction fast_get_libuv_now_; + static v8::CFunction fast_schedule_timers_; + static v8::CFunction fast_toggle_timer_ref_; + static v8::CFunction fast_toggle_immediate_ref_; +}; + +} // namespace timers + +} // namespace node + +#endif // defined(NODE_WANT_INTERNALS) && NODE_WANT_INTERNALS + +#endif // SRC_NODE_TIMERS_H_ diff --git a/test/parallel/test-timers-now.js b/test/parallel/test-timers-now.js index 91b3f63496da2b..3988fd6d6dba45 100644 --- a/test/parallel/test-timers-now.js +++ b/test/parallel/test-timers-now.js @@ -4,7 +4,7 @@ require('../common'); const assert = require('assert'); const { internalBinding } = require('internal/test/binding'); -const { getLibuvNow } = internalBinding('timers'); +const binding = internalBinding('timers'); // Return value of getLibuvNow() should easily fit in a SMI after start-up. -assert(getLibuvNow() < 0x3ffffff); +assert(binding.getLibuvNow() < 0x3ffffff); diff --git a/test/parallel/test-timers-ordering.js b/test/parallel/test-timers-ordering.js index 6c6daecef3a563..89779b3ee17248 100644 --- a/test/parallel/test-timers-ordering.js +++ b/test/parallel/test-timers-ordering.js @@ -25,7 +25,7 @@ require('../common'); const assert = require('assert'); const { internalBinding } = require('internal/test/binding'); -const { getLibuvNow } = internalBinding('timers'); +const binding = internalBinding('timers'); const N = 30; @@ -39,7 +39,7 @@ function f(i) { last_i = i; // Check that this iteration is fired at least 1ms later than the previous - const now = getLibuvNow(); + const now = binding.getLibuvNow(); assert(now >= last_ts + 1, `current ts ${now} < prev ts ${last_ts} + 1`); last_ts = now;