From 13a4887ee94d61d990dd22100aa7c17c39e3200a Mon Sep 17 00:00:00 2001 From: Brian White Date: Sat, 24 Dec 2016 19:50:05 -0500 Subject: [PATCH] buffer: improve allocation performance MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit assertSize() is adjusted to be inlineable according to V8's default function size limits when determining inlineability. This results in up to 11% performance gains when allocating any kind of Buffer. Avoid avoids use of in, resulting in ~50% improvement when creating a Buffer from an array-like object. PR-URL: https://github.com/nodejs/node/pull/10443 Reviewed-By: James M Snell Reviewed-By: Colin Ihrig Reviewed-By: Luigi Pinca Reviewed-By: Сковорода Никита Андреевич Reviewed-By: Anna Henningsen --- benchmark/buffers/buffer-from.js | 11 ++++++++++- lib/buffer.js | 8 ++++---- 2 files changed, 14 insertions(+), 5 deletions(-) diff --git a/benchmark/buffers/buffer-from.js b/benchmark/buffers/buffer-from.js index c35a0f23e06958..c7889b2ec63f22 100644 --- a/benchmark/buffers/buffer-from.js +++ b/benchmark/buffers/buffer-from.js @@ -10,7 +10,8 @@ const bench = common.createBenchmark(main, { 'buffer', 'uint8array', 'string', - 'string-base64' + 'string-base64', + 'object' ], len: [10, 2048], n: [1024] @@ -25,6 +26,7 @@ function main(conf) { const str = 'a'.repeat(len); const buffer = Buffer.allocUnsafe(len); const uint8array = new Uint8Array(len); + const obj = { length: null }; // Results in a new, empty Buffer var i; @@ -80,6 +82,13 @@ function main(conf) { } bench.end(n); break; + case 'object': + bench.start(); + for (i = 0; i < n * 1024; i++) { + Buffer.from(obj); + } + bench.end(n); + break; default: assert.fail(null, null, 'Should not get here'); } diff --git a/lib/buffer.js b/lib/buffer.js index e8c083b5df5a13..875dc13ac3089b 100644 --- a/lib/buffer.js +++ b/lib/buffer.js @@ -109,6 +109,9 @@ Buffer.from = function(value, encodingOrOffset, length) { Object.setPrototypeOf(Buffer, Uint8Array); +// The 'assertSize' method will remove itself from the callstack when an error +// occurs. This is done simply to keep the internal details of the +// implementation from bleeding out to users. function assertSize(size) { let err = null; @@ -121,9 +124,6 @@ function assertSize(size) { 'than ' + binding.kMaxLength); if (err) { - // The following hides the 'assertSize' method from the - // callstack. This is done simply to hide the internal - // details of the implementation from bleeding out to users. Error.captureStackTrace(err, assertSize); throw err; } @@ -263,7 +263,7 @@ function fromObject(obj) { } if (obj) { - if ('length' in obj || isArrayBuffer(obj.buffer) || + if (obj.length !== undefined || isArrayBuffer(obj.buffer) || isSharedArrayBuffer(obj.buffer)) { if (typeof obj.length !== 'number' || obj.length !== obj.length) { return new FastBuffer();