From 18d947450d3c42452cf27cd8d99985e6563c0db8 Mon Sep 17 00:00:00 2001 From: Joyee Cheung Date: Fri, 4 Nov 2022 18:33:03 +0100 Subject: [PATCH 1/2] util: use private symbols in JS land directly Instead of calling into C++ to use the private symbols, use an ObjectTemplate to create an object that holds the symbols and use them directly from JS land. PR-URL: /~https://github.com/nodejs/node/pull/45379 Backport-PR-URL: /~https://github.com/nodejs/node/pull/45674 Reviewed-By: Chengzhong Wu Reviewed-By: Antoine du Hamel Reviewed-By: James M Snell Signed-off-by: Daeyeon Jeong --- lib/internal/bootstrap/node.js | 8 +-- lib/internal/buffer.js | 8 ++- lib/internal/errors.js | 14 ++--- lib/internal/util.js | 15 +++-- src/node_util.cc | 61 ++++--------------- ...test-internal-util-decorate-error-stack.js | 15 ++--- test/parallel/test-util-internal.js | 24 +++----- 7 files changed, 51 insertions(+), 94 deletions(-) diff --git a/lib/internal/bootstrap/node.js b/lib/internal/bootstrap/node.js index 18d23253b1c94e..4e81019a3f65c6 100644 --- a/lib/internal/bootstrap/node.js +++ b/lib/internal/bootstrap/node.js @@ -75,8 +75,9 @@ const { exposeInterface, } = require('internal/util'); const { - exiting_aliased_Uint32Array, - getHiddenValue, + privateSymbols: { + exiting_aliased_Uint32Array, + }, } = internalBinding('util'); setupProcessObject(); @@ -86,8 +87,7 @@ setupBuffer(); process.domain = null; { - const exitingAliasedUint32Array = - getHiddenValue(process, exiting_aliased_Uint32Array); + const exitingAliasedUint32Array = process[exiting_aliased_Uint32Array]; ObjectDefineProperty(process, '_exiting', { __proto__: null, get() { diff --git a/lib/internal/buffer.js b/lib/internal/buffer.js index bd38cf48a7fc6e..7b8331f47514c4 100644 --- a/lib/internal/buffer.js +++ b/lib/internal/buffer.js @@ -32,9 +32,11 @@ const { utf8Write, getZeroFillToggle } = internalBinding('buffer'); + const { - untransferable_object_private_symbol, - setHiddenValue, + privateSymbols: { + untransferable_object_private_symbol, + }, } = internalBinding('util'); // Temporary buffers to convert numbers. @@ -1048,7 +1050,7 @@ function addBufferPrototypeMethods(proto) { function markAsUntransferable(obj) { if ((typeof obj !== 'object' && typeof obj !== 'function') || obj === null) return; // This object is a primitive and therefore already untransferable. - setHiddenValue(obj, untransferable_object_private_symbol, true); + obj[untransferable_object_private_symbol] = true; } // A toggle used to access the zero fill setting of the array buffer allocator diff --git a/lib/internal/errors.js b/lib/internal/errors.js index f68855c148fcfe..5e08c1b1b3a7db 100644 --- a/lib/internal/errors.js +++ b/lib/internal/errors.js @@ -809,16 +809,14 @@ const fatalExceptionStackEnhancers = { } }; +const { + privateSymbols: { + arrow_message_private_symbol, + } +} = internalBinding('util'); // Ensures the printed error line is from user code. -let _kArrowMessagePrivateSymbol, _setHiddenValue; function setArrowMessage(err, arrowMessage) { - if (!_kArrowMessagePrivateSymbol) { - ({ - arrow_message_private_symbol: _kArrowMessagePrivateSymbol, - setHiddenValue: _setHiddenValue, - } = internalBinding('util')); - } - _setHiddenValue(err, _kArrowMessagePrivateSymbol, arrowMessage); + err[arrow_message_private_symbol] = arrowMessage; } // Hide stack lines before the first user code line. diff --git a/lib/internal/util.js b/lib/internal/util.js index 6e3b60d77ba44a..f4c08ecdddbdaf 100644 --- a/lib/internal/util.js +++ b/lib/internal/util.js @@ -42,10 +42,10 @@ const { } = require('internal/errors'); const { signals } = internalBinding('constants').os; const { - getHiddenValue, - setHiddenValue, - arrow_message_private_symbol: kArrowMessagePrivateSymbolIndex, - decorated_private_symbol: kDecoratedPrivateSymbolIndex, + privateSymbols: { + arrow_message_private_symbol, + decorated_private_symbol, + }, sleep: _sleep, toUSVString: _toUSVString, } = internalBinding('util'); @@ -143,15 +143,14 @@ function deprecate(fn, msg, code, useEmitSync) { } function decorateErrorStack(err) { - if (!(isError(err) && err.stack) || - getHiddenValue(err, kDecoratedPrivateSymbolIndex) === true) + if (!(isError(err) && err.stack) || err[decorated_private_symbol]) return; - const arrow = getHiddenValue(err, kArrowMessagePrivateSymbolIndex); + const arrow = err[arrow_message_private_symbol]; if (arrow) { err.stack = arrow + err.stack; - setHiddenValue(err, kDecoratedPrivateSymbolIndex, true); + err[decorated_private_symbol] = true; } } diff --git a/src/node_util.cc b/src/node_util.cc index 58f58478e59944..eed4531fe97cdd 100644 --- a/src/node_util.cc +++ b/src/node_util.cc @@ -26,7 +26,6 @@ using v8::Object; using v8::ONLY_CONFIGURABLE; using v8::ONLY_ENUMERABLE; using v8::ONLY_WRITABLE; -using v8::Private; using v8::Promise; using v8::PropertyFilter; using v8::Proxy; @@ -159,44 +158,6 @@ static void PreviewEntries(const FunctionCallbackInfo& args) { Array::New(env->isolate(), ret, arraysize(ret))); } -inline Local IndexToPrivateSymbol(Environment* env, uint32_t index) { -#define V(name, _) &Environment::name, - static Local (Environment::*const methods[])() const = { - PER_ISOLATE_PRIVATE_SYMBOL_PROPERTIES(V) - }; -#undef V - CHECK_LT(index, arraysize(methods)); - return (env->*methods[index])(); -} - -static void GetHiddenValue(const FunctionCallbackInfo& args) { - Environment* env = Environment::GetCurrent(args); - - CHECK(args[0]->IsObject()); - CHECK(args[1]->IsUint32()); - - Local obj = args[0].As(); - uint32_t index = args[1].As()->Value(); - Local private_symbol = IndexToPrivateSymbol(env, index); - Local ret; - if (obj->GetPrivate(env->context(), private_symbol).ToLocal(&ret)) - args.GetReturnValue().Set(ret); -} - -static void SetHiddenValue(const FunctionCallbackInfo& args) { - Environment* env = Environment::GetCurrent(args); - - CHECK(args[0]->IsObject()); - CHECK(args[1]->IsUint32()); - - Local obj = args[0].As(); - uint32_t index = args[1].As()->Value(); - Local private_symbol = IndexToPrivateSymbol(env, index); - bool ret; - if (obj->SetPrivate(env->context(), private_symbol, args[2]).To(&ret)) - args.GetReturnValue().Set(ret); -} - static void Sleep(const FunctionCallbackInfo& args) { CHECK(args[0]->IsUint32()); uint32_t msec = args[0].As()->Value(); @@ -379,8 +340,6 @@ static void ToUSVString(const FunctionCallbackInfo& args) { } void RegisterExternalReferences(ExternalReferenceRegistry* registry) { - registry->Register(GetHiddenValue); - registry->Register(SetHiddenValue); registry->Register(GetPromiseDetails); registry->Register(GetProxyDetails); registry->Register(PreviewEntries); @@ -404,16 +363,22 @@ void Initialize(Local target, Environment* env = Environment::GetCurrent(context); Isolate* isolate = env->isolate(); -#define V(name, _) \ - target->Set(context, \ - FIXED_ONE_BYTE_STRING(env->isolate(), #name), \ - Integer::NewFromUnsigned(env->isolate(), index++)).Check(); { - uint32_t index = 0; + Local tmpl = v8::ObjectTemplate::New(isolate); +#define V(PropertyName, _) \ + tmpl->Set(FIXED_ONE_BYTE_STRING(env->isolate(), #PropertyName), \ + env->PropertyName()); + PER_ISOLATE_PRIVATE_SYMBOL_PROPERTIES(V) - } #undef V + target + ->Set(context, + FIXED_ONE_BYTE_STRING(isolate, "privateSymbols"), + tmpl->NewInstance(context).ToLocalChecked()) + .Check(); + } + #define V(name) \ target->Set(context, \ FIXED_ONE_BYTE_STRING(env->isolate(), #name), \ @@ -424,8 +389,6 @@ void Initialize(Local target, V(kRejected); #undef V - SetMethodNoSideEffect(context, target, "getHiddenValue", GetHiddenValue); - SetMethod(context, target, "setHiddenValue", SetHiddenValue); SetMethodNoSideEffect( context, target, "getPromiseDetails", GetPromiseDetails); SetMethodNoSideEffect(context, target, "getProxyDetails", GetProxyDetails); diff --git a/test/parallel/test-internal-util-decorate-error-stack.js b/test/parallel/test-internal-util-decorate-error-stack.js index b36714b44b8b72..3566d9375fb81c 100644 --- a/test/parallel/test-internal-util-decorate-error-stack.js +++ b/test/parallel/test-internal-util-decorate-error-stack.js @@ -5,12 +5,14 @@ const fixtures = require('../common/fixtures'); const assert = require('assert'); const internalUtil = require('internal/util'); const { internalBinding } = require('internal/test/binding'); -const binding = internalBinding('util'); +const { + privateSymbols: { + arrow_message_private_symbol, + decorated_private_symbol, + } +} = internalBinding('util'); const spawnSync = require('child_process').spawnSync; -const kArrowMessagePrivateSymbolIndex = binding.arrow_message_private_symbol; -const kDecoratedPrivateSymbolIndex = binding.decorated_private_symbol; - const decorateErrorStack = internalUtil.decorateErrorStack; // Verify that decorateErrorStack does not throw with non-objects. @@ -73,9 +75,8 @@ const arrowMessage = 'arrow_message'; err = new Error('foo'); originalStack = err.stack; -binding.setHiddenValue(err, kArrowMessagePrivateSymbolIndex, arrowMessage); +err[arrow_message_private_symbol] = arrowMessage; decorateErrorStack(err); assert.strictEqual(err.stack, `${arrowMessage}${originalStack}`); -assert.strictEqual( - binding.getHiddenValue(err, kDecoratedPrivateSymbolIndex), true); +assert.strictEqual(err[decorated_private_symbol], true); diff --git a/test/parallel/test-util-internal.js b/test/parallel/test-util-internal.js index 36d65e54d5b6ca..e2b500daa70060 100644 --- a/test/parallel/test-util-internal.js +++ b/test/parallel/test-util-internal.js @@ -7,30 +7,24 @@ const fixtures = require('../common/fixtures'); const { internalBinding } = require('internal/test/binding'); const { - getHiddenValue, - setHiddenValue, - arrow_message_private_symbol: kArrowMessagePrivateSymbolIndex + privateSymbols: { + arrow_message_private_symbol, + }, } = internalBinding('util'); -assert.strictEqual( - getHiddenValue({}, kArrowMessagePrivateSymbolIndex), - undefined); - const obj = {}; -assert.strictEqual( - setHiddenValue(obj, kArrowMessagePrivateSymbolIndex, 'bar'), - true); -assert.strictEqual( - getHiddenValue(obj, kArrowMessagePrivateSymbolIndex), - 'bar'); +assert.strictEqual(obj[arrow_message_private_symbol], undefined); + +obj[arrow_message_private_symbol] = 'bar'; +assert.strictEqual(obj[arrow_message_private_symbol], 'bar'); +assert.deepStrictEqual(Reflect.ownKeys(obj), []); let arrowMessage; try { require(fixtures.path('syntax', 'bad_syntax')); } catch (err) { - arrowMessage = - getHiddenValue(err, kArrowMessagePrivateSymbolIndex); + arrowMessage = err[arrow_message_private_symbol]; } assert.match(arrowMessage, /bad_syntax\.js:1/); From f06e8e9861089846d1d76bfa877efe9d72dd80dc Mon Sep 17 00:00:00 2001 From: Daeyeon Jeong Date: Thu, 24 Nov 2022 11:09:29 +0900 Subject: [PATCH 2/2] src,lib: group properties used as constants from `util` binding Properties used as constants in `util` internal binding are scattered. This suggests using an object holding all of them for better maintenance. Signed-off-by: Daeyeon Jeong PR-URL: /~https://github.com/nodejs/node/pull/45539 Backport-PR-URL: /~https://github.com/nodejs/node/pull/45674 Reviewed-By: Darshan Sen Reviewed-By: Chengzhong Wu Reviewed-By: Minwoo Jung Reviewed-By: Joyee Cheung --- lib/buffer.js | 6 ++-- lib/internal/util/comparisons.js | 8 +++--- lib/internal/util/inspect.js | 12 ++++---- lib/internal/webstreams/util.js | 4 ++- lib/repl.js | 8 +++--- src/node_util.cc | 48 ++++++++++++++++++++------------ 6 files changed, 50 insertions(+), 36 deletions(-) diff --git a/lib/buffer.js b/lib/buffer.js index 898bc5032e5f8d..08aa52b3883ea1 100644 --- a/lib/buffer.js +++ b/lib/buffer.js @@ -67,11 +67,11 @@ const { kStringMaxLength } = internalBinding('buffer'); const { - getOwnNonIndexProperties, - propertyFilter: { + constants: { ALL_PROPERTIES, - ONLY_ENUMERABLE + ONLY_ENUMERABLE, }, + getOwnNonIndexProperties, } = internalBinding('util'); const { customInspectSymbol, diff --git a/lib/internal/util/comparisons.js b/lib/internal/util/comparisons.js index c126bd6346dae7..e41fcf2daf92a0 100644 --- a/lib/internal/util/comparisons.js +++ b/lib/internal/util/comparisons.js @@ -46,11 +46,11 @@ const { isFloat64Array, } = types; const { - getOwnNonIndexProperties, - propertyFilter: { + constants: { ONLY_ENUMERABLE, - SKIP_SYMBOLS - } + SKIP_SYMBOLS, + }, + getOwnNonIndexProperties, } = internalBinding('util'); const kStrict = true; diff --git a/lib/internal/util/inspect.js b/lib/internal/util/inspect.js index fed5b8fd068fa3..abbb0d9c8ebc86 100644 --- a/lib/internal/util/inspect.js +++ b/lib/internal/util/inspect.js @@ -97,18 +97,18 @@ const { } = primordials; const { + constants: { + ALL_PROPERTIES, + ONLY_ENUMERABLE, + kPending, + kRejected, + }, getOwnNonIndexProperties, getPromiseDetails, getProxyDetails, - kPending, - kRejected, previewEntries, getConstructorName: internalGetConstructorName, getExternalValue, - propertyFilter: { - ALL_PROPERTIES, - ONLY_ENUMERABLE - } } = internalBinding('util'); const { diff --git a/lib/internal/webstreams/util.js b/lib/internal/webstreams/util.js index 1cf2d3a1874b10..5e0a75d2b30265 100644 --- a/lib/internal/webstreams/util.js +++ b/lib/internal/webstreams/util.js @@ -40,8 +40,10 @@ const { } = require('util'); const { + constants: { + kPending, + }, getPromiseDetails, - kPending, } = internalBinding('util'); const assert = require('internal/assert'); diff --git a/lib/repl.js b/lib/repl.js index 256d910d17e454..0780b5a54743c6 100644 --- a/lib/repl.js +++ b/lib/repl.js @@ -166,11 +166,11 @@ const { setupReverseSearch, } = require('internal/repl/utils'); const { - getOwnNonIndexProperties, - propertyFilter: { + constants: { ALL_PROPERTIES, - SKIP_SYMBOLS - } + SKIP_SYMBOLS, + }, + getOwnNonIndexProperties, } = internalBinding('util'); const { startSigintWatchdog, diff --git a/src/node_util.cc b/src/node_util.cc index eed4531fe97cdd..858d8ed42208b3 100644 --- a/src/node_util.cc +++ b/src/node_util.cc @@ -379,16 +379,38 @@ void Initialize(Local target, .Check(); } -#define V(name) \ - target->Set(context, \ - FIXED_ONE_BYTE_STRING(env->isolate(), #name), \ - Integer::New(env->isolate(), Promise::PromiseState::name)) \ - .FromJust() - V(kPending); - V(kFulfilled); - V(kRejected); + { + Local constants = Object::New(isolate); +#define V(name) \ + constants \ + ->Set(context, \ + FIXED_ONE_BYTE_STRING(isolate, #name), \ + Integer::New(isolate, Promise::PromiseState::name)) \ + .Check(); + + V(kPending); + V(kFulfilled); + V(kRejected); +#undef V + +#define V(name) \ + constants \ + ->Set(context, \ + FIXED_ONE_BYTE_STRING(isolate, #name), \ + Integer::New(isolate, PropertyFilter::name)) \ + .Check(); + + V(ALL_PROPERTIES); + V(ONLY_WRITABLE); + V(ONLY_ENUMERABLE); + V(ONLY_CONFIGURABLE); + V(SKIP_STRINGS); + V(SKIP_SYMBOLS); #undef V + target->Set(context, env->constants_string(), constants).Check(); + } + SetMethodNoSideEffect( context, target, "getPromiseDetails", GetPromiseDetails); SetMethodNoSideEffect(context, target, "getProxyDetails", GetProxyDetails); @@ -402,16 +424,6 @@ void Initialize(Local target, SetMethod( context, target, "arrayBufferViewHasBuffer", ArrayBufferViewHasBuffer); - Local constants = Object::New(env->isolate()); - NODE_DEFINE_CONSTANT(constants, ALL_PROPERTIES); - NODE_DEFINE_CONSTANT(constants, ONLY_WRITABLE); - NODE_DEFINE_CONSTANT(constants, ONLY_ENUMERABLE); - NODE_DEFINE_CONSTANT(constants, ONLY_CONFIGURABLE); - NODE_DEFINE_CONSTANT(constants, SKIP_STRINGS); - NODE_DEFINE_CONSTANT(constants, SKIP_SYMBOLS); - target->Set(context, - FIXED_ONE_BYTE_STRING(env->isolate(), "propertyFilter"), - constants).Check(); Local should_abort_on_uncaught_toggle = FIXED_ONE_BYTE_STRING(env->isolate(), "shouldAbortOnUncaughtToggle");