monaca-lib
Version:
Monaca cloud API bindings for JavaScript
1,665 lines (1,488 loc) • 81.8 kB
JavaScript
/*
* Copyright (C) 2011 Google Inc. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are
* met:
*
* * Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* * Redistributions in binary form must reproduce the above
* copyright notice, this list of conditions and the following disclaimer
* in the documentation and/or other materials provided with the
* distribution.
* * Neither the name of Google Inc. nor the names of its
* contributors may be used to endorse or promote products derived from
* this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
/**
* @interface
*/
WebInspector.HeapSnapshotItem = function() { }
WebInspector.HeapSnapshotItem.prototype = {
/**
* @return {number}
*/
itemIndex: function() { },
/**
* @return {!Object}
*/
serialize: function() { }
};
/**
* @constructor
* @implements {WebInspector.HeapSnapshotItem}
* @param {!WebInspector.HeapSnapshot} snapshot
* @param {number=} edgeIndex
*/
WebInspector.HeapSnapshotEdge = function(snapshot, edgeIndex)
{
this._snapshot = snapshot;
this._edges = snapshot.containmentEdges;
this.edgeIndex = edgeIndex || 0;
}
WebInspector.HeapSnapshotEdge.prototype = {
/**
* @return {!WebInspector.HeapSnapshotEdge}
*/
clone: function()
{
return new WebInspector.HeapSnapshotEdge(this._snapshot, this.edgeIndex);
},
/**
* @return {boolean}
*/
hasStringName: function()
{
throw new Error("Not implemented");
},
/**
* @return {string}
*/
name: function()
{
throw new Error("Not implemented");
},
/**
* @return {!WebInspector.HeapSnapshotNode}
*/
node: function()
{
return this._snapshot.createNode(this.nodeIndex());
},
/**
* @return {number}
*/
nodeIndex: function()
{
return this._edges[this.edgeIndex + this._snapshot._edgeToNodeOffset];
},
/**
* @override
* @return {string}
*/
toString: function()
{
return "HeapSnapshotEdge: " + this.name();
},
/**
* @return {string}
*/
type: function()
{
return this._snapshot._edgeTypes[this._type()];
},
/**
* @override
* @return {number}
*/
itemIndex: function()
{
return this.edgeIndex;
},
/**
* @override
* @return {!WebInspector.HeapSnapshotCommon.Edge}
*/
serialize: function()
{
return new WebInspector.HeapSnapshotCommon.Edge(this.name(), this.node().serialize(), this.type(), this.edgeIndex);
},
_type: function()
{
return this._edges[this.edgeIndex + this._snapshot._edgeTypeOffset];
}
};
/**
* @interface
*/
WebInspector.HeapSnapshotItemIterator = function() { }
WebInspector.HeapSnapshotItemIterator.prototype = {
/**
* @return {boolean}
*/
hasNext: function() { },
/**
* @return {!WebInspector.HeapSnapshotItem}
*/
item: function() { },
next: function() { }
};
/**
* @interface
*/
WebInspector.HeapSnapshotItemIndexProvider = function() { }
WebInspector.HeapSnapshotItemIndexProvider.prototype = {
/**
* @param {number} newIndex
* @return {!WebInspector.HeapSnapshotItem}
*/
itemForIndex: function(newIndex) { },
};
/**
* @constructor
* @implements {WebInspector.HeapSnapshotItemIndexProvider}
* @param {!WebInspector.HeapSnapshot} snapshot
*/
WebInspector.HeapSnapshotNodeIndexProvider = function(snapshot)
{
this._node = snapshot.createNode();
}
WebInspector.HeapSnapshotNodeIndexProvider.prototype = {
/**
* @override
* @param {number} index
* @return {!WebInspector.HeapSnapshotNode}
*/
itemForIndex: function(index)
{
this._node.nodeIndex = index;
return this._node;
}
};
/**
* @constructor
* @implements {WebInspector.HeapSnapshotItemIndexProvider}
* @param {!WebInspector.HeapSnapshot} snapshot
*/
WebInspector.HeapSnapshotEdgeIndexProvider = function(snapshot)
{
this._edge = snapshot.createEdge(0);
}
WebInspector.HeapSnapshotEdgeIndexProvider.prototype = {
/**
* @override
* @param {number} index
* @return {!WebInspector.HeapSnapshotEdge}
*/
itemForIndex: function(index)
{
this._edge.edgeIndex = index;
return this._edge;
}
};
/**
* @constructor
* @implements {WebInspector.HeapSnapshotItemIndexProvider}
* @param {!WebInspector.HeapSnapshot} snapshot
*/
WebInspector.HeapSnapshotRetainerEdgeIndexProvider = function(snapshot)
{
this._retainerEdge = snapshot.createRetainingEdge(0);
}
WebInspector.HeapSnapshotRetainerEdgeIndexProvider.prototype = {
/**
* @override
* @param {number} index
* @return {!WebInspector.HeapSnapshotRetainerEdge}
*/
itemForIndex: function(index)
{
this._retainerEdge.setRetainerIndex(index);
return this._retainerEdge;
}
};
/**
* @constructor
* @implements {WebInspector.HeapSnapshotItemIterator}
* @param {!WebInspector.HeapSnapshotNode} node
*/
WebInspector.HeapSnapshotEdgeIterator = function(node)
{
this._sourceNode = node;
this.edge = node._snapshot.createEdge(node.edgeIndexesStart());
}
WebInspector.HeapSnapshotEdgeIterator.prototype = {
/**
* @override
* @return {boolean}
*/
hasNext: function()
{
return this.edge.edgeIndex < this._sourceNode.edgeIndexesEnd();
},
/**
* @override
* @return {!WebInspector.HeapSnapshotEdge}
*/
item: function()
{
return this.edge;
},
/**
* @override
*/
next: function()
{
this.edge.edgeIndex += this.edge._snapshot._edgeFieldsCount;
}
};
/**
* @constructor
* @implements {WebInspector.HeapSnapshotItem}
* @param {!WebInspector.HeapSnapshot} snapshot
* @param {number} retainerIndex
*/
WebInspector.HeapSnapshotRetainerEdge = function(snapshot, retainerIndex)
{
this._snapshot = snapshot;
this.setRetainerIndex(retainerIndex);
}
WebInspector.HeapSnapshotRetainerEdge.prototype = {
/**
* @return {!WebInspector.HeapSnapshotRetainerEdge}
*/
clone: function()
{
return new WebInspector.HeapSnapshotRetainerEdge(this._snapshot, this.retainerIndex());
},
/**
* @return {boolean}
*/
hasStringName: function()
{
return this._edge().hasStringName();
},
/**
* @return {string}
*/
name: function()
{
return this._edge().name();
},
/**
* @return {!WebInspector.HeapSnapshotNode}
*/
node: function()
{
return this._node();
},
/**
* @return {number}
*/
nodeIndex: function()
{
return this._retainingNodeIndex;
},
/**
* @return {number}
*/
retainerIndex: function()
{
return this._retainerIndex;
},
/**
* @param {number} retainerIndex
*/
setRetainerIndex: function(retainerIndex)
{
if (retainerIndex === this._retainerIndex)
return;
this._retainerIndex = retainerIndex;
this._globalEdgeIndex = this._snapshot._retainingEdges[retainerIndex];
this._retainingNodeIndex = this._snapshot._retainingNodes[retainerIndex];
this._edgeInstance = null;
this._nodeInstance = null;
},
/**
* @param {number} edgeIndex
*/
set edgeIndex(edgeIndex)
{
this.setRetainerIndex(edgeIndex);
},
_node: function()
{
if (!this._nodeInstance)
this._nodeInstance = this._snapshot.createNode(this._retainingNodeIndex);
return this._nodeInstance;
},
_edge: function()
{
if (!this._edgeInstance)
this._edgeInstance = this._snapshot.createEdge(this._globalEdgeIndex);
return this._edgeInstance;
},
/**
* @override
* @return {string}
*/
toString: function()
{
return this._edge().toString();
},
/**
* @override
* @return {number}
*/
itemIndex: function()
{
return this._retainerIndex;
},
/**
* @override
* @return {!WebInspector.HeapSnapshotCommon.Edge}
*/
serialize: function()
{
return new WebInspector.HeapSnapshotCommon.Edge(this.name(), this.node().serialize(), this.type(), this._globalEdgeIndex);
},
/**
* @return {string}
*/
type: function()
{
return this._edge().type();
}
}
/**
* @constructor
* @implements {WebInspector.HeapSnapshotItemIterator}
* @param {!WebInspector.HeapSnapshotNode} retainedNode
*/
WebInspector.HeapSnapshotRetainerEdgeIterator = function(retainedNode)
{
var snapshot = retainedNode._snapshot;
var retainedNodeOrdinal = retainedNode.ordinal();
var retainerIndex = snapshot._firstRetainerIndex[retainedNodeOrdinal];
this._retainersEnd = snapshot._firstRetainerIndex[retainedNodeOrdinal + 1];
this.retainer = snapshot.createRetainingEdge(retainerIndex);
}
WebInspector.HeapSnapshotRetainerEdgeIterator.prototype = {
/**
* @override
* @return {boolean}
*/
hasNext: function()
{
return this.retainer.retainerIndex() < this._retainersEnd;
},
/**
* @override
* @return {!WebInspector.HeapSnapshotRetainerEdge}
*/
item: function()
{
return this.retainer;
},
/**
* @override
*/
next: function()
{
this.retainer.setRetainerIndex(this.retainer.retainerIndex() + 1);
}
};
/**
* @constructor
* @implements {WebInspector.HeapSnapshotItem}
* @param {!WebInspector.HeapSnapshot} snapshot
* @param {number=} nodeIndex
*/
WebInspector.HeapSnapshotNode = function(snapshot, nodeIndex)
{
this._snapshot = snapshot;
this.nodeIndex = nodeIndex || 0;
}
WebInspector.HeapSnapshotNode.prototype = {
/**
* @return {number}
*/
distance: function()
{
return this._snapshot._nodeDistances[this.nodeIndex / this._snapshot._nodeFieldCount];
},
/**
* @return {string}
*/
className: function()
{
throw new Error("Not implemented");
},
/**
* @return {number}
*/
classIndex: function()
{
throw new Error("Not implemented");
},
/**
* @return {number}
*/
dominatorIndex: function()
{
var nodeFieldCount = this._snapshot._nodeFieldCount;
return this._snapshot._dominatorsTree[this.nodeIndex / this._snapshot._nodeFieldCount] * nodeFieldCount;
},
/**
* @return {!WebInspector.HeapSnapshotEdgeIterator}
*/
edges: function()
{
return new WebInspector.HeapSnapshotEdgeIterator(this);
},
/**
* @return {number}
*/
edgesCount: function()
{
return (this.edgeIndexesEnd() - this.edgeIndexesStart()) / this._snapshot._edgeFieldsCount;
},
/**
* @return {number}
*/
id: function()
{
throw new Error("Not implemented");
},
/**
* @return {boolean}
*/
isRoot: function()
{
return this.nodeIndex === this._snapshot._rootNodeIndex;
},
/**
* @return {string}
*/
name: function()
{
return this._snapshot.strings[this._name()];
},
/**
* @return {number}
*/
retainedSize: function()
{
return this._snapshot._retainedSizes[this.ordinal()];
},
/**
* @return {!WebInspector.HeapSnapshotRetainerEdgeIterator}
*/
retainers: function()
{
return new WebInspector.HeapSnapshotRetainerEdgeIterator(this);
},
/**
* @return {number}
*/
retainersCount: function()
{
var snapshot = this._snapshot;
var ordinal = this.ordinal();
return snapshot._firstRetainerIndex[ordinal + 1] - snapshot._firstRetainerIndex[ordinal];
},
/**
* @return {number}
*/
selfSize: function()
{
var snapshot = this._snapshot;
return snapshot.nodes[this.nodeIndex + snapshot._nodeSelfSizeOffset];
},
/**
* @return {string}
*/
type: function()
{
return this._snapshot._nodeTypes[this._type()];
},
/**
* @return {number}
*/
traceNodeId: function()
{
var snapshot = this._snapshot;
return snapshot.nodes[this.nodeIndex + snapshot._nodeTraceNodeIdOffset];
},
/**
* @override
* @return {number}
*/
itemIndex: function()
{
return this.nodeIndex;
},
/**
* @override
* @return {!WebInspector.HeapSnapshotCommon.Node}
*/
serialize: function()
{
return new WebInspector.HeapSnapshotCommon.Node(this.id(), this.name(), this.distance(), this.nodeIndex, this.retainedSize(), this.selfSize(), this.type());
},
/**
* @return {number}
*/
_name: function()
{
var snapshot = this._snapshot;
return snapshot.nodes[this.nodeIndex + snapshot._nodeNameOffset];
},
/**
* @return {number}
*/
edgeIndexesStart: function()
{
return this._snapshot._firstEdgeIndexes[this.ordinal()];
},
/**
* @return {number}
*/
edgeIndexesEnd: function()
{
return this._snapshot._firstEdgeIndexes[this.ordinal() + 1];
},
/**
* @return {number}
*/
ordinal: function()
{
return this.nodeIndex / this._snapshot._nodeFieldCount;
},
/**
* @return {number}
*/
_nextNodeIndex: function()
{
return this.nodeIndex + this._snapshot._nodeFieldCount;
},
/**
* @return {number}
*/
_type: function()
{
var snapshot = this._snapshot;
return snapshot.nodes[this.nodeIndex + snapshot._nodeTypeOffset];
}
};
/**
* @constructor
* @implements {WebInspector.HeapSnapshotItemIterator}
* @param {!WebInspector.HeapSnapshotNode} node
*/
WebInspector.HeapSnapshotNodeIterator = function(node)
{
this.node = node;
this._nodesLength = node._snapshot.nodes.length;
}
WebInspector.HeapSnapshotNodeIterator.prototype = {
/**
* @override
* @return {boolean}
*/
hasNext: function()
{
return this.node.nodeIndex < this._nodesLength;
},
/**
* @override
* @return {!WebInspector.HeapSnapshotNode}
*/
item: function()
{
return this.node;
},
/**
* @override
*/
next: function()
{
this.node.nodeIndex = this.node._nextNodeIndex();
}
}
/**
* @constructor
* @implements {WebInspector.HeapSnapshotItemIterator}
* @param {!WebInspector.HeapSnapshotItemIndexProvider} itemProvider
* @param {!Array.<number>|!Uint32Array} indexes
*/
WebInspector.HeapSnapshotIndexRangeIterator = function(itemProvider, indexes)
{
this._itemProvider = itemProvider;
this._indexes = indexes;
this._position = 0;
}
WebInspector.HeapSnapshotIndexRangeIterator.prototype = {
/**
* @override
* @return {boolean}
*/
hasNext: function()
{
return this._position < this._indexes.length;
},
/**
* @override
* @return {!WebInspector.HeapSnapshotItem}
*/
item: function()
{
var index = this._indexes[this._position];
return this._itemProvider.itemForIndex(index);
},
/**
* @override
*/
next: function()
{
++this._position;
}
}
/**
* @constructor
* @implements {WebInspector.HeapSnapshotItemIterator}
* @param {!WebInspector.HeapSnapshotItemIterator} iterator
* @param {function(!WebInspector.HeapSnapshotItem):boolean=} filter
*/
WebInspector.HeapSnapshotFilteredIterator = function(iterator, filter)
{
this._iterator = iterator;
this._filter = filter;
this._skipFilteredItems();
}
WebInspector.HeapSnapshotFilteredIterator.prototype = {
/**
* @override
* @return {boolean}
*/
hasNext: function()
{
return this._iterator.hasNext();
},
/**
* @override
* @return {!WebInspector.HeapSnapshotItem}
*/
item: function()
{
return this._iterator.item();
},
/**
* @override
*/
next: function()
{
this._iterator.next();
this._skipFilteredItems();
},
_skipFilteredItems: function()
{
while (this._iterator.hasNext() && !this._filter(this._iterator.item())) {
this._iterator.next();
}
}
}
/**
* @param {!WebInspector.HeapSnapshotWorkerDispatcher=} dispatcher
* @constructor
*/
WebInspector.HeapSnapshotProgress = function(dispatcher)
{
this._dispatcher = dispatcher;
}
WebInspector.HeapSnapshotProgress.prototype = {
/**
* @param {string} status
*/
updateStatus: function(status)
{
this._sendUpdateEvent(WebInspector.UIString(status));
},
/**
* @param {string} title
* @param {number} value
* @param {number} total
*/
updateProgress: function(title, value, total)
{
var percentValue = ((total ? (value / total) : 0) * 100).toFixed(0);
this._sendUpdateEvent(WebInspector.UIString(title, percentValue));
},
/**
* @param {string} error
*/
reportProblem: function(error)
{
// May be undefined in tests.
if (this._dispatcher)
this._dispatcher.sendEvent(WebInspector.HeapSnapshotProgressEvent.BrokenSnapshot, error);
},
/**
* @param {string} text
*/
_sendUpdateEvent: function(text)
{
// May be undefined in tests.
if (this._dispatcher)
this._dispatcher.sendEvent(WebInspector.HeapSnapshotProgressEvent.Update, text);
}
}
/**
* @param {string} title
* @constructor
*/
WebInspector.HeapSnapshotProblemReport = function(title)
{
this._errors = [title];
}
WebInspector.HeapSnapshotProblemReport.prototype = {
/**
* @param {string} error
*/
addError: function(error)
{
if (this._errors.length > 100)
return;
this._errors.push(error);
},
/**
* @override
* @return {string}
*/
toString: function()
{
return this._errors.join("\n ");
}
}
/**
* @param {!Object} profile
* @param {!WebInspector.HeapSnapshotProgress} progress
* @constructor
*/
WebInspector.HeapSnapshot = function(profile, progress)
{
/** @type {!Uint32Array} */
this.nodes = profile.nodes;
/** @type {!Uint32Array} */
this.containmentEdges = profile.edges;
/** @type {!HeapSnapshotMetainfo} */
this._metaNode = profile.snapshot.meta;
/** @type {!Array.<string>} */
this.strings = profile.strings;
this._progress = progress;
this._noDistance = -5;
this._rootNodeIndex = 0;
if (profile.snapshot.root_index)
this._rootNodeIndex = profile.snapshot.root_index;
this._snapshotDiffs = {};
this._aggregatesForDiff = null;
this._aggregates = {};
this._aggregatesSortedFlags = {};
this._init();
if (profile.snapshot.trace_function_count) {
this._progress.updateStatus("Building allocation statistics\u2026");
var nodes = this.nodes;
var nodesLength = nodes.length;
var nodeFieldCount = this._nodeFieldCount;
var node = this.rootNode();
var liveObjects = {};
for (var nodeIndex = 0; nodeIndex < nodesLength; nodeIndex += nodeFieldCount) {
node.nodeIndex = nodeIndex;
var traceNodeId = node.traceNodeId();
var stats = liveObjects[traceNodeId];
if (!stats)
liveObjects[traceNodeId] = stats = { count: 0, size: 0, ids: [] };
stats.count++;
stats.size += node.selfSize();
stats.ids.push(node.id());
}
this._allocationProfile = new WebInspector.AllocationProfile(profile, liveObjects);
this._progress.updateStatus("Done");
}
}
/**
* @constructor
*/
function HeapSnapshotMetainfo()
{
// New format.
this.node_fields = [];
this.node_types = [];
this.edge_fields = [];
this.edge_types = [];
this.trace_function_info_fields = [];
this.trace_node_fields = [];
this.type_strings = {};
}
/**
* @constructor
*/
function HeapSnapshotHeader()
{
// New format.
this.title = "";
this.meta = new HeapSnapshotMetainfo();
this.node_count = 0;
this.edge_count = 0;
}
WebInspector.HeapSnapshot.prototype = {
_init: function()
{
var meta = this._metaNode;
this._nodeTypeOffset = meta.node_fields.indexOf("type");
this._nodeNameOffset = meta.node_fields.indexOf("name");
this._nodeIdOffset = meta.node_fields.indexOf("id");
this._nodeSelfSizeOffset = meta.node_fields.indexOf("self_size");
this._nodeEdgeCountOffset = meta.node_fields.indexOf("edge_count");
this._nodeTraceNodeIdOffset = meta.node_fields.indexOf("trace_node_id");
this._nodeFieldCount = meta.node_fields.length;
this._nodeTypes = meta.node_types[this._nodeTypeOffset];
this._nodeArrayType = this._nodeTypes.indexOf("array");
this._nodeHiddenType = this._nodeTypes.indexOf("hidden");
this._nodeObjectType = this._nodeTypes.indexOf("object");
this._nodeNativeType = this._nodeTypes.indexOf("native");
this._nodeConsStringType = this._nodeTypes.indexOf("concatenated string");
this._nodeSlicedStringType = this._nodeTypes.indexOf("sliced string");
this._nodeCodeType = this._nodeTypes.indexOf("code");
this._nodeSyntheticType = this._nodeTypes.indexOf("synthetic");
this._edgeFieldsCount = meta.edge_fields.length;
this._edgeTypeOffset = meta.edge_fields.indexOf("type");
this._edgeNameOffset = meta.edge_fields.indexOf("name_or_index");
this._edgeToNodeOffset = meta.edge_fields.indexOf("to_node");
this._edgeTypes = meta.edge_types[this._edgeTypeOffset];
this._edgeTypes.push("invisible");
this._edgeElementType = this._edgeTypes.indexOf("element");
this._edgeHiddenType = this._edgeTypes.indexOf("hidden");
this._edgeInternalType = this._edgeTypes.indexOf("internal");
this._edgeShortcutType = this._edgeTypes.indexOf("shortcut");
this._edgeWeakType = this._edgeTypes.indexOf("weak");
this._edgeInvisibleType = this._edgeTypes.indexOf("invisible");
this.nodeCount = this.nodes.length / this._nodeFieldCount;
this._edgeCount = this.containmentEdges.length / this._edgeFieldsCount;
this._retainedSizes = new Float64Array(this.nodeCount);
this._firstEdgeIndexes = new Uint32Array(this.nodeCount + 1);
this._retainingNodes = new Uint32Array(this._edgeCount);
this._retainingEdges = new Uint32Array(this._edgeCount);
this._firstRetainerIndex = new Uint32Array(this.nodeCount + 1);
this._nodeDistances = new Int32Array(this.nodeCount);
this._firstDominatedNodeIndex = new Uint32Array(this.nodeCount + 1);
this._dominatedNodes = new Uint32Array(this.nodeCount - 1);
this._progress.updateStatus("Building edge indexes\u2026");
this._buildEdgeIndexes();
this._progress.updateStatus("Building retainers\u2026");
this._buildRetainers();
this._progress.updateStatus("Calculating node flags\u2026");
this._calculateFlags();
this._progress.updateStatus("Calculating distances\u2026");
this.calculateDistances();
this._progress.updateStatus("Building postorder index\u2026");
var result = this._buildPostOrderIndex();
// Actually it is array that maps node ordinal number to dominator node ordinal number.
this._progress.updateStatus("Building dominator tree\u2026");
this._dominatorsTree = this._buildDominatorTree(result.postOrderIndex2NodeOrdinal, result.nodeOrdinal2PostOrderIndex);
this._progress.updateStatus("Calculating retained sizes\u2026");
this._calculateRetainedSizes(result.postOrderIndex2NodeOrdinal);
this._progress.updateStatus("Buiding dominated nodes\u2026");
this._buildDominatedNodes();
this._progress.updateStatus("Calculating statistics\u2026");
this._calculateStatistics();
this._progress.updateStatus("Finished processing.");
},
_buildEdgeIndexes: function()
{
var nodes = this.nodes;
var nodeCount = this.nodeCount;
var firstEdgeIndexes = this._firstEdgeIndexes;
var nodeFieldCount = this._nodeFieldCount;
var edgeFieldsCount = this._edgeFieldsCount;
var nodeEdgeCountOffset = this._nodeEdgeCountOffset;
firstEdgeIndexes[nodeCount] = this.containmentEdges.length;
for (var nodeOrdinal = 0, edgeIndex = 0; nodeOrdinal < nodeCount; ++nodeOrdinal) {
firstEdgeIndexes[nodeOrdinal] = edgeIndex;
edgeIndex += nodes[nodeOrdinal * nodeFieldCount + nodeEdgeCountOffset] * edgeFieldsCount;
}
},
_buildRetainers: function()
{
var retainingNodes = this._retainingNodes;
var retainingEdges = this._retainingEdges;
// Index of the first retainer in the _retainingNodes and _retainingEdges
// arrays. Addressed by retained node index.
var firstRetainerIndex = this._firstRetainerIndex;
var containmentEdges = this.containmentEdges;
var edgeFieldsCount = this._edgeFieldsCount;
var nodeFieldCount = this._nodeFieldCount;
var edgeToNodeOffset = this._edgeToNodeOffset;
var firstEdgeIndexes = this._firstEdgeIndexes;
var nodeCount = this.nodeCount;
for (var toNodeFieldIndex = edgeToNodeOffset, l = containmentEdges.length; toNodeFieldIndex < l; toNodeFieldIndex += edgeFieldsCount) {
var toNodeIndex = containmentEdges[toNodeFieldIndex];
if (toNodeIndex % nodeFieldCount)
throw new Error("Invalid toNodeIndex " + toNodeIndex);
++firstRetainerIndex[toNodeIndex / nodeFieldCount];
}
for (var i = 0, firstUnusedRetainerSlot = 0; i < nodeCount; i++) {
var retainersCount = firstRetainerIndex[i];
firstRetainerIndex[i] = firstUnusedRetainerSlot;
retainingNodes[firstUnusedRetainerSlot] = retainersCount;
firstUnusedRetainerSlot += retainersCount;
}
firstRetainerIndex[nodeCount] = retainingNodes.length;
var nextNodeFirstEdgeIndex = firstEdgeIndexes[0];
for (var srcNodeOrdinal = 0; srcNodeOrdinal < nodeCount; ++srcNodeOrdinal) {
var firstEdgeIndex = nextNodeFirstEdgeIndex;
nextNodeFirstEdgeIndex = firstEdgeIndexes[srcNodeOrdinal + 1];
var srcNodeIndex = srcNodeOrdinal * nodeFieldCount;
for (var edgeIndex = firstEdgeIndex; edgeIndex < nextNodeFirstEdgeIndex; edgeIndex += edgeFieldsCount) {
var toNodeIndex = containmentEdges[edgeIndex + edgeToNodeOffset];
if (toNodeIndex % nodeFieldCount)
throw new Error("Invalid toNodeIndex " + toNodeIndex);
var firstRetainerSlotIndex = firstRetainerIndex[toNodeIndex / nodeFieldCount];
var nextUnusedRetainerSlotIndex = firstRetainerSlotIndex + (--retainingNodes[firstRetainerSlotIndex]);
retainingNodes[nextUnusedRetainerSlotIndex] = srcNodeIndex;
retainingEdges[nextUnusedRetainerSlotIndex] = edgeIndex;
}
}
},
/**
* @param {number=} nodeIndex
*/
createNode: function(nodeIndex)
{
throw new Error("Not implemented");
},
/**
* @param {number} edgeIndex
* @return {!WebInspector.JSHeapSnapshotEdge}
*/
createEdge: function(edgeIndex)
{
throw new Error("Not implemented");
},
/**
* @param {number} retainerIndex
* @return {!WebInspector.JSHeapSnapshotRetainerEdge}
*/
createRetainingEdge: function(retainerIndex)
{
throw new Error("Not implemented");
},
_allNodes: function()
{
return new WebInspector.HeapSnapshotNodeIterator(this.rootNode());
},
/**
* @return {!WebInspector.HeapSnapshotNode}
*/
rootNode: function()
{
return this.createNode(this._rootNodeIndex);
},
get rootNodeIndex()
{
return this._rootNodeIndex;
},
get totalSize()
{
return this.rootNode().retainedSize();
},
_getDominatedIndex: function(nodeIndex)
{
if (nodeIndex % this._nodeFieldCount)
throw new Error("Invalid nodeIndex: " + nodeIndex);
return this._firstDominatedNodeIndex[nodeIndex / this._nodeFieldCount];
},
/**
* @param {!WebInspector.HeapSnapshotCommon.NodeFilter} nodeFilter
* @return {undefined|function(!WebInspector.HeapSnapshotNode):boolean}
*/
_createFilter: function(nodeFilter)
{
var minNodeId = nodeFilter.minNodeId;
var maxNodeId = nodeFilter.maxNodeId;
var allocationNodeId = nodeFilter.allocationNodeId;
var filter;
if (typeof allocationNodeId === "number") {
filter = this._createAllocationStackFilter(allocationNodeId);
filter.key = "AllocationNodeId: " + allocationNodeId;
} else if (typeof minNodeId === "number" && typeof maxNodeId === "number") {
filter = this._createNodeIdFilter(minNodeId, maxNodeId);
filter.key = "NodeIdRange: " + minNodeId + ".." + maxNodeId;
}
return filter;
},
/**
* @param {!WebInspector.HeapSnapshotCommon.SearchConfig} searchConfig
* @param {!WebInspector.HeapSnapshotCommon.NodeFilter} nodeFilter
* @return {!Array.<number>}
*/
search: function(searchConfig, nodeFilter)
{
var query = searchConfig.query;
function filterString(matchedStringIndexes, string, index)
{
if (string.indexOf(query) !== -1)
matchedStringIndexes.add(index);
return matchedStringIndexes;
}
var regexp = searchConfig.isRegex ? new RegExp(query) : createPlainTextSearchRegex(query, "i");
function filterRegexp(matchedStringIndexes, string, index)
{
if (regexp.test(string))
matchedStringIndexes.add(index);
return matchedStringIndexes;
}
var stringFilter = (searchConfig.isRegex || !searchConfig.caseSensitive) ? filterRegexp : filterString;
var stringIndexes = this.strings.reduce(stringFilter, new Set());
if (!stringIndexes.size)
return [];
var filter = this._createFilter(nodeFilter);
var nodeIds = [];
var nodesLength = this.nodes.length;
var nodes = this.nodes;
var nodeNameOffset = this._nodeNameOffset;
var nodeIdOffset = this._nodeIdOffset;
var nodeFieldCount = this._nodeFieldCount;
var node = this.rootNode();
for (var nodeIndex = 0; nodeIndex < nodesLength; nodeIndex += nodeFieldCount) {
node.nodeIndex = nodeIndex;
if (filter && !filter(node))
continue;
if (stringIndexes.has(nodes[nodeIndex + nodeNameOffset]))
nodeIds.push(nodes[nodeIndex + nodeIdOffset]);
}
return nodeIds;
},
/**
* @param {!WebInspector.HeapSnapshotCommon.NodeFilter} nodeFilter
* @return {!Object.<string, !WebInspector.HeapSnapshotCommon.Aggregate>}
*/
aggregatesWithFilter: function(nodeFilter)
{
var filter = this._createFilter(nodeFilter);
var key = filter ? filter.key : "allObjects";
return this.aggregates(false, key, filter);
},
/**
* @param {number} minNodeId
* @param {number} maxNodeId
* @return {function(!WebInspector.HeapSnapshotNode):boolean}
*/
_createNodeIdFilter: function(minNodeId, maxNodeId)
{
/**
* @param {!WebInspector.HeapSnapshotNode} node
* @return {boolean}
*/
function nodeIdFilter(node)
{
var id = node.id();
return id > minNodeId && id <= maxNodeId;
}
return nodeIdFilter;
},
/**
* @param {number} bottomUpAllocationNodeId
* @return {function(!WebInspector.HeapSnapshotNode):boolean|undefined}
*/
_createAllocationStackFilter: function(bottomUpAllocationNodeId)
{
var traceIds = this._allocationProfile.traceIds(bottomUpAllocationNodeId);
if (!traceIds.length)
return undefined;
var set = {};
for (var i = 0; i < traceIds.length; i++)
set[traceIds[i]] = true;
/**
* @param {!WebInspector.HeapSnapshotNode} node
* @return {boolean}
*/
function traceIdFilter(node)
{
return !!set[node.traceNodeId()];
};
return traceIdFilter;
},
/**
* @param {boolean} sortedIndexes
* @param {string=} key
* @param {function(!WebInspector.HeapSnapshotNode):boolean=} filter
* @return {!Object.<string, !WebInspector.HeapSnapshotCommon.Aggregate>}
*/
aggregates: function(sortedIndexes, key, filter)
{
var aggregatesByClassName = key && this._aggregates[key];
if (!aggregatesByClassName) {
var aggregates = this._buildAggregates(filter);
this._calculateClassesRetainedSize(aggregates.aggregatesByClassIndex, filter);
aggregatesByClassName = aggregates.aggregatesByClassName;
if (key)
this._aggregates[key] = aggregatesByClassName;
}
if (sortedIndexes && (!key || !this._aggregatesSortedFlags[key])) {
this._sortAggregateIndexes(aggregatesByClassName);
if (key)
this._aggregatesSortedFlags[key] = sortedIndexes;
}
return aggregatesByClassName;
},
/**
* @return {!Array.<!WebInspector.HeapSnapshotCommon.SerializedAllocationNode>}
*/
allocationTracesTops: function()
{
return this._allocationProfile.serializeTraceTops();
},
/**
* @param {number} nodeId
* @return {!WebInspector.HeapSnapshotCommon.AllocationNodeCallers}
*/
allocationNodeCallers: function(nodeId)
{
return this._allocationProfile.serializeCallers(nodeId);
},
/**
* @param {number} nodeIndex
* @return {?Array.<!WebInspector.HeapSnapshotCommon.AllocationStackFrame>}
*/
allocationStack: function(nodeIndex)
{
var node = this.createNode(nodeIndex);
var allocationNodeId = node.traceNodeId();
if (!allocationNodeId)
return null;
return this._allocationProfile.serializeAllocationStack(allocationNodeId);
},
/**
* @return {!Object.<string, !WebInspector.HeapSnapshotCommon.AggregateForDiff>}
*/
aggregatesForDiff: function()
{
if (this._aggregatesForDiff)
return this._aggregatesForDiff;
var aggregatesByClassName = this.aggregates(true, "allObjects");
this._aggregatesForDiff = {};
var node = this.createNode();
for (var className in aggregatesByClassName) {
var aggregate = aggregatesByClassName[className];
var indexes = aggregate.idxs;
var ids = new Array(indexes.length);
var selfSizes = new Array(indexes.length);
for (var i = 0; i < indexes.length; i++) {
node.nodeIndex = indexes[i];
ids[i] = node.id();
selfSizes[i] = node.selfSize();
}
this._aggregatesForDiff[className] = {
indexes: indexes,
ids: ids,
selfSizes: selfSizes
};
}
return this._aggregatesForDiff;
},
/**
* @protected
* @param {!WebInspector.HeapSnapshotNode} node
* @return {boolean}
*/
isUserRoot: function(node)
{
return true;
},
/**
* @param {function(!WebInspector.HeapSnapshotNode)} action
* @param {boolean=} userRootsOnly
*/
forEachRoot: function(action, userRootsOnly)
{
for (var iter = this.rootNode().edges(); iter.hasNext(); iter.next()) {
var node = iter.edge.node();
if (!userRootsOnly || this.isUserRoot(node))
action(node);
}
},
/**
* @param {function(!WebInspector.HeapSnapshotNode,!WebInspector.HeapSnapshotEdge):boolean=} filter
*/
calculateDistances: function(filter)
{
var nodeCount = this.nodeCount;
var distances = this._nodeDistances;
var noDistance = this._noDistance;
for (var i = 0; i < nodeCount; ++i)
distances[i] = noDistance;
var nodesToVisit = new Uint32Array(this.nodeCount);
var nodesToVisitLength = 0;
/**
* @param {number} distance
* @param {!WebInspector.HeapSnapshotNode} node
*/
function enqueueNode(distance, node)
{
var ordinal = node.ordinal();
if (distances[ordinal] !== noDistance)
return;
distances[ordinal] = distance;
nodesToVisit[nodesToVisitLength++] = node.nodeIndex;
}
this.forEachRoot(enqueueNode.bind(null, 1), true);
this._bfs(nodesToVisit, nodesToVisitLength, distances, filter);
// bfs for the rest of objects
nodesToVisitLength = 0;
this.forEachRoot(enqueueNode.bind(null, WebInspector.HeapSnapshotCommon.baseSystemDistance), false);
this._bfs(nodesToVisit, nodesToVisitLength, distances, filter);
},
/**
* @param {!Uint32Array} nodesToVisit
* @param {number} nodesToVisitLength
* @param {!Int32Array} distances
* @param {function(!WebInspector.HeapSnapshotNode,!WebInspector.HeapSnapshotEdge):boolean=} filter
*/
_bfs: function(nodesToVisit, nodesToVisitLength, distances, filter)
{
// Preload fields into local variables for better performance.
var edgeFieldsCount = this._edgeFieldsCount;
var nodeFieldCount = this._nodeFieldCount;
var containmentEdges = this.containmentEdges;
var firstEdgeIndexes = this._firstEdgeIndexes;
var edgeToNodeOffset = this._edgeToNodeOffset;
var edgeTypeOffset = this._edgeTypeOffset;
var nodeCount = this.nodeCount;
var edgeWeakType = this._edgeWeakType;
var noDistance = this._noDistance;
var index = 0;
var edge = this.createEdge(0);
var node = this.createNode(0);
while (index < nodesToVisitLength) {
var nodeIndex = nodesToVisit[index++]; // shift generates too much garbage.
var nodeOrdinal = nodeIndex / nodeFieldCount;
var distance = distances[nodeOrdinal] + 1;
var firstEdgeIndex = firstEdgeIndexes[nodeOrdinal];
var edgesEnd = firstEdgeIndexes[nodeOrdinal + 1];
node.nodeIndex = nodeIndex;
for (var edgeIndex = firstEdgeIndex; edgeIndex < edgesEnd; edgeIndex += edgeFieldsCount) {
var edgeType = containmentEdges[edgeIndex + edgeTypeOffset];
if (edgeType === edgeWeakType)
continue;
var childNodeIndex = containmentEdges[edgeIndex + edgeToNodeOffset];
var childNodeOrdinal = childNodeIndex / nodeFieldCount;
if (distances[childNodeOrdinal] !== noDistance)
continue;
edge.edgeIndex = edgeIndex;
if (filter && !filter(node, edge))
continue;
distances[childNodeOrdinal] = distance;
nodesToVisit[nodesToVisitLength++] = childNodeIndex;
}
}
if (nodesToVisitLength > nodeCount)
throw new Error("BFS failed. Nodes to visit (" + nodesToVisitLength + ") is more than nodes count (" + nodeCount + ")");
},
_buildAggregates: function(filter)
{
var aggregates = {};
var aggregatesByClassName = {};
var classIndexes = [];
var nodes = this.nodes;
var mapAndFlag = this.userObjectsMapAndFlag();
var flags = mapAndFlag ? mapAndFlag.map : null;
var flag = mapAndFlag ? mapAndFlag.flag : 0;
var nodesLength = nodes.length;
var nodeNativeType = this._nodeNativeType;
var nodeFieldCount = this._nodeFieldCount;
var selfSizeOffset = this._nodeSelfSizeOffset;
var nodeTypeOffset = this._nodeTypeOffset;
var node = this.rootNode();
var nodeDistances = this._nodeDistances;
for (var nodeIndex = 0; nodeIndex < nodesLength; nodeIndex += nodeFieldCount) {
var nodeOrdinal = nodeIndex / nodeFieldCount;
if (flags && !(flags[nodeOrdinal] & flag))
continue;
node.nodeIndex = nodeIndex;
if (filter && !filter(node))
continue;
var selfSize = nodes[nodeIndex + selfSizeOffset];
if (!selfSize && nodes[nodeIndex + nodeTypeOffset] !== nodeNativeType)
continue;
var classIndex = node.classIndex();
if (!(classIndex in aggregates)) {
var nodeType = node.type();
var nameMatters = nodeType === "object" || nodeType === "native";
var value = {
count: 1,
distance: nodeDistances[nodeOrdinal],
self: selfSize,
maxRet: 0,
type: nodeType,
name: nameMatters ? node.name() : null,
idxs: [nodeIndex]
};
aggregates[classIndex] = value;
classIndexes.push(classIndex);
aggregatesByClassName[node.className()] = value;
} else {
var clss = aggregates[classIndex];
clss.distance = Math.min(clss.distance, nodeDistances[nodeOrdinal]);
++clss.count;
clss.self += selfSize;
clss.idxs.push(nodeIndex);
}
}
// Shave off provisionally allocated space.
for (var i = 0, l = classIndexes.length; i < l; ++i) {
var classIndex = classIndexes[i];
aggregates[classIndex].idxs = aggregates[classIndex].idxs.slice();
}
return {aggregatesByClassName: aggregatesByClassName, aggregatesByClassIndex: aggregates};
},
_calculateClassesRetainedSize: function(aggregates, filter)
{
var rootNodeIndex = this._rootNodeIndex;
var node = this.createNode(rootNodeIndex);
var list = [rootNodeIndex];
var sizes = [-1];
var classes = [];
var seenClassNameIndexes = {};
var nodeFieldCount = this._nodeFieldCount;
var nodeTypeOffset = this._nodeTypeOffset;
var nodeNativeType = this._nodeNativeType;
var dominatedNodes = this._dominatedNodes;
var nodes = this.nodes;
var mapAndFlag = this.userObjectsMapAndFlag();
var flags = mapAndFlag ? mapAndFlag.map : null;
var flag = mapAndFlag ? mapAndFlag.flag : 0;
var firstDominatedNodeIndex = this._firstDominatedNodeIndex;
while (list.length) {
var nodeIndex = list.pop();
node.nodeIndex = nodeIndex;
var classIndex = node.classIndex();
var seen = !!seenClassNameIndexes[classIndex];
var nodeOrdinal = nodeIndex / nodeFieldCount;
var dominatedIndexFrom = firstDominatedNodeIndex[nodeOrdinal];
var dominatedIndexTo = firstDominatedNodeIndex[nodeOrdinal + 1];
if (!seen &&
(!flags || (flags[nodeOrdinal] & flag)) &&
(!filter || filter(node)) &&
(node.selfSize() || nodes[nodeIndex + nodeTypeOffset] === nodeNativeType)
) {
aggregates[classIndex].maxRet += node.retainedSize();
if (dominatedIndexFrom !== dominatedIndexTo) {
seenClassNameIndexes[classIndex] = true;
sizes.push(list.length);
classes.push(classIndex);
}
}
for (var i = dominatedIndexFrom; i < dominatedIndexTo; i++)
list.push(dominatedNodes[i]);
var l = list.length;
while (sizes[sizes.length - 1] === l) {
sizes.pop();
classIndex = classes.pop();
seenClassNameIndexes[classIndex] = false;
}
}
},
_sortAggregateIndexes: function(aggregates)
{
var nodeA = this.createNode();
var nodeB = this.createNode();
for (var clss in aggregates)
aggregates[clss].idxs.sort(
function(idxA, idxB) {
nodeA.nodeIndex = idxA;
nodeB.nodeIndex = idxB;
return nodeA.id() < nodeB.id() ? -1 : 1;
});
},
_buildPostOrderIndex: function()
{
var nodeFieldCount = this._nodeFieldCount;
var nodeCount = this.nodeCount;
var rootNodeOrdinal = this._rootNodeIndex / nodeFieldCount;
var edgeFieldsCount = this._edgeFieldsCount;
var edgeTypeOffset = this._edgeTypeOffset;
var edgeToNodeOffset = this._edgeToNodeOffset;
var edgeShortcutType = this._edgeShortcutType;
var edgeWeakType = this._edgeWeakType;
var firstEdgeIndexes = this._firstEdgeIndexes;
var containmentEdges = this.containmentEdges;
var mapAndFlag = this.userObjectsMapAndFlag();
var flags = mapAndFlag ? mapAndFlag.map : null;
var flag = mapAndFlag ? mapAndFlag.flag : 0;
var stackNodes = new Uint32Array(nodeCount);
var stackCurrentEdge = new Uint32Array(nodeCount);
var postOrderIndex2NodeOrdinal = new Uint32Array(nodeCount);
var nodeOrdinal2PostOrderIndex = new Uint32Array(nodeCount);
var visited = new Uint8Array(nodeCount);
var postOrderIndex = 0;
var stackTop = 0;
stackNodes[0] = rootNodeOrdinal;
stackCurrentEdge[0] = firstEdgeIndexes[rootNodeOrdinal];
visited[rootNodeOrdinal] = 1;
var iteration = 0;
while (true) {
++iteration;
while (stackTop >= 0) {
var nodeOrdinal = stackNodes[stackTop];
var edgeIndex = stackCurrentEdge[stackTop];
var edgesEnd = firstEdgeIndexes[nodeOrdinal + 1];
if (edgeIndex < edgesEnd) {
stackCurrentEdge[stackTop] += edgeFieldsCount;
var edgeType = containmentEdges[edgeIndex + edgeTypeOffset];
if (edgeType === edgeWeakType || edgeType === edgeShortcutType)
continue;
var childNodeIndex = containmentEdges[edgeIndex + edgeToNodeOffset];
var childNodeOrdinal = childNodeIndex / nodeFieldCount;
if (visited[childNodeOrdinal])
continue;
var nodeFlag = !flags || (flags[nodeOrdinal] & flag);
var childNodeFlag = !flags || (flags[childNodeOrdinal] & flag);
// We are skipping the edges from non-page-owned nodes to page-owned nodes.
// Otherwise the dominators for the objects that also were retained by debugger would be affected.
if (nodeOrdinal !== rootNodeOrdinal && childNodeFlag && !nodeFlag)
continue;
++stackTop;
stackNodes[stackTop] = childNodeOrdinal;
stackCurrentEdge[stackTop] = firstEdgeIndexes[childNodeOrdinal];
visited[childNodeOrdinal] = 1;
} else {
// Done with all the node children
nodeOrdinal2PostOrderIndex[nodeOrdinal] = postOrderIndex;
postOrderIndex2NodeOrdinal[postOrderIndex++] = nodeOrdinal;
--stackTop;
}
}
if (postOrderIndex === nodeCount || iteration > 1)
break;
var errors = new WebInspector.HeapSnapshotProblemReport("Heap snapshot: " + (nodeCount - postOrderIndex) + " nodes are unreachable from the root. Following nodes have only weak retainers:");
var dumpNode = this.rootNode();
// Remove root from the result (last node in the array) and put it at the bottom of the stack so that it is
// visited after all orphan nodes and their subgraphs.
--postOrderIndex;
stackTop = 0;
stackNodes[0] = rootNodeOrdinal;
stackCurrentEdge[0] = firstEdgeIndexes[rootNode