2014-12-09 21:57:48 +00:00
|
|
|
// Copyright (c) 2014, StrongLoop Inc.
|
|
|
|
//
|
|
|
|
// Permission to use, copy, modify, and/or distribute this software for any
|
|
|
|
// purpose with or without fee is hereby granted, provided that the above
|
|
|
|
// copyright notice and this permission notice appear in all copies.
|
|
|
|
//
|
|
|
|
// THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
|
|
|
|
// WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
|
|
|
|
// MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
|
|
|
|
// ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
|
|
|
|
// WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
|
|
|
|
// ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
|
|
|
|
// OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
|
|
|
|
|
|
|
|
'use strict';
|
|
|
|
|
2019-04-09 07:55:53 +00:00
|
|
|
const { ObjectPrototype } = primordials;
|
|
|
|
|
2017-04-26 23:48:05 +00:00
|
|
|
const { Buffer } = require('buffer');
|
2018-12-11 15:24:22 +00:00
|
|
|
const { validateString } = require('internal/validators');
|
2017-06-08 08:17:02 +00:00
|
|
|
const {
|
|
|
|
Serializer: _Serializer,
|
|
|
|
Deserializer: _Deserializer
|
2018-08-06 20:59:11 +00:00
|
|
|
} = internalBinding('serdes');
|
2019-03-07 16:51:36 +00:00
|
|
|
const assert = require('internal/assert');
|
2018-08-17 08:33:45 +00:00
|
|
|
const { copy } = internalBinding('buffer');
|
2019-10-29 20:24:23 +00:00
|
|
|
const { inspect } = require('internal/util/inspect');
|
2017-01-27 05:38:11 +00:00
|
|
|
const { FastBuffer } = require('internal/buffer');
|
2019-05-12 12:30:29 +00:00
|
|
|
const { getValidatedPath } = require('internal/fs/utils');
|
2019-03-07 16:51:36 +00:00
|
|
|
const { toNamespacedPath } = require('path');
|
|
|
|
const {
|
|
|
|
createHeapSnapshotStream,
|
|
|
|
triggerHeapSnapshot
|
|
|
|
} = internalBinding('heap_utils');
|
|
|
|
const { Readable } = require('stream');
|
|
|
|
const { owner_symbol } = require('internal/async_hooks').symbols;
|
|
|
|
const {
|
|
|
|
kUpdateTimer,
|
|
|
|
onStreamRead,
|
|
|
|
} = require('internal/stream_base_commons');
|
|
|
|
const kHandle = Symbol('kHandle');
|
|
|
|
|
|
|
|
|
|
|
|
function writeHeapSnapshot(filename) {
|
|
|
|
if (filename !== undefined) {
|
2019-05-12 12:30:29 +00:00
|
|
|
filename = getValidatedPath(filename);
|
2019-03-07 16:51:36 +00:00
|
|
|
filename = toNamespacedPath(filename);
|
|
|
|
}
|
|
|
|
return triggerHeapSnapshot(filename);
|
|
|
|
}
|
|
|
|
|
|
|
|
class HeapSnapshotStream extends Readable {
|
|
|
|
constructor(handle) {
|
|
|
|
super({ autoDestroy: true });
|
|
|
|
this[kHandle] = handle;
|
|
|
|
handle[owner_symbol] = this;
|
|
|
|
handle.onread = onStreamRead;
|
|
|
|
}
|
|
|
|
|
|
|
|
_read() {
|
|
|
|
if (this[kHandle])
|
|
|
|
this[kHandle].readStart();
|
|
|
|
}
|
|
|
|
|
|
|
|
_destroy() {
|
|
|
|
// Release the references on the handle so that
|
|
|
|
// it can be garbage collected.
|
|
|
|
this[kHandle][owner_symbol] = undefined;
|
|
|
|
this[kHandle] = undefined;
|
|
|
|
}
|
|
|
|
|
|
|
|
[kUpdateTimer]() {
|
|
|
|
// Does nothing
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function getHeapSnapshot() {
|
|
|
|
const handle = createHeapSnapshotStream();
|
|
|
|
assert(handle);
|
|
|
|
return new HeapSnapshotStream(handle);
|
|
|
|
}
|
2015-01-21 16:36:59 +00:00
|
|
|
|
2017-06-08 08:17:02 +00:00
|
|
|
// Calling exposed c++ functions directly throws exception as it expected to be
|
|
|
|
// called with new operator and caused an assert to fire.
|
|
|
|
// Creating JS wrapper so that it gets caught at JS layer.
|
|
|
|
class Serializer extends _Serializer { }
|
|
|
|
|
|
|
|
class Deserializer extends _Deserializer { }
|
|
|
|
|
2017-04-26 23:48:05 +00:00
|
|
|
const {
|
|
|
|
cachedDataVersionTag,
|
2017-10-26 23:31:23 +00:00
|
|
|
setFlagsFromString: _setFlagsFromString,
|
2017-04-26 23:48:05 +00:00
|
|
|
heapStatisticsArrayBuffer,
|
|
|
|
heapSpaceStatisticsArrayBuffer,
|
2019-05-30 13:41:03 +00:00
|
|
|
heapCodeStatisticsArrayBuffer,
|
2017-04-26 23:48:05 +00:00
|
|
|
updateHeapStatisticsArrayBuffer,
|
|
|
|
updateHeapSpaceStatisticsArrayBuffer,
|
2019-05-30 13:41:03 +00:00
|
|
|
updateHeapCodeStatisticsArrayBuffer,
|
2017-04-26 23:48:05 +00:00
|
|
|
|
2019-05-30 13:41:03 +00:00
|
|
|
// Properties for heap statistics buffer extraction.
|
2017-04-26 23:48:05 +00:00
|
|
|
kTotalHeapSizeIndex,
|
|
|
|
kTotalHeapSizeExecutableIndex,
|
|
|
|
kTotalPhysicalSizeIndex,
|
|
|
|
kTotalAvailableSize,
|
|
|
|
kUsedHeapSizeIndex,
|
|
|
|
kHeapSizeLimitIndex,
|
|
|
|
kDoesZapGarbageIndex,
|
|
|
|
kMallocedMemoryIndex,
|
|
|
|
kPeakMallocedMemoryIndex,
|
2019-05-30 13:41:03 +00:00
|
|
|
kNumberOfNativeContextsIndex,
|
|
|
|
kNumberOfDetachedContextsIndex,
|
|
|
|
|
|
|
|
// Properties for heap spaces statistics buffer extraction.
|
2017-04-26 23:48:05 +00:00
|
|
|
kHeapSpaces,
|
|
|
|
kHeapSpaceStatisticsPropertiesCount,
|
|
|
|
kSpaceSizeIndex,
|
|
|
|
kSpaceUsedSizeIndex,
|
|
|
|
kSpaceAvailableSizeIndex,
|
2019-05-28 10:41:12 +00:00
|
|
|
kPhysicalSpaceSizeIndex,
|
2019-05-30 13:41:03 +00:00
|
|
|
|
|
|
|
// Properties for heap code statistics buffer extraction.
|
|
|
|
kCodeAndMetadataSizeIndex,
|
|
|
|
kBytecodeAndMetadataSizeIndex,
|
|
|
|
kExternalScriptSourceSizeIndex
|
2018-08-12 22:27:01 +00:00
|
|
|
} = internalBinding('v8');
|
2017-04-26 23:48:05 +00:00
|
|
|
|
|
|
|
const kNumberOfHeapSpaces = kHeapSpaces.length;
|
|
|
|
|
2015-01-21 16:36:59 +00:00
|
|
|
const heapStatisticsBuffer =
|
2017-04-26 23:48:05 +00:00
|
|
|
new Float64Array(heapStatisticsArrayBuffer);
|
|
|
|
|
2015-12-29 10:54:35 +00:00
|
|
|
const heapSpaceStatisticsBuffer =
|
2017-04-26 23:48:05 +00:00
|
|
|
new Float64Array(heapSpaceStatisticsArrayBuffer);
|
|
|
|
|
2019-05-30 13:41:03 +00:00
|
|
|
const heapCodeStatisticsBuffer =
|
|
|
|
new Float64Array(heapCodeStatisticsArrayBuffer);
|
|
|
|
|
2017-10-26 23:31:23 +00:00
|
|
|
function setFlagsFromString(flags) {
|
2018-12-11 15:24:22 +00:00
|
|
|
validateString(flags, 'flags');
|
2017-10-26 23:31:23 +00:00
|
|
|
_setFlagsFromString(flags);
|
|
|
|
}
|
|
|
|
|
2017-04-26 23:48:05 +00:00
|
|
|
function getHeapStatistics() {
|
2015-06-19 11:23:56 +00:00
|
|
|
const buffer = heapStatisticsBuffer;
|
2015-01-16 16:15:17 +00:00
|
|
|
|
2017-04-26 23:48:05 +00:00
|
|
|
updateHeapStatisticsArrayBuffer();
|
2015-01-16 16:15:17 +00:00
|
|
|
|
|
|
|
return {
|
|
|
|
'total_heap_size': buffer[kTotalHeapSizeIndex],
|
|
|
|
'total_heap_size_executable': buffer[kTotalHeapSizeExecutableIndex],
|
|
|
|
'total_physical_size': buffer[kTotalPhysicalSizeIndex],
|
2015-08-11 18:06:41 +00:00
|
|
|
'total_available_size': buffer[kTotalAvailableSize],
|
2015-01-16 16:15:17 +00:00
|
|
|
'used_heap_size': buffer[kUsedHeapSizeIndex],
|
2016-09-17 09:52:26 +00:00
|
|
|
'heap_size_limit': buffer[kHeapSizeLimitIndex],
|
|
|
|
'malloced_memory': buffer[kMallocedMemoryIndex],
|
|
|
|
'peak_malloced_memory': buffer[kPeakMallocedMemoryIndex],
|
2019-05-28 10:41:12 +00:00
|
|
|
'does_zap_garbage': buffer[kDoesZapGarbageIndex],
|
|
|
|
'number_of_native_contexts': buffer[kNumberOfNativeContextsIndex],
|
|
|
|
'number_of_detached_contexts': buffer[kNumberOfDetachedContextsIndex]
|
2015-01-16 16:15:17 +00:00
|
|
|
};
|
2017-04-26 23:48:05 +00:00
|
|
|
}
|
2015-12-29 10:54:35 +00:00
|
|
|
|
2017-04-26 23:48:05 +00:00
|
|
|
function getHeapSpaceStatistics() {
|
2015-12-29 10:54:35 +00:00
|
|
|
const heapSpaceStatistics = new Array(kNumberOfHeapSpaces);
|
|
|
|
const buffer = heapSpaceStatisticsBuffer;
|
2017-04-26 23:48:05 +00:00
|
|
|
updateHeapSpaceStatisticsArrayBuffer();
|
2015-12-29 10:54:35 +00:00
|
|
|
|
2016-09-30 22:31:47 +00:00
|
|
|
for (var i = 0; i < kNumberOfHeapSpaces; i++) {
|
2015-12-29 10:54:35 +00:00
|
|
|
const propertyOffset = i * kHeapSpaceStatisticsPropertiesCount;
|
|
|
|
heapSpaceStatistics[i] = {
|
|
|
|
space_name: kHeapSpaces[i],
|
|
|
|
space_size: buffer[propertyOffset + kSpaceSizeIndex],
|
|
|
|
space_used_size: buffer[propertyOffset + kSpaceUsedSizeIndex],
|
|
|
|
space_available_size: buffer[propertyOffset + kSpaceAvailableSizeIndex],
|
|
|
|
physical_space_size: buffer[propertyOffset + kPhysicalSpaceSizeIndex]
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
return heapSpaceStatistics;
|
2017-04-26 23:48:05 +00:00
|
|
|
}
|
2017-01-27 05:38:11 +00:00
|
|
|
|
2019-05-30 13:41:03 +00:00
|
|
|
function getHeapCodeStatistics() {
|
|
|
|
const buffer = heapCodeStatisticsBuffer;
|
|
|
|
|
|
|
|
updateHeapCodeStatisticsArrayBuffer();
|
|
|
|
return {
|
|
|
|
'code_and_metadata_size': buffer[kCodeAndMetadataSizeIndex],
|
|
|
|
'bytecode_and_metadata_size': buffer[kBytecodeAndMetadataSizeIndex],
|
|
|
|
'external_script_source_size': buffer[kExternalScriptSourceSizeIndex]
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2017-01-27 05:38:11 +00:00
|
|
|
/* V8 serialization API */
|
|
|
|
|
|
|
|
/* JS methods for the base objects */
|
|
|
|
Serializer.prototype._getDataCloneError = Error;
|
|
|
|
|
2017-04-26 23:48:05 +00:00
|
|
|
Deserializer.prototype.readRawBytes = function readRawBytes(length) {
|
2017-01-27 05:38:11 +00:00
|
|
|
const offset = this._readRawBytes(length);
|
|
|
|
// `this.buffer` can be a Buffer or a plain Uint8Array, so just calling
|
|
|
|
// `.slice()` doesn't work.
|
|
|
|
return new FastBuffer(this.buffer.buffer,
|
|
|
|
this.buffer.byteOffset + offset,
|
|
|
|
length);
|
|
|
|
};
|
|
|
|
|
|
|
|
/* Keep track of how to handle different ArrayBufferViews.
|
|
|
|
* The default Serializer for Node does not use the V8 methods for serializing
|
|
|
|
* those objects because Node's `Buffer` objects use pooled allocation in many
|
|
|
|
* cases, and their underlying `ArrayBuffer`s would show up in the
|
|
|
|
* serialization. Because a) those may contain sensitive data and the user
|
|
|
|
* may not be aware of that and b) they are often much larger than the `Buffer`
|
|
|
|
* itself, custom serialization is applied. */
|
|
|
|
const arrayBufferViewTypes = [Int8Array, Uint8Array, Uint8ClampedArray,
|
|
|
|
Int16Array, Uint16Array, Int32Array, Uint32Array,
|
|
|
|
Float32Array, Float64Array, DataView];
|
|
|
|
|
|
|
|
const arrayBufferViewTypeToIndex = new Map();
|
|
|
|
|
|
|
|
{
|
|
|
|
const dummy = new ArrayBuffer();
|
|
|
|
for (const [i, ctor] of arrayBufferViewTypes.entries()) {
|
2019-04-09 07:55:53 +00:00
|
|
|
const tag = ObjectPrototype.toString(new ctor(dummy));
|
2017-01-27 05:38:11 +00:00
|
|
|
arrayBufferViewTypeToIndex.set(tag, i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-06-07 17:21:47 +00:00
|
|
|
const bufferConstructorIndex = arrayBufferViewTypes.push(FastBuffer) - 1;
|
2017-01-27 05:38:11 +00:00
|
|
|
|
|
|
|
class DefaultSerializer extends Serializer {
|
|
|
|
constructor() {
|
|
|
|
super();
|
|
|
|
|
|
|
|
this._setTreatArrayBufferViewsAsHostObjects(true);
|
|
|
|
}
|
|
|
|
|
|
|
|
_writeHostObject(abView) {
|
|
|
|
let i = 0;
|
|
|
|
if (abView.constructor === Buffer) {
|
|
|
|
i = bufferConstructorIndex;
|
|
|
|
} else {
|
2019-04-09 07:55:53 +00:00
|
|
|
const tag = ObjectPrototype.toString(abView);
|
2017-01-27 05:38:11 +00:00
|
|
|
i = arrayBufferViewTypeToIndex.get(tag);
|
|
|
|
|
|
|
|
if (i === undefined) {
|
2019-10-29 20:24:23 +00:00
|
|
|
throw new this._getDataCloneError(
|
|
|
|
`Unserializable host object: ${inspect(abView)}`);
|
2017-01-27 05:38:11 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
this.writeUint32(i);
|
|
|
|
this.writeUint32(abView.byteLength);
|
|
|
|
this.writeRawBytes(new Uint8Array(abView.buffer,
|
|
|
|
abView.byteOffset,
|
|
|
|
abView.byteLength));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class DefaultDeserializer extends Deserializer {
|
|
|
|
_readHostObject() {
|
|
|
|
const typeIndex = this.readUint32();
|
|
|
|
const ctor = arrayBufferViewTypes[typeIndex];
|
|
|
|
const byteLength = this.readUint32();
|
|
|
|
const byteOffset = this._readRawBytes(byteLength);
|
|
|
|
const BYTES_PER_ELEMENT = ctor.BYTES_PER_ELEMENT || 1;
|
|
|
|
|
|
|
|
const offset = this.buffer.byteOffset + byteOffset;
|
|
|
|
if (offset % BYTES_PER_ELEMENT === 0) {
|
|
|
|
return new ctor(this.buffer.buffer,
|
|
|
|
offset,
|
|
|
|
byteLength / BYTES_PER_ELEMENT);
|
|
|
|
} else {
|
|
|
|
// Copy to an aligned buffer first.
|
2017-04-26 23:48:05 +00:00
|
|
|
const buffer_copy = Buffer.allocUnsafe(byteLength);
|
|
|
|
copy(this.buffer, buffer_copy, 0, byteOffset, byteOffset + byteLength);
|
|
|
|
return new ctor(buffer_copy.buffer,
|
|
|
|
buffer_copy.byteOffset,
|
2017-01-27 05:38:11 +00:00
|
|
|
byteLength / BYTES_PER_ELEMENT);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-04-26 23:48:05 +00:00
|
|
|
function serialize(value) {
|
2017-01-27 05:38:11 +00:00
|
|
|
const ser = new DefaultSerializer();
|
|
|
|
ser.writeHeader();
|
|
|
|
ser.writeValue(value);
|
|
|
|
return ser.releaseBuffer();
|
2017-04-26 23:48:05 +00:00
|
|
|
}
|
2017-01-27 05:38:11 +00:00
|
|
|
|
2017-04-26 23:48:05 +00:00
|
|
|
function deserialize(buffer) {
|
2017-01-27 05:38:11 +00:00
|
|
|
const der = new DefaultDeserializer(buffer);
|
|
|
|
der.readHeader();
|
|
|
|
return der.readValue();
|
2017-04-26 23:48:05 +00:00
|
|
|
}
|
|
|
|
|
2018-04-19 08:45:25 +00:00
|
|
|
module.exports = {
|
2017-04-26 23:48:05 +00:00
|
|
|
cachedDataVersionTag,
|
2019-03-07 16:51:36 +00:00
|
|
|
getHeapSnapshot,
|
2017-04-26 23:48:05 +00:00
|
|
|
getHeapStatistics,
|
|
|
|
getHeapSpaceStatistics,
|
2019-05-30 13:41:03 +00:00
|
|
|
getHeapCodeStatistics,
|
2017-04-26 23:48:05 +00:00
|
|
|
setFlagsFromString,
|
|
|
|
Serializer,
|
|
|
|
Deserializer,
|
|
|
|
DefaultSerializer,
|
|
|
|
DefaultDeserializer,
|
|
|
|
deserialize,
|
2019-03-07 16:51:36 +00:00
|
|
|
serialize,
|
|
|
|
writeHeapSnapshot
|
2017-01-27 05:38:11 +00:00
|
|
|
};
|