From a6ad048b8996795a71142a07b771ba9bfe83fa55 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Vinicius=20Louren=C3=A7o?= <12551007+H4ad@users.noreply.github.com> Date: Tue, 26 Sep 2023 22:19:26 -0300 Subject: [PATCH] perf_hooks: reduce overhead of new performance_entries PR-URL: https://github.com/nodejs/node/pull/49803 Reviewed-By: Matteo Collina Reviewed-By: Benjamin Gruenbaum Reviewed-By: Yagiz Nizipli Reviewed-By: Stephen Belanger --- benchmark/perf_hooks/timerfied.js | 36 ++++++++++++++++++++++++++ lib/internal/perf/performance_entry.js | 26 ++++++++++++------- 2 files changed, 52 insertions(+), 10 deletions(-) create mode 100644 benchmark/perf_hooks/timerfied.js diff --git a/benchmark/perf_hooks/timerfied.js b/benchmark/perf_hooks/timerfied.js new file mode 100644 index 00000000000000..50be0a47fc1b5a --- /dev/null +++ b/benchmark/perf_hooks/timerfied.js @@ -0,0 +1,36 @@ +'use strict'; + +const assert = require('assert'); +const common = require('../common.js'); + +const { + PerformanceObserver, + performance, +} = require('perf_hooks'); + +function randomFn() { + return Math.random(); +} + +const bench = common.createBenchmark(main, { + n: [1e5], + observe: ['function'], +}); + +let _result; + +function main({ n, observe }) { + const obs = new PerformanceObserver(() => { + bench.end(n); + }); + obs.observe({ entryTypes: [observe], buffered: true }); + + const timerfied = performance.timerify(randomFn); + + bench.start(); + for (let i = 0; i < n; i++) + _result = timerfied(); + + // Avoid V8 deadcode (elimination) + assert.ok(_result); +} diff --git a/lib/internal/perf/performance_entry.js b/lib/internal/perf/performance_entry.js index 036bfc173bd024..aa97a652626606 100644 --- a/lib/internal/perf/performance_entry.js +++ b/lib/internal/perf/performance_entry.js @@ -2,7 +2,6 @@ const { ObjectDefineProperties, - ReflectConstruct, Symbol, } = primordials; @@ -25,14 +24,17 @@ const kEntryType = Symbol('PerformanceEntry.EntryType'); const kStartTime = Symbol('PerformanceEntry.StartTime'); const kDuration = Symbol('PerformanceEntry.Duration'); const kDetail = Symbol('NodePerformanceEntry.Detail'); +const kSkipThrow = Symbol('kSkipThrow'); function isPerformanceEntry(obj) { return obj?.[kName] !== undefined; } class PerformanceEntry { - constructor() { - throw new ERR_ILLEGAL_CONSTRUCTOR(); + constructor(skipThrowSymbol = undefined) { + if (skipThrowSymbol !== kSkipThrow) { + throw new ERR_ILLEGAL_CONSTRUCTOR(); + } } get name() { @@ -92,9 +94,11 @@ function initPerformanceEntry(entry, name, type, start, duration) { } function createPerformanceEntry(name, type, start, duration) { - return ReflectConstruct(function PerformanceEntry() { - initPerformanceEntry(this, name, type, start, duration); - }, [], PerformanceEntry); + const entry = new PerformanceEntry(kSkipThrow); + + initPerformanceEntry(entry, name, type, start, duration); + + return entry; } /** @@ -119,10 +123,12 @@ class PerformanceNodeEntry extends PerformanceEntry { } function createPerformanceNodeEntry(name, type, start, duration, detail) { - return ReflectConstruct(function PerformanceNodeEntry() { - initPerformanceEntry(this, name, type, start, duration); - this[kDetail] = detail; - }, [], PerformanceNodeEntry); + const entry = new PerformanceNodeEntry(kSkipThrow); + + initPerformanceEntry(entry, name, type, start, duration); + entry[kDetail] = detail; + + return entry; } module.exports = {