910 lines
27 KiB
JavaScript
910 lines
27 KiB
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.
|
||
|
*/
|
||
|
|
||
|
WebInspector.HeapSnapshotArraySlice = function(snapshot, arrayName, start, end)
|
||
|
{
|
||
|
// Note: we don't reference snapshot contents directly to avoid
|
||
|
// holding references to big chunks of data.
|
||
|
this._snapshot = snapshot;
|
||
|
this._arrayName = arrayName;
|
||
|
this._start = start;
|
||
|
this.length = end - start;
|
||
|
}
|
||
|
|
||
|
WebInspector.HeapSnapshotArraySlice.prototype = {
|
||
|
item: function(index)
|
||
|
{
|
||
|
return this._snapshot[this._arrayName][this._start + index];
|
||
|
}
|
||
|
}
|
||
|
|
||
|
WebInspector.HeapSnapshotEdge = function(snapshot, edges, edgeIndex)
|
||
|
{
|
||
|
this._snapshot = snapshot;
|
||
|
this._edges = edges;
|
||
|
this.edgeIndex = edgeIndex || 0;
|
||
|
}
|
||
|
|
||
|
WebInspector.HeapSnapshotEdge.prototype = {
|
||
|
clone: function()
|
||
|
{
|
||
|
return new WebInspector.HeapSnapshotEdge(this._snapshot, this._edges, this.edgeIndex);
|
||
|
},
|
||
|
|
||
|
get hasStringName()
|
||
|
{
|
||
|
if (!this.isShortcut)
|
||
|
return this._hasStringName;
|
||
|
return isNaN(parseInt(this._name, 10));
|
||
|
},
|
||
|
|
||
|
get isElement()
|
||
|
{
|
||
|
return this._type() === this._snapshot._edgeElementType;
|
||
|
},
|
||
|
|
||
|
get isHidden()
|
||
|
{
|
||
|
return this._type() === this._snapshot._edgeHiddenType;
|
||
|
},
|
||
|
|
||
|
get isInternal()
|
||
|
{
|
||
|
return this._type() === this._snapshot._edgeInternalType;
|
||
|
},
|
||
|
|
||
|
get isShortcut()
|
||
|
{
|
||
|
return this._type() === this._snapshot._edgeShortcutType;
|
||
|
},
|
||
|
|
||
|
get name()
|
||
|
{
|
||
|
if (!this.isShortcut)
|
||
|
return this._name;
|
||
|
var numName = parseInt(this._name, 10);
|
||
|
return isNaN(numName) ? this._name : numName;
|
||
|
},
|
||
|
|
||
|
get node()
|
||
|
{
|
||
|
return new WebInspector.HeapSnapshotNode(this._snapshot, this.nodeIndex);
|
||
|
},
|
||
|
|
||
|
get nodeIndex()
|
||
|
{
|
||
|
return this._edges.item(this.edgeIndex + this._snapshot._edgeToNodeOffset);
|
||
|
},
|
||
|
|
||
|
get rawEdges()
|
||
|
{
|
||
|
return this._edges;
|
||
|
},
|
||
|
|
||
|
toString: function()
|
||
|
{
|
||
|
switch (this.type) {
|
||
|
case "context": return "->" + this.name;
|
||
|
case "element": return "[" + this.name + "]";
|
||
|
case "property":
|
||
|
return this.name.indexOf(" ") === -1 ? "." + this.name : "[\"" + this.name + "\"]";
|
||
|
case "shortcut":
|
||
|
var name = this.name;
|
||
|
if (typeof name === "string")
|
||
|
return this.name.indexOf(" ") === -1 ? "." + this.name : "[\"" + this.name + "\"]";
|
||
|
else
|
||
|
return "[" + this.name + "]";
|
||
|
case "internal":
|
||
|
case "hidden":
|
||
|
return "{" + this.name + "}";
|
||
|
};
|
||
|
return "?" + this.name + "?";
|
||
|
},
|
||
|
|
||
|
get type()
|
||
|
{
|
||
|
return this._snapshot._edgeTypes[this._type()];
|
||
|
},
|
||
|
|
||
|
get _hasStringName()
|
||
|
{
|
||
|
return !this.isElement && !this.isHidden;
|
||
|
},
|
||
|
|
||
|
get _name()
|
||
|
{
|
||
|
return this._hasStringName ? this._snapshot._strings[this._nameOrIndex] : this._nameOrIndex;
|
||
|
},
|
||
|
|
||
|
get _nameOrIndex()
|
||
|
{
|
||
|
return this._edges.item(this.edgeIndex + this._snapshot._edgeNameOffset);
|
||
|
},
|
||
|
|
||
|
_type: function()
|
||
|
{
|
||
|
return this._edges.item(this.edgeIndex + this._snapshot._edgeTypeOffset);
|
||
|
}
|
||
|
};
|
||
|
|
||
|
WebInspector.HeapSnapshotEdgeIterator = function(edge)
|
||
|
{
|
||
|
this.edge = edge;
|
||
|
}
|
||
|
|
||
|
WebInspector.HeapSnapshotEdgeIterator.prototype = {
|
||
|
first: function()
|
||
|
{
|
||
|
this.edge.edgeIndex = 0;
|
||
|
},
|
||
|
|
||
|
hasNext: function()
|
||
|
{
|
||
|
return this.edge.edgeIndex < this.edge._edges.length;
|
||
|
},
|
||
|
|
||
|
get index()
|
||
|
{
|
||
|
return this.edge.edgeIndex;
|
||
|
},
|
||
|
|
||
|
set index(newIndex)
|
||
|
{
|
||
|
this.edge.edgeIndex = newIndex;
|
||
|
},
|
||
|
|
||
|
get item()
|
||
|
{
|
||
|
return this.edge;
|
||
|
},
|
||
|
|
||
|
next: function()
|
||
|
{
|
||
|
this.edge.edgeIndex += this.edge._snapshot._edgeFieldsCount;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
WebInspector.HeapSnapshotNode = function(snapshot, nodeIndex)
|
||
|
{
|
||
|
this._snapshot = snapshot;
|
||
|
this._firstNodeIndex = nodeIndex;
|
||
|
this.nodeIndex = nodeIndex;
|
||
|
}
|
||
|
|
||
|
WebInspector.HeapSnapshotNode.prototype = {
|
||
|
get className()
|
||
|
{
|
||
|
switch (this.type) {
|
||
|
case "hidden":
|
||
|
return WebInspector.UIString("(system)");
|
||
|
case "object":
|
||
|
return this.name;
|
||
|
case "code":
|
||
|
return WebInspector.UIString("(compiled code)");
|
||
|
default:
|
||
|
return "(" + this.type + ")";
|
||
|
}
|
||
|
},
|
||
|
|
||
|
dominatorIndex: function()
|
||
|
{
|
||
|
return this._nodes[this.nodeIndex + this._snapshot._dominatorOffset];
|
||
|
},
|
||
|
|
||
|
get edges()
|
||
|
{
|
||
|
return new WebInspector.HeapSnapshotEdgeIterator(new WebInspector.HeapSnapshotEdge(this._snapshot, this.rawEdges));
|
||
|
},
|
||
|
|
||
|
get edgesCount()
|
||
|
{
|
||
|
return this._nodes[this.nodeIndex + this._snapshot._edgesCountOffset];
|
||
|
},
|
||
|
|
||
|
get id()
|
||
|
{
|
||
|
return this._nodes[this.nodeIndex + this._snapshot._nodeIdOffset];
|
||
|
},
|
||
|
|
||
|
get instancesCount()
|
||
|
{
|
||
|
return this._nodes[this.nodeIndex + this._snapshot._nodeInstancesCountOffset];
|
||
|
},
|
||
|
|
||
|
get isHidden()
|
||
|
{
|
||
|
return this._type() === this._snapshot._nodeHiddenType;
|
||
|
},
|
||
|
|
||
|
get isRoot()
|
||
|
{
|
||
|
return this.nodeIndex === this._snapshot._rootNodeIndex;
|
||
|
},
|
||
|
|
||
|
get name()
|
||
|
{
|
||
|
return this._snapshot._strings[this._name()];
|
||
|
},
|
||
|
|
||
|
get rawEdges()
|
||
|
{
|
||
|
var firstEdgeIndex = this._firstEdgeIndex();
|
||
|
return new WebInspector.HeapSnapshotArraySlice(this._snapshot, "_nodes", firstEdgeIndex, firstEdgeIndex + this.edgesCount * this._snapshot._edgeFieldsCount);
|
||
|
},
|
||
|
|
||
|
get retainedSize()
|
||
|
{
|
||
|
return this._nodes[this.nodeIndex + this._snapshot._nodeRetainedSizeOffset];
|
||
|
},
|
||
|
|
||
|
get retainers()
|
||
|
{
|
||
|
return new WebInspector.HeapSnapshotEdgeIterator(new WebInspector.HeapSnapshotEdge(this._snapshot, this._snapshot.retainers(this)));
|
||
|
},
|
||
|
|
||
|
get selfSize()
|
||
|
{
|
||
|
return this._nodes[this.nodeIndex + this._snapshot._nodeSelfSizeOffset];
|
||
|
},
|
||
|
|
||
|
get type()
|
||
|
{
|
||
|
return this._snapshot._nodeTypes[this._type()];
|
||
|
},
|
||
|
|
||
|
_name: function()
|
||
|
{
|
||
|
return this._nodes[this.nodeIndex + this._snapshot._nodeNameOffset];
|
||
|
},
|
||
|
|
||
|
get _nodes()
|
||
|
{
|
||
|
return this._snapshot._nodes;
|
||
|
},
|
||
|
|
||
|
_firstEdgeIndex: function()
|
||
|
{
|
||
|
return this.nodeIndex + this._snapshot._firstEdgeOffset;
|
||
|
},
|
||
|
|
||
|
get _nextNodeIndex()
|
||
|
{
|
||
|
return this._firstEdgeIndex() + this.edgesCount * this._snapshot._edgeFieldsCount;
|
||
|
},
|
||
|
|
||
|
_type: function()
|
||
|
{
|
||
|
return this._nodes[this.nodeIndex + this._snapshot._nodeTypeOffset];
|
||
|
}
|
||
|
};
|
||
|
|
||
|
WebInspector.HeapSnapshotNodeIterator = function(node)
|
||
|
{
|
||
|
this.node = node;
|
||
|
}
|
||
|
|
||
|
WebInspector.HeapSnapshotNodeIterator.prototype = {
|
||
|
first: function()
|
||
|
{
|
||
|
this.node.nodeIndex = this.node._firstNodeIndex;
|
||
|
},
|
||
|
|
||
|
hasNext: function()
|
||
|
{
|
||
|
return this.node.nodeIndex < this.node._nodes.length;
|
||
|
},
|
||
|
|
||
|
get index()
|
||
|
{
|
||
|
return this.node.nodeIndex;
|
||
|
},
|
||
|
|
||
|
set index(newIndex)
|
||
|
{
|
||
|
this.node.nodeIndex = newIndex;
|
||
|
},
|
||
|
|
||
|
get item()
|
||
|
{
|
||
|
return this.node;
|
||
|
},
|
||
|
|
||
|
next: function()
|
||
|
{
|
||
|
this.node.nodeIndex = this.node._nextNodeIndex;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
WebInspector.HeapSnapshot = function(profile)
|
||
|
{
|
||
|
this._nodes = profile.nodes;
|
||
|
this._strings = profile.strings;
|
||
|
|
||
|
this._init();
|
||
|
}
|
||
|
|
||
|
WebInspector.HeapSnapshot.prototype = {
|
||
|
_init: function()
|
||
|
{
|
||
|
this._metaNodeIndex = 0;
|
||
|
this._rootNodeIndex = 1;
|
||
|
var meta = this._nodes[this._metaNodeIndex];
|
||
|
this._nodeTypeOffset = meta.fields.indexOf("type");
|
||
|
this._nodeNameOffset = meta.fields.indexOf("name");
|
||
|
this._nodeIdOffset = meta.fields.indexOf("id");
|
||
|
this._nodeInstancesCountOffset = this._nodeIdOffset;
|
||
|
this._nodeSelfSizeOffset = meta.fields.indexOf("self_size");
|
||
|
this._nodeRetainedSizeOffset = meta.fields.indexOf("retained_size");
|
||
|
this._dominatorOffset = meta.fields.indexOf("dominator");
|
||
|
this._edgesCountOffset = meta.fields.indexOf("children_count");
|
||
|
this._firstEdgeOffset = meta.fields.indexOf("children");
|
||
|
this._nodeTypes = meta.types[this._nodeTypeOffset];
|
||
|
this._nodeHiddenType = this._nodeTypes.indexOf("hidden");
|
||
|
var edgesMeta = meta.types[this._firstEdgeOffset];
|
||
|
this._edgeFieldsCount = edgesMeta.fields.length;
|
||
|
this._edgeTypeOffset = edgesMeta.fields.indexOf("type");
|
||
|
this._edgeNameOffset = edgesMeta.fields.indexOf("name_or_index");
|
||
|
this._edgeToNodeOffset = edgesMeta.fields.indexOf("to_node");
|
||
|
this._edgeTypes = edgesMeta.types[this._edgeTypeOffset];
|
||
|
this._edgeElementType = this._edgeTypes.indexOf("element");
|
||
|
this._edgeHiddenType = this._edgeTypes.indexOf("hidden");
|
||
|
this._edgeInternalType = this._edgeTypes.indexOf("internal");
|
||
|
this._edgeShortcutType = this._edgeTypes.indexOf("shortcut");
|
||
|
},
|
||
|
|
||
|
dispose: function()
|
||
|
{
|
||
|
delete this._nodes;
|
||
|
delete this._strings;
|
||
|
if (this._idsMap)
|
||
|
delete this._idsMap;
|
||
|
if (this._retainers) {
|
||
|
delete this._retainers;
|
||
|
delete this._nodesToRetainers;
|
||
|
}
|
||
|
if (this._aggregates) {
|
||
|
delete this._aggregates;
|
||
|
this._aggregatesWithIndexes = false;
|
||
|
}
|
||
|
},
|
||
|
|
||
|
get allNodes()
|
||
|
{
|
||
|
return new WebInspector.HeapSnapshotNodeIterator(this.rootNode);
|
||
|
},
|
||
|
|
||
|
get nodesCount()
|
||
|
{
|
||
|
if (this._nodesCount)
|
||
|
return this._nodesCount;
|
||
|
|
||
|
this._nodesCount = 0;
|
||
|
for (var iter = this.allNodes; iter.hasNext(); iter.next())
|
||
|
++this._nodesCount;
|
||
|
return this._nodesCount;
|
||
|
},
|
||
|
|
||
|
restore: function(profile)
|
||
|
{
|
||
|
this._nodes = profile.nodes;
|
||
|
this._strings = profile.strings;
|
||
|
},
|
||
|
|
||
|
get rootNode()
|
||
|
{
|
||
|
return new WebInspector.HeapSnapshotNode(this, this._rootNodeIndex);
|
||
|
},
|
||
|
|
||
|
get totalSize()
|
||
|
{
|
||
|
return this.rootNode.retainedSize;
|
||
|
},
|
||
|
|
||
|
get idsMap()
|
||
|
{
|
||
|
if (this._idsMap)
|
||
|
return this._idsMap;
|
||
|
|
||
|
this._idsMap = [];
|
||
|
for (var iter = this.allNodes; iter.hasNext(); iter.next()) {
|
||
|
this._idsMap[iter.node.id] = true;
|
||
|
}
|
||
|
return this._idsMap;
|
||
|
},
|
||
|
|
||
|
retainers: function(node)
|
||
|
{
|
||
|
if (!this._retainers)
|
||
|
this._buildRetainers();
|
||
|
|
||
|
var retIndexFrom = this._nodesToRetainers[node.nodeIndex];
|
||
|
var retIndexTo = this._nodesToRetainers[node._nextNodeIndex];
|
||
|
return new WebInspector.HeapSnapshotArraySlice(this, "_retainers", retIndexFrom, retIndexTo);
|
||
|
},
|
||
|
|
||
|
aggregates: function(withNodeIndexes)
|
||
|
{
|
||
|
if (!this._aggregates)
|
||
|
this._buildAggregates();
|
||
|
if (withNodeIndexes && !this._aggregatesWithIndexes)
|
||
|
this._buildAggregatesIndexes();
|
||
|
return this._aggregates;
|
||
|
},
|
||
|
|
||
|
_buildRetainers: function()
|
||
|
{
|
||
|
this._nodesToRetainers = [];
|
||
|
for (var nodesIter = this.allNodes; nodesIter.hasNext(); nodesIter.next()) {
|
||
|
var node = nodesIter.node;
|
||
|
if (!(node.nodeIndex in this._nodesToRetainers))
|
||
|
this._nodesToRetainers[node.nodeIndex] = 0;
|
||
|
for (var edgesIter = node.edges; edgesIter.hasNext(); edgesIter.next()) {
|
||
|
var edge = edgesIter.edge;
|
||
|
var nodeIndex = edge.nodeIndex;
|
||
|
if (!(nodeIndex in this._nodesToRetainers))
|
||
|
this._nodesToRetainers[nodeIndex] = 0;
|
||
|
this._nodesToRetainers[nodeIndex] += this._edgeFieldsCount;
|
||
|
}
|
||
|
}
|
||
|
nodesIter = this.allNodes;
|
||
|
var node = nodesIter.node;
|
||
|
var prevIndex = this._nodesToRetainers[node.nodeIndex] = 0;
|
||
|
var prevRetsCount = this._nodesToRetainers[node.nodeIndex];
|
||
|
nodesIter.next();
|
||
|
for (; nodesIter.hasNext(); nodesIter.next()) {
|
||
|
node = nodesIter.node;
|
||
|
var savedRefsCount = this._nodesToRetainers[node.nodeIndex];
|
||
|
this._nodesToRetainers[node.nodeIndex] = prevIndex + prevRetsCount;
|
||
|
prevIndex = this._nodesToRetainers[node.nodeIndex];
|
||
|
prevRetsCount = savedRefsCount;
|
||
|
}
|
||
|
this._retainers = new Array(prevIndex + prevRetsCount);
|
||
|
this._nodesToRetainers[this._nodes.length] = this._retainers.length;
|
||
|
for (nodesIter = this.allNodes; nodesIter.hasNext(); nodesIter.next()) {
|
||
|
node = nodesIter.node;
|
||
|
var retsCount = this._nodesToRetainers[node._nextNodeIndex] - this._nodesToRetainers[node.nodeIndex];
|
||
|
if (retsCount > 0) {
|
||
|
this._retainers[this._nodesToRetainers[node.nodeIndex]] = retsCount;
|
||
|
}
|
||
|
}
|
||
|
for (nodesIter = this.allNodes; nodesIter.hasNext(); nodesIter.next()) {
|
||
|
node = nodesIter.node;
|
||
|
for (var edgesIter = node.edges; edgesIter.hasNext(); edgesIter.next()) {
|
||
|
var edge = edgesIter.edge;
|
||
|
var nodeIndex = edge.nodeIndex;
|
||
|
var retIndex = this._nodesToRetainers[nodeIndex];
|
||
|
this._retainers[retIndex] -= this._edgeFieldsCount;
|
||
|
var idx = retIndex + this._retainers[retIndex];
|
||
|
this._retainers[idx + this._edgeTypeOffset] = edge._type();
|
||
|
this._retainers[idx + this._edgeNameOffset] = edge._nameOrIndex;
|
||
|
this._retainers[idx + this._edgeToNodeOffset] = node.nodeIndex;
|
||
|
}
|
||
|
}
|
||
|
},
|
||
|
|
||
|
_buildAggregates: function()
|
||
|
{
|
||
|
this._aggregates = {};
|
||
|
for (var iter = this.allNodes; iter.hasNext(); iter.next()) {
|
||
|
var node = iter.node;
|
||
|
var className = node.className;
|
||
|
var nameMatters = node.type === "object";
|
||
|
if (node.selfSize === 0)
|
||
|
continue;
|
||
|
if (!(className in this._aggregates))
|
||
|
this._aggregates[className] = { count: 0, self: 0, maxRet: 0, type: node.type, name: nameMatters ? node.name : null, idxs: [] };
|
||
|
var clss = this._aggregates[className];
|
||
|
++clss.count;
|
||
|
clss.self += node.selfSize;
|
||
|
if (node.retainedSize > clss.maxRet)
|
||
|
clss.maxRet = node.retainedSize;
|
||
|
}
|
||
|
},
|
||
|
|
||
|
_buildAggregatesIndexes: function()
|
||
|
{
|
||
|
for (var iter = this.allNodes; iter.hasNext(); iter.next()) {
|
||
|
var node = iter.node;
|
||
|
var className = node.className;
|
||
|
var clss = this._aggregates[className];
|
||
|
if (clss)
|
||
|
clss.idxs.push(node.nodeIndex);
|
||
|
}
|
||
|
|
||
|
var nodeA = new WebInspector.HeapSnapshotNode(this);
|
||
|
var nodeB = new WebInspector.HeapSnapshotNode(this);
|
||
|
for (var clss in this._aggregates)
|
||
|
this._aggregates[clss].idxs.sort(
|
||
|
function(idxA, idxB) {
|
||
|
nodeA.nodeIndex = idxA;
|
||
|
nodeB.nodeIndex = idxB;
|
||
|
return nodeA.id < nodeB.id ? -1 : 1;
|
||
|
});
|
||
|
|
||
|
this._aggregatesWithIndexes = true;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
WebInspector.HeapSnapshotFilteredOrderedIterator = function(snapshot, iterator, filter)
|
||
|
{
|
||
|
this._snapshot = snapshot;
|
||
|
this._filter = filter;
|
||
|
this._iterator = iterator;
|
||
|
this._iterationOrder = null;
|
||
|
this._position = 0;
|
||
|
this._lastComparator = null;
|
||
|
}
|
||
|
|
||
|
WebInspector.HeapSnapshotFilteredOrderedIterator.prototype = {
|
||
|
_createIterationOrder: function()
|
||
|
{
|
||
|
this._iterationOrder = [];
|
||
|
var iterator = this._iterator;
|
||
|
if (!this._filter) {
|
||
|
for (iterator.first(); iterator.hasNext(); iterator.next())
|
||
|
this._iterationOrder.push(iterator.index);
|
||
|
} else {
|
||
|
for (iterator.first(); iterator.hasNext(); iterator.next()) {
|
||
|
if (this._filter(iterator.item))
|
||
|
this._iterationOrder.push(iterator.index);
|
||
|
}
|
||
|
}
|
||
|
},
|
||
|
|
||
|
first: function()
|
||
|
{
|
||
|
this._position = 0;
|
||
|
},
|
||
|
|
||
|
hasNext: function()
|
||
|
{
|
||
|
return this._position < this._iterationOrder.length;
|
||
|
},
|
||
|
|
||
|
get isEmpty()
|
||
|
{
|
||
|
if (this._iterationOrder)
|
||
|
return !this._iterationOrder.length;
|
||
|
var iterator = this._iterator;
|
||
|
if (!this._filter) {
|
||
|
iterator.first();
|
||
|
return !iterator.hasNext();
|
||
|
}
|
||
|
for (iterator.first(); iterator.hasNext(); iterator.next())
|
||
|
if (this._filter(iterator.item)) return false;
|
||
|
return true;
|
||
|
},
|
||
|
|
||
|
get item()
|
||
|
{
|
||
|
this._iterator.index = this._iterationOrder[this._position];
|
||
|
return this._iterator.item;
|
||
|
},
|
||
|
|
||
|
get lastComparator()
|
||
|
{
|
||
|
return this._lastComparator;
|
||
|
},
|
||
|
|
||
|
get length()
|
||
|
{
|
||
|
if (!this._iterationOrder)
|
||
|
this._createIterationOrder();
|
||
|
return this._iterationOrder.length;
|
||
|
},
|
||
|
|
||
|
next: function()
|
||
|
{
|
||
|
++this._position;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
WebInspector.HeapSnapshotFilteredOrderedIterator.prototype.createComparator = function(fieldNames)
|
||
|
{
|
||
|
return {fieldName1:fieldNames[0], ascending1:fieldNames[1], fieldName2:fieldNames[2], ascending2:fieldNames[3]};
|
||
|
}
|
||
|
|
||
|
WebInspector.HeapSnapshotEdgesProvider = function(snapshot, rawEdges, filter)
|
||
|
{
|
||
|
WebInspector.HeapSnapshotFilteredOrderedIterator.call(this, snapshot, new WebInspector.HeapSnapshotEdgeIterator(new WebInspector.HeapSnapshotEdge(snapshot, rawEdges)), filter);
|
||
|
}
|
||
|
|
||
|
WebInspector.HeapSnapshotEdgesProvider.prototype = {
|
||
|
sort: function(comparator)
|
||
|
{
|
||
|
if (this._lastComparator === comparator)
|
||
|
return false;
|
||
|
this._lastComparator = comparator;
|
||
|
var fieldName1 = comparator.fieldName1;
|
||
|
var fieldName2 = comparator.fieldName2;
|
||
|
var ascending1 = comparator.ascending1;
|
||
|
var ascending2 = comparator.ascending2;
|
||
|
|
||
|
var edgeA = this._iterator.item.clone();
|
||
|
var edgeB = edgeA.clone();
|
||
|
var nodeA = new WebInspector.HeapSnapshotNode(this._snapshot);
|
||
|
var nodeB = new WebInspector.HeapSnapshotNode(this._snapshot);
|
||
|
|
||
|
function sortByEdgeFieldName(ascending, indexA, indexB)
|
||
|
{
|
||
|
edgeA.edgeIndex = indexA;
|
||
|
edgeB.edgeIndex = indexB;
|
||
|
if (edgeB.name === "__proto__") return -1;
|
||
|
if (edgeA.name === "__proto__") return 1;
|
||
|
var result =
|
||
|
edgeA.hasStringName === edgeB.hasStringName ?
|
||
|
(edgeA.name < edgeB.name ? -1 : (edgeA.name > edgeB.name ? 1 : 0)) :
|
||
|
(edgeA.hasStringName ? -1 : 1);
|
||
|
return ascending ? result : -result;
|
||
|
}
|
||
|
|
||
|
function sortByNodeField(fieldName, ascending, indexA, indexB)
|
||
|
{
|
||
|
edgeA.edgeIndex = indexA;
|
||
|
edgeB.edgeIndex = indexB;
|
||
|
nodeA.nodeIndex = edgeA.nodeIndex;
|
||
|
nodeB.nodeIndex = edgeB.nodeIndex;
|
||
|
var valueA = nodeA[fieldName];
|
||
|
var valueB = nodeB[fieldName];
|
||
|
var result = valueA < valueB ? -1 : (valueA > valueB ? 1 : 0);
|
||
|
return ascending ? result : -result;
|
||
|
}
|
||
|
|
||
|
if (!this._iterationOrder)
|
||
|
this._createIterationOrder();
|
||
|
|
||
|
function sortByEdgeAndNode(indexA, indexB) {
|
||
|
var result = sortByEdgeFieldName(ascending1, indexA, indexB);
|
||
|
if (result === 0)
|
||
|
result = sortByNodeField(fieldName2, ascending2, indexA, indexB);
|
||
|
return result;
|
||
|
}
|
||
|
|
||
|
function sortByNodeAndEdge(indexA, indexB) {
|
||
|
var result = sortByNodeField(fieldName1, ascending1, indexA, indexB);
|
||
|
if (result === 0)
|
||
|
result = sortByEdgeFieldName(ascending2, indexA, indexB);
|
||
|
return result;
|
||
|
}
|
||
|
|
||
|
function sortByNodeAndNode(indexA, indexB) {
|
||
|
var result = sortByNodeField(fieldName1, ascending1, indexA, indexB);
|
||
|
if (result === 0)
|
||
|
result = sortByNodeField(fieldName2, ascending2, indexA, indexB);
|
||
|
return result;
|
||
|
}
|
||
|
|
||
|
if (fieldName1 === "!edgeName")
|
||
|
this._iterationOrder.sort(sortByEdgeAndNode);
|
||
|
else if (fieldName2 === "!edgeName")
|
||
|
this._iterationOrder.sort(sortByNodeAndEdge);
|
||
|
else
|
||
|
this._iterationOrder.sort(sortByNodeAndNode);
|
||
|
return true;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
WebInspector.HeapSnapshotEdgesProvider.prototype.__proto__ = WebInspector.HeapSnapshotFilteredOrderedIterator.prototype;
|
||
|
|
||
|
WebInspector.HeapSnapshotNodesProvider = function(snapshot, nodes, filter)
|
||
|
{
|
||
|
WebInspector.HeapSnapshotFilteredOrderedIterator.call(this, snapshot, nodes, filter);
|
||
|
}
|
||
|
|
||
|
WebInspector.HeapSnapshotNodesProvider.prototype = {
|
||
|
sort: function(comparator)
|
||
|
{
|
||
|
if (this._lastComparator === comparator)
|
||
|
return false;
|
||
|
this._lastComparator = comparator;
|
||
|
var fieldName1 = comparator.fieldName1;
|
||
|
var fieldName2 = comparator.fieldName2;
|
||
|
var ascending1 = comparator.ascending1;
|
||
|
var ascending2 = comparator.ascending2;
|
||
|
|
||
|
var nodeA = new WebInspector.HeapSnapshotNode(this._snapshot);
|
||
|
var nodeB = new WebInspector.HeapSnapshotNode(this._snapshot);
|
||
|
|
||
|
function sortByNodeField(fieldName, ascending, indexA, indexB)
|
||
|
{
|
||
|
nodeA.nodeIndex = indexA;
|
||
|
nodeB.nodeIndex = indexB;
|
||
|
var valueA = nodeA[fieldName];
|
||
|
var valueB = nodeB[fieldName];
|
||
|
var result = valueA < valueB ? -1 : (valueA > valueB ? 1 : 0);
|
||
|
return ascending ? result : -result;
|
||
|
}
|
||
|
|
||
|
if (!this._iterationOrder)
|
||
|
this._createIterationOrder();
|
||
|
|
||
|
function sortByComparator(indexA, indexB) {
|
||
|
var result = sortByNodeField(fieldName1, ascending1, indexA, indexB);
|
||
|
if (result === 0)
|
||
|
result = sortByNodeField(fieldName2, ascending2, indexA, indexB);
|
||
|
return result;
|
||
|
}
|
||
|
|
||
|
this._iterationOrder.sort(sortByComparator);
|
||
|
return true;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
WebInspector.HeapSnapshotNodesProvider.prototype.__proto__ = WebInspector.HeapSnapshotFilteredOrderedIterator.prototype;
|
||
|
|
||
|
WebInspector.HeapSnapshotPathFinder = function(snapshot, targetNodeIndex)
|
||
|
{
|
||
|
this._snapshot = snapshot;
|
||
|
this._maxLength = 1;
|
||
|
this._lengthLimit = 15;
|
||
|
this._targetNodeIndex = targetNodeIndex;
|
||
|
this._currentPath = null;
|
||
|
this._skipHidden = !WebInspector.DetailedHeapshotView.prototype.showHiddenData;
|
||
|
this._rootChildren = this._fillRootChildren();
|
||
|
}
|
||
|
|
||
|
WebInspector.HeapSnapshotPathFinder.prototype = {
|
||
|
findNext: function()
|
||
|
{
|
||
|
for (var i = 0; i < 100000; ++i) {
|
||
|
if (!this._buildNextPath()) {
|
||
|
if (++this._maxLength >= this._lengthLimit)
|
||
|
return null;
|
||
|
this._currentPath = null;
|
||
|
if (!this._buildNextPath())
|
||
|
return null;
|
||
|
}
|
||
|
if (this._isPathFound())
|
||
|
return {path:this._pathToString(this._currentPath), len:this._currentPath.length};
|
||
|
}
|
||
|
|
||
|
return false;
|
||
|
},
|
||
|
|
||
|
_fillRootChildren: function()
|
||
|
{
|
||
|
var result = [];
|
||
|
for (var iter = this._snapshot.rootNode.edges; iter.hasNext(); iter.next())
|
||
|
result[iter.edge.nodeIndex] = true;
|
||
|
return result;
|
||
|
},
|
||
|
|
||
|
_appendToCurrentPath: function(iter)
|
||
|
{
|
||
|
this._currentPath._cache[this._lastEdge.nodeIndex] = true;
|
||
|
this._currentPath.push(iter);
|
||
|
},
|
||
|
|
||
|
_removeLastFromCurrentPath: function()
|
||
|
{
|
||
|
this._currentPath.pop();
|
||
|
delete this._currentPath._cache[this._lastEdge.nodeIndex];
|
||
|
},
|
||
|
|
||
|
_hasInPath: function(nodeIndex)
|
||
|
{
|
||
|
return this._targetNodeIndex === nodeIndex
|
||
|
|| !!this._currentPath._cache[nodeIndex];
|
||
|
},
|
||
|
|
||
|
_isPathFound: function()
|
||
|
{
|
||
|
return this._currentPath.length === this._maxLength
|
||
|
&& this._lastEdge.nodeIndex in this._rootChildren;
|
||
|
},
|
||
|
|
||
|
get _lastEdgeIter()
|
||
|
{
|
||
|
return this._currentPath[this._currentPath.length - 1];
|
||
|
},
|
||
|
|
||
|
get _lastEdge()
|
||
|
{
|
||
|
return this._lastEdgeIter.edge;
|
||
|
},
|
||
|
|
||
|
_skipEdge: function(edge)
|
||
|
{
|
||
|
return (this._skipHidden && (edge.isHidden || edge.node.isHidden))
|
||
|
|| this._hasInPath(edge.nodeIndex);
|
||
|
},
|
||
|
|
||
|
_nextEdgeIter: function()
|
||
|
{
|
||
|
var iter = this._lastEdgeIter;
|
||
|
while (this._skipEdge(iter.edge) && iter.hasNext())
|
||
|
iter.next();
|
||
|
return iter;
|
||
|
},
|
||
|
|
||
|
_buildNextPath: function()
|
||
|
{
|
||
|
if (this._currentPath !== null) {
|
||
|
var iter = this._lastEdgeIter;
|
||
|
while (true) {
|
||
|
iter.next();
|
||
|
if (iter.hasNext())
|
||
|
return true;
|
||
|
while (true) {
|
||
|
if (this._currentPath.length > 1) {
|
||
|
this._removeLastFromCurrentPath();
|
||
|
iter = this._lastEdgeIter;
|
||
|
iter.next();
|
||
|
iter = this._nextEdgeIter();
|
||
|
if (iter.hasNext()) {
|
||
|
while (this._currentPath.length < this._maxLength) {
|
||
|
iter = this._nextEdgeIter();
|
||
|
if (iter.hasNext())
|
||
|
this._appendToCurrentPath(iter.edge.node.retainers);
|
||
|
else
|
||
|
return true;
|
||
|
}
|
||
|
return true;
|
||
|
}
|
||
|
} else
|
||
|
return false;
|
||
|
}
|
||
|
}
|
||
|
} else {
|
||
|
var node = new WebInspector.HeapSnapshotNode(this._snapshot, this._targetNodeIndex);
|
||
|
this._currentPath = [node.retainers];
|
||
|
this._currentPath._cache = {};
|
||
|
while (this._currentPath.length < this._maxLength) {
|
||
|
var iter = this._nextEdgeIter();
|
||
|
if (iter.hasNext())
|
||
|
this._appendToCurrentPath(iter.edge.node.retainers);
|
||
|
else
|
||
|
break;
|
||
|
}
|
||
|
return true;
|
||
|
}
|
||
|
},
|
||
|
|
||
|
_nodeToString: function(node)
|
||
|
{
|
||
|
if (node.id === 1)
|
||
|
return node.name;
|
||
|
else
|
||
|
return node.name + "@" + node.id;
|
||
|
},
|
||
|
|
||
|
_pathToString: function(path)
|
||
|
{
|
||
|
if (!path)
|
||
|
return "";
|
||
|
var sPath = [];
|
||
|
for (var j = 0; j < path.length; ++j)
|
||
|
sPath.push(path[j].edge.toString());
|
||
|
sPath.push(this._nodeToString(path[path.length - 1].edge.node));
|
||
|
sPath.reverse();
|
||
|
return sPath.join("");
|
||
|
}
|
||
|
};
|