Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(664)

Unified Diff: third_party/WebKit/Source/devtools/front_end/heap_snapshot_worker/HeapSnapshot.js

Issue 2466123002: DevTools: reformat front-end code to match chromium style. (Closed)
Patch Set: all done Created 4 years, 1 month ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
Index: third_party/WebKit/Source/devtools/front_end/heap_snapshot_worker/HeapSnapshot.js
diff --git a/third_party/WebKit/Source/devtools/front_end/heap_snapshot_worker/HeapSnapshot.js b/third_party/WebKit/Source/devtools/front_end/heap_snapshot_worker/HeapSnapshot.js
index 3f1cf1c710ac1ee9afeeac93e45dee5df1d0c5a0..8e4b87b2c486d1d0866cf44801fe077df9a49bfa 100644
--- a/third_party/WebKit/Source/devtools/front_end/heap_snapshot_worker/HeapSnapshot.js
+++ b/third_party/WebKit/Source/devtools/front_end/heap_snapshot_worker/HeapSnapshot.js
@@ -27,868 +27,807 @@
* (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 = function() {};
WebInspector.HeapSnapshotItem.prototype = {
- /**
- * @return {number}
- */
- itemIndex: function() { },
-
- /**
- * @return {!Object}
- */
- serialize: function() { }
+ /**
+ * @return {number}
+ */
+ itemIndex: function() {},
+
+ /**
+ * @return {!Object}
+ */
+ serialize: function() {}
};
/**
- * @constructor
* @implements {WebInspector.HeapSnapshotItem}
- * @param {!WebInspector.HeapSnapshot} snapshot
- * @param {number=} edgeIndex
+ * @unrestricted
*/
-WebInspector.HeapSnapshotEdge = function(snapshot, edgeIndex)
-{
+WebInspector.HeapSnapshotEdge = class {
+ /**
+ * @param {!WebInspector.HeapSnapshot} snapshot
+ * @param {number=} edgeIndex
+ */
+ constructor(snapshot, edgeIndex) {
this._snapshot = snapshot;
this._edges = snapshot.containmentEdges;
this.edgeIndex = edgeIndex || 0;
+ }
+
+ /**
+ * @return {!WebInspector.HeapSnapshotEdge}
+ */
+ clone() {
+ return new WebInspector.HeapSnapshotEdge(this._snapshot, this.edgeIndex);
+ }
+
+ /**
+ * @return {boolean}
+ */
+ hasStringName() {
+ throw new Error('Not implemented');
+ }
+
+ /**
+ * @return {string}
+ */
+ name() {
+ throw new Error('Not implemented');
+ }
+
+ /**
+ * @return {!WebInspector.HeapSnapshotNode}
+ */
+ node() {
+ return this._snapshot.createNode(this.nodeIndex());
+ }
+
+ /**
+ * @return {number}
+ */
+ nodeIndex() {
+ return this._edges[this.edgeIndex + this._snapshot._edgeToNodeOffset];
+ }
+
+ /**
+ * @override
+ * @return {string}
+ */
+ toString() {
+ return 'HeapSnapshotEdge: ' + this.name();
+ }
+
+ /**
+ * @return {string}
+ */
+ type() {
+ return this._snapshot._edgeTypes[this.rawType()];
+ }
+
+ /**
+ * @override
+ * @return {number}
+ */
+ itemIndex() {
+ return this.edgeIndex;
+ }
+
+ /**
+ * @override
+ * @return {!WebInspector.HeapSnapshotCommon.Edge}
+ */
+ serialize() {
+ return new WebInspector.HeapSnapshotCommon.Edge(this.name(), this.node().serialize(), this.type(), this.edgeIndex);
+ }
+
+ /**
+ * @protected
+ * @return {number}
+ */
+ rawType() {
+ return this._edges[this.edgeIndex + this._snapshot._edgeTypeOffset];
+ }
};
-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 = function() {};
WebInspector.HeapSnapshotItemIterator.prototype = {
- /**
- * @return {boolean}
- */
- hasNext: function() { },
+ /**
+ * @return {boolean}
+ */
+ hasNext: function() {},
- /**
- * @return {!WebInspector.HeapSnapshotItem}
- */
- item: function() { },
+ /**
+ * @return {!WebInspector.HeapSnapshotItem}
+ */
+ item: function() {},
- next: function() { }
+ next: function() {}
};
-
/**
* @interface
*/
-WebInspector.HeapSnapshotItemIndexProvider = function() { };
+WebInspector.HeapSnapshotItemIndexProvider = function() {};
WebInspector.HeapSnapshotItemIndexProvider.prototype = {
- /**
- * @param {number} newIndex
- * @return {!WebInspector.HeapSnapshotItem}
- */
- itemForIndex: function(newIndex) { },
+ /**
+ * @param {number} newIndex
+ * @return {!WebInspector.HeapSnapshotItem}
+ */
+ itemForIndex: function(newIndex) {},
};
/**
- * @constructor
* @implements {WebInspector.HeapSnapshotItemIndexProvider}
- * @param {!WebInspector.HeapSnapshot} snapshot
+ * @unrestricted
*/
-WebInspector.HeapSnapshotNodeIndexProvider = function(snapshot)
-{
+WebInspector.HeapSnapshotNodeIndexProvider = class {
+ /**
+ * @param {!WebInspector.HeapSnapshot} snapshot
+ */
+ constructor(snapshot) {
this._node = snapshot.createNode();
+ }
+
+ /**
+ * @override
+ * @param {number} index
+ * @return {!WebInspector.HeapSnapshotNode}
+ */
+ itemForIndex(index) {
+ this._node.nodeIndex = index;
+ return this._node;
+ }
};
-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
+ * @unrestricted
*/
-WebInspector.HeapSnapshotEdgeIndexProvider = function(snapshot)
-{
+WebInspector.HeapSnapshotEdgeIndexProvider = class {
+ /**
+ * @param {!WebInspector.HeapSnapshot} snapshot
+ */
+ constructor(snapshot) {
this._edge = snapshot.createEdge(0);
+ }
+
+ /**
+ * @override
+ * @param {number} index
+ * @return {!WebInspector.HeapSnapshotEdge}
+ */
+ itemForIndex(index) {
+ this._edge.edgeIndex = index;
+ return this._edge;
+ }
};
-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
+ * @unrestricted
*/
-WebInspector.HeapSnapshotRetainerEdgeIndexProvider = function(snapshot)
-{
+WebInspector.HeapSnapshotRetainerEdgeIndexProvider = class {
+ /**
+ * @param {!WebInspector.HeapSnapshot} snapshot
+ */
+ constructor(snapshot) {
this._retainerEdge = snapshot.createRetainingEdge(0);
+ }
+
+ /**
+ * @override
+ * @param {number} index
+ * @return {!WebInspector.HeapSnapshotRetainerEdge}
+ */
+ itemForIndex(index) {
+ this._retainerEdge.setRetainerIndex(index);
+ return this._retainerEdge;
+ }
};
-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
+ * @unrestricted
*/
-WebInspector.HeapSnapshotEdgeIterator = function(node)
-{
+WebInspector.HeapSnapshotEdgeIterator = class {
+ /**
+ * @param {!WebInspector.HeapSnapshotNode} node
+ */
+ constructor(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;
- }
+ }
+
+ /**
+ * @override
+ * @return {boolean}
+ */
+ hasNext() {
+ return this.edge.edgeIndex < this._sourceNode.edgeIndexesEnd();
+ }
+
+ /**
+ * @override
+ * @return {!WebInspector.HeapSnapshotEdge}
+ */
+ item() {
+ return this.edge;
+ }
+
+ /**
+ * @override
+ */
+ next() {
+ this.edge.edgeIndex += this.edge._snapshot._edgeFieldsCount;
+ }
};
/**
- * @constructor
* @implements {WebInspector.HeapSnapshotItem}
- * @param {!WebInspector.HeapSnapshot} snapshot
- * @param {number} retainerIndex
+ * @unrestricted
*/
-WebInspector.HeapSnapshotRetainerEdge = function(snapshot, retainerIndex)
-{
+WebInspector.HeapSnapshotRetainerEdge = class {
+ /**
+ * @param {!WebInspector.HeapSnapshot} snapshot
+ * @param {number} retainerIndex
+ */
+ constructor(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();
- }
+ }
+
+ /**
+ * @return {!WebInspector.HeapSnapshotRetainerEdge}
+ */
+ clone() {
+ return new WebInspector.HeapSnapshotRetainerEdge(this._snapshot, this.retainerIndex());
+ }
+
+ /**
+ * @return {boolean}
+ */
+ hasStringName() {
+ return this._edge().hasStringName();
+ }
+
+ /**
+ * @return {string}
+ */
+ name() {
+ return this._edge().name();
+ }
+
+ /**
+ * @return {!WebInspector.HeapSnapshotNode}
+ */
+ node() {
+ return this._node();
+ }
+
+ /**
+ * @return {number}
+ */
+ nodeIndex() {
+ return this._retainingNodeIndex;
+ }
+
+ /**
+ * @return {number}
+ */
+ retainerIndex() {
+ return this._retainerIndex;
+ }
+
+ /**
+ * @param {number} retainerIndex
+ */
+ setRetainerIndex(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() {
+ if (!this._nodeInstance)
+ this._nodeInstance = this._snapshot.createNode(this._retainingNodeIndex);
+ return this._nodeInstance;
+ }
+
+ _edge() {
+ if (!this._edgeInstance)
+ this._edgeInstance = this._snapshot.createEdge(this._globalEdgeIndex);
+ return this._edgeInstance;
+ }
+
+ /**
+ * @override
+ * @return {string}
+ */
+ toString() {
+ return this._edge().toString();
+ }
+
+ /**
+ * @override
+ * @return {number}
+ */
+ itemIndex() {
+ return this._retainerIndex;
+ }
+
+ /**
+ * @override
+ * @return {!WebInspector.HeapSnapshotCommon.Edge}
+ */
+ serialize() {
+ return new WebInspector.HeapSnapshotCommon.Edge(
+ this.name(), this.node().serialize(), this.type(), this._globalEdgeIndex);
+ }
+
+ /**
+ * @return {string}
+ */
+ type() {
+ return this._edge().type();
+ }
};
/**
- * @constructor
* @implements {WebInspector.HeapSnapshotItemIterator}
- * @param {!WebInspector.HeapSnapshotNode} retainedNode
+ * @unrestricted
*/
-WebInspector.HeapSnapshotRetainerEdgeIterator = function(retainedNode)
-{
+WebInspector.HeapSnapshotRetainerEdgeIterator = class {
+ /**
+ * @param {!WebInspector.HeapSnapshotNode} retainedNode
+ */
+ constructor(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);
- }
+ }
+
+ /**
+ * @override
+ * @return {boolean}
+ */
+ hasNext() {
+ return this.retainer.retainerIndex() < this._retainersEnd;
+ }
+
+ /**
+ * @override
+ * @return {!WebInspector.HeapSnapshotRetainerEdge}
+ */
+ item() {
+ return this.retainer;
+ }
+
+ /**
+ * @override
+ */
+ next() {
+ this.retainer.setRetainerIndex(this.retainer.retainerIndex() + 1);
+ }
};
/**
- * @constructor
* @implements {WebInspector.HeapSnapshotItem}
- * @param {!WebInspector.HeapSnapshot} snapshot
- * @param {number=} nodeIndex
+ * @unrestricted
*/
-WebInspector.HeapSnapshotNode = function(snapshot, nodeIndex)
-{
+WebInspector.HeapSnapshotNode = class {
+ /**
+ * @param {!WebInspector.HeapSnapshot} snapshot
+ * @param {number=} nodeIndex
+ */
+ constructor(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];
- }
+ }
+
+ /**
+ * @return {number}
+ */
+ distance() {
+ return this._snapshot._nodeDistances[this.nodeIndex / this._snapshot._nodeFieldCount];
+ }
+
+ /**
+ * @return {string}
+ */
+ className() {
+ throw new Error('Not implemented');
+ }
+
+ /**
+ * @return {number}
+ */
+ classIndex() {
+ throw new Error('Not implemented');
+ }
+
+ /**
+ * @return {number}
+ */
+ dominatorIndex() {
+ var nodeFieldCount = this._snapshot._nodeFieldCount;
+ return this._snapshot._dominatorsTree[this.nodeIndex / this._snapshot._nodeFieldCount] * nodeFieldCount;
+ }
+
+ /**
+ * @return {!WebInspector.HeapSnapshotEdgeIterator}
+ */
+ edges() {
+ return new WebInspector.HeapSnapshotEdgeIterator(this);
+ }
+
+ /**
+ * @return {number}
+ */
+ edgesCount() {
+ return (this.edgeIndexesEnd() - this.edgeIndexesStart()) / this._snapshot._edgeFieldsCount;
+ }
+
+ /**
+ * @return {number}
+ */
+ id() {
+ throw new Error('Not implemented');
+ }
+
+ /**
+ * @return {boolean}
+ */
+ isRoot() {
+ return this.nodeIndex === this._snapshot._rootNodeIndex;
+ }
+
+ /**
+ * @return {string}
+ */
+ name() {
+ return this._snapshot.strings[this._name()];
+ }
+
+ /**
+ * @return {number}
+ */
+ retainedSize() {
+ return this._snapshot._retainedSizes[this.ordinal()];
+ }
+
+ /**
+ * @return {!WebInspector.HeapSnapshotRetainerEdgeIterator}
+ */
+ retainers() {
+ return new WebInspector.HeapSnapshotRetainerEdgeIterator(this);
+ }
+
+ /**
+ * @return {number}
+ */
+ retainersCount() {
+ var snapshot = this._snapshot;
+ var ordinal = this.ordinal();
+ return snapshot._firstRetainerIndex[ordinal + 1] - snapshot._firstRetainerIndex[ordinal];
+ }
+
+ /**
+ * @return {number}
+ */
+ selfSize() {
+ var snapshot = this._snapshot;
+ return snapshot.nodes[this.nodeIndex + snapshot._nodeSelfSizeOffset];
+ }
+
+ /**
+ * @return {string}
+ */
+ type() {
+ return this._snapshot._nodeTypes[this.rawType()];
+ }
+
+ /**
+ * @return {number}
+ */
+ traceNodeId() {
+ var snapshot = this._snapshot;
+ return snapshot.nodes[this.nodeIndex + snapshot._nodeTraceNodeIdOffset];
+ }
+
+ /**
+ * @override
+ * @return {number}
+ */
+ itemIndex() {
+ return this.nodeIndex;
+ }
+
+ /**
+ * @override
+ * @return {!WebInspector.HeapSnapshotCommon.Node}
+ */
+ serialize() {
+ return new WebInspector.HeapSnapshotCommon.Node(
+ this.id(), this.name(), this.distance(), this.nodeIndex, this.retainedSize(), this.selfSize(), this.type());
+ }
+
+ /**
+ * @return {number}
+ */
+ _name() {
+ var snapshot = this._snapshot;
+ return snapshot.nodes[this.nodeIndex + snapshot._nodeNameOffset];
+ }
+
+ /**
+ * @return {number}
+ */
+ edgeIndexesStart() {
+ return this._snapshot._firstEdgeIndexes[this.ordinal()];
+ }
+
+ /**
+ * @return {number}
+ */
+ edgeIndexesEnd() {
+ return this._snapshot._firstEdgeIndexes[this.ordinal() + 1];
+ }
+
+ /**
+ * @return {number}
+ */
+ ordinal() {
+ return this.nodeIndex / this._snapshot._nodeFieldCount;
+ }
+
+ /**
+ * @return {number}
+ */
+ _nextNodeIndex() {
+ return this.nodeIndex + this._snapshot._nodeFieldCount;
+ }
+
+ /**
+ * @protected
+ * @return {number}
+ */
+ rawType() {
+ var snapshot = this._snapshot;
+ return snapshot.nodes[this.nodeIndex + snapshot._nodeTypeOffset];
+ }
};
/**
- * @constructor
* @implements {WebInspector.HeapSnapshotItemIterator}
- * @param {!WebInspector.HeapSnapshotNode} node
+ * @unrestricted
*/
-WebInspector.HeapSnapshotNodeIterator = function(node)
-{
+WebInspector.HeapSnapshotNodeIterator = class {
+ /**
+ * @param {!WebInspector.HeapSnapshotNode} node
+ */
+ constructor(node) {
this.node = node;
this._nodesLength = node._snapshot.nodes.length;
+ }
+
+ /**
+ * @override
+ * @return {boolean}
+ */
+ hasNext() {
+ return this.node.nodeIndex < this._nodesLength;
+ }
+
+ /**
+ * @override
+ * @return {!WebInspector.HeapSnapshotNode}
+ */
+ item() {
+ return this.node;
+ }
+
+ /**
+ * @override
+ */
+ next() {
+ this.node.nodeIndex = this.node._nextNodeIndex();
+ }
};
-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
+ * @unrestricted
*/
-WebInspector.HeapSnapshotIndexRangeIterator = function(itemProvider, indexes)
-{
+WebInspector.HeapSnapshotIndexRangeIterator = class {
+ /**
+ * @param {!WebInspector.HeapSnapshotItemIndexProvider} itemProvider
+ * @param {!Array.<number>|!Uint32Array} indexes
+ */
+ constructor(itemProvider, indexes) {
this._itemProvider = itemProvider;
this._indexes = indexes;
this._position = 0;
+ }
+
+ /**
+ * @override
+ * @return {boolean}
+ */
+ hasNext() {
+ return this._position < this._indexes.length;
+ }
+
+ /**
+ * @override
+ * @return {!WebInspector.HeapSnapshotItem}
+ */
+ item() {
+ var index = this._indexes[this._position];
+ return this._itemProvider.itemForIndex(index);
+ }
+
+ /**
+ * @override
+ */
+ next() {
+ ++this._position;
+ }
};
-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
+ * @unrestricted
*/
-WebInspector.HeapSnapshotFilteredIterator = function(iterator, filter)
-{
+WebInspector.HeapSnapshotFilteredIterator = class {
+ /**
+ * @param {!WebInspector.HeapSnapshotItemIterator} iterator
+ * @param {function(!WebInspector.HeapSnapshotItem):boolean=} filter
+ */
+ constructor(iterator, filter) {
this._iterator = iterator;
this._filter = filter;
this._skipFilteredItems();
+ }
+
+ /**
+ * @override
+ * @return {boolean}
+ */
+ hasNext() {
+ return this._iterator.hasNext();
+ }
+
+ /**
+ * @override
+ * @return {!WebInspector.HeapSnapshotItem}
+ */
+ item() {
+ return this._iterator.item();
+ }
+
+ /**
+ * @override
+ */
+ next() {
+ this._iterator.next();
+ this._skipFilteredItems();
+ }
+
+ _skipFilteredItems() {
+ while (this._iterator.hasNext() && !this._filter(this._iterator.item())) {
+ this._iterator.next();
+ }
+ }
};
-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);
- }
-};
-
+/**
+ * @unrestricted
+ */
+WebInspector.HeapSnapshotProgress = class {
+ /**
+ * @param {!WebInspector.HeapSnapshotWorkerDispatcher=} dispatcher
+ */
+ constructor(dispatcher) {
+ this._dispatcher = dispatcher;
+ }
+
+ /**
+ * @param {string} status
+ */
+ updateStatus(status) {
+ this._sendUpdateEvent(WebInspector.UIString(status));
+ }
+
+ /**
+ * @param {string} title
+ * @param {number} value
+ * @param {number} total
+ */
+ updateProgress(title, value, total) {
+ var percentValue = ((total ? (value / total) : 0) * 100).toFixed(0);
+ this._sendUpdateEvent(WebInspector.UIString(title, percentValue));
+ }
+
+ /**
+ * @param {string} error
+ */
+ reportProblem(error) {
+ // May be undefined in tests.
+ if (this._dispatcher)
+ this._dispatcher.sendEvent(WebInspector.HeapSnapshotProgressEvent.BrokenSnapshot, error);
+ }
+
+ /**
+ * @param {string} text
+ */
+ _sendUpdateEvent(text) {
+ // May be undefined in tests.
+ if (this._dispatcher)
+ this._dispatcher.sendEvent(WebInspector.HeapSnapshotProgressEvent.Update, text);
+ }
+};
/**
- * @param {string} title
- * @constructor
+ * @unrestricted
*/
-WebInspector.HeapSnapshotProblemReport = function(title)
-{
+WebInspector.HeapSnapshotProblemReport = class {
+ /**
+ * @param {string} title
+ */
+ constructor(title) {
this._errors = [title];
+ }
+
+ /**
+ * @param {string} error
+ */
+ addError(error) {
+ if (this._errors.length > 100)
+ return;
+ this._errors.push(error);
+ }
+
+ /**
+ * @override
+ * @return {string}
+ */
+ toString() {
+ return this._errors.join('\n ');
+ }
};
-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
+ * @unrestricted
*/
-WebInspector.HeapSnapshot = function(profile, progress)
-{
+WebInspector.HeapSnapshot = class {
+ /**
+ * @param {!Object} profile
+ * @param {!WebInspector.HeapSnapshotProgress} progress
+ */
+ constructor(profile, progress) {
/** @type {!Uint32Array} */
this.nodes = profile.nodes;
/** @type {!Uint32Array} */
@@ -906,1347 +845,1301 @@ WebInspector.HeapSnapshot = function(profile, progress)
this._noDistance = -5;
this._rootNodeIndex = 0;
if (profile.snapshot.root_index)
- this._rootNodeIndex = profile.snapshot.root_index;
+ this._rootNodeIndex = profile.snapshot.root_index;
this._snapshotDiffs = {};
this._aggregatesForDiff = null;
this._aggregates = {};
this._aggregatesSortedFlags = {};
this._profile = profile;
-};
-
-/**
- * @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.sample_fields = [];
- this.type_strings = {};
-}
-
-/**
- * @constructor
- */
-function HeapSnapshotHeader()
-{
- // New format.
- this.title = "";
- this.meta = new HeapSnapshotMetainfo();
- this.node_count = 0;
- this.edge_count = 0;
- this.trace_function_count = 0;
-}
-
-WebInspector.HeapSnapshot.prototype = {
- /**
- * @protected
- */
- initialize: 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("Building dominated nodes\u2026");
- this._buildDominatedNodes();
- this._progress.updateStatus("Calculating statistics\u2026");
- this._calculateStatistics();
- this._progress.updateStatus("Calculating samples\u2026");
- this._buildSamples();
- this._progress.updateStatus("Finished processing.");
-
- if (this._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(this._profile, liveObjects);
- this._progress.updateStatus("Done");
- }
- },
-
- _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();
- }
+ }
+
+ /**
+ * @protected
+ */
+ initialize() {
+ 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('Building dominated nodes\u2026');
+ this._buildDominatedNodes();
+ this._progress.updateStatus('Calculating statistics\u2026');
+ this.calculateStatistics();
+ this._progress.updateStatus('Calculating samples\u2026');
+ this._buildSamples();
+ this._progress.updateStatus('Finished processing.');
+
+ if (this._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(this._profile, liveObjects);
+ this._progress.updateStatus('Done');
+ }
+ }
+
+ _buildEdgeIndexes() {
+ 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() {
+ 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(nodeIndex) {
+ throw new Error('Not implemented');
+ }
+
+ /**
+ * @param {number} edgeIndex
+ * @return {!WebInspector.JSHeapSnapshotEdge}
+ */
+ createEdge(edgeIndex) {
+ throw new Error('Not implemented');
+ }
+
+ /**
+ * @param {number} retainerIndex
+ * @return {!WebInspector.JSHeapSnapshotRetainerEdge}
+ */
+ createRetainingEdge(retainerIndex) {
+ throw new Error('Not implemented');
+ }
+
+ _allNodes() {
+ return new WebInspector.HeapSnapshotNodeIterator(this.rootNode());
+ }
+
+ /**
+ * @return {!WebInspector.HeapSnapshotNode}
+ */
+ rootNode() {
+ return this.createNode(this._rootNodeIndex);
+ }
+
+ get rootNodeIndex() {
+ return this._rootNodeIndex;
+ }
+
+ get totalSize() {
+ return this.rootNode().retainedSize();
+ }
+
+ _getDominatedIndex(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(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(searchConfig, nodeFilter) {
+ var query = searchConfig.query;
+
+ function filterString(matchedStringIndexes, string, index) {
+ if (string.indexOf(query) !== -1)
+ matchedStringIndexes.add(index);
+ return matchedStringIndexes;
+ }
- this._aggregatesForDiff[className] = {
- indexes: indexes,
- ids: ids,
- selfSizes: selfSizes
- };
- }
- return this._aggregatesForDiff;
- },
+ 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(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(minNodeId, maxNodeId) {
/**
- * @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;
- });
- },
-
+ function nodeIdFilter(node) {
+ var id = node.id();
+ return id > minNodeId && id <= maxNodeId;
+ }
+ return nodeIdFilter;
+ }
+
+ /**
+ * @param {number} bottomUpAllocationNodeId
+ * @return {function(!WebInspector.HeapSnapshotNode):boolean|undefined}
+ */
+ _createAllocationStackFilter(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;
/**
- * The function checks is the edge should be considered during building
- * postorder iterator and dominator tree.
- *
- * @param {number} nodeIndex
- * @param {number} edgeType
+ * @param {!WebInspector.HeapSnapshotNode} node
* @return {boolean}
*/
- _isEssentialEdge: function(nodeIndex, edgeType)
- {
- // Shortcuts at the root node have special meaning of marking user global objects.
- return edgeType !== this._edgeWeakType && (edgeType !== this._edgeShortcutType || nodeIndex === this._rootNodeIndex);
- },
-
- _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 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 (!this._isEssentialEdge(nodeOrdinal * nodeFieldCount, edgeType))
- 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[rootNodeOrdinal + 1]; // no need to reiterate its edges
- for (var i = 0; i < nodeCount; ++i) {
- if (visited[i] || !this._hasOnlyWeakRetainers(i))
- continue;
-
- // Add all nodes that have only weak retainers to traverse their subgraphs.
- stackNodes[++stackTop] = i;
- stackCurrentEdge[stackTop] = firstEdgeIndexes[i];
- visited[i] = 1;
-
- dumpNode.nodeIndex = i * nodeFieldCount;
- var retainers = [];
- for (var it = dumpNode.retainers(); it.hasNext(); it.next())
- retainers.push(`${it.item().node().name()}@${it.item().node().id()}.${it.item().name()}`);
- errors.addError(`${dumpNode.name()} @${dumpNode.id()} weak retainers: ${retainers.join(", ")}`);
- }
- console.warn(errors.toString());
- }
+ 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(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 we already processed all orphan nodes that have only weak retainers and still have some orphans...
- if (postOrderIndex !== nodeCount) {
- var errors = new WebInspector.HeapSnapshotProblemReport("Still found " + (nodeCount - postOrderIndex) + " unreachable nodes in heap snapshot:");
- 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;
- for (var i = 0; i < nodeCount; ++i) {
- if (visited[i])
- continue;
- dumpNode.nodeIndex = i * nodeFieldCount;
- errors.addError(dumpNode.name() + " @" + dumpNode.id());
- // Fix it by giving the node a postorder index anyway.
- nodeOrdinal2PostOrderIndex[i] = postOrderIndex;
- postOrderIndex2NodeOrdinal[postOrderIndex++] = i;
- }
- nodeOrdinal2PostOrderIndex[rootNodeOrdinal] = postOrderIndex;
- postOrderIndex2NodeOrdinal[postOrderIndex++] = rootNodeOrdinal;
- console.warn(errors.toString());
- }
+ if (sortedIndexes && (!key || !this._aggregatesSortedFlags[key])) {
+ this._sortAggregateIndexes(aggregatesByClassName);
+ if (key)
+ this._aggregatesSortedFlags[key] = sortedIndexes;
+ }
+ return aggregatesByClassName;
+ }
+
+ /**
+ * @return {!Array.<!WebInspector.HeapSnapshotCommon.SerializedAllocationNode>}
+ */
+ allocationTracesTops() {
+ return this._allocationProfile.serializeTraceTops();
+ }
+
+ /**
+ * @param {number} nodeId
+ * @return {!WebInspector.HeapSnapshotCommon.AllocationNodeCallers}
+ */
+ allocationNodeCallers(nodeId) {
+ return this._allocationProfile.serializeCallers(nodeId);
+ }
+
+ /**
+ * @param {number} nodeIndex
+ * @return {?Array.<!WebInspector.HeapSnapshotCommon.AllocationStackFrame>}
+ */
+ allocationStack(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() {
+ 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(node) {
+ return true;
+ }
+
+ /**
+ * @param {function(!WebInspector.HeapSnapshotNode)} action
+ * @param {boolean=} userRootsOnly
+ */
+ forEachRoot(action, userRootsOnly) {
+ for (var iter = this.rootNode().edges(); iter.hasNext(); iter.next()) {
+ var node = iter.edge.node();
+ if (!userRootsOnly || this.isUserRoot(node))
+ action(node);
+ }
+ }
- return {postOrderIndex2NodeOrdinal: postOrderIndex2NodeOrdinal, nodeOrdinal2PostOrderIndex: nodeOrdinal2PostOrderIndex};
- },
+ /**
+ * @param {function(!WebInspector.HeapSnapshotNode,!WebInspector.HeapSnapshotEdge):boolean=} filter
+ */
+ calculateDistances(filter) {
+ var nodeCount = this.nodeCount;
+ var distances = this._nodeDistances;
+ var noDistance = this._noDistance;
+ for (var i = 0; i < nodeCount; ++i)
+ distances[i] = noDistance;
- /**
- * @param {number} nodeOrdinal
- * @return {boolean}
- */
- _hasOnlyWeakRetainers: function(nodeOrdinal)
- {
- var edgeTypeOffset = this._edgeTypeOffset;
- var edgeWeakType = this._edgeWeakType;
- var edgeShortcutType = this._edgeShortcutType;
- var containmentEdges = this.containmentEdges;
- var retainingEdges = this._retainingEdges;
- var beginRetainerIndex = this._firstRetainerIndex[nodeOrdinal];
- var endRetainerIndex = this._firstRetainerIndex[nodeOrdinal + 1];
- for (var retainerIndex = beginRetainerIndex; retainerIndex < endRetainerIndex; ++retainerIndex) {
- var retainerEdgeIndex = retainingEdges[retainerIndex];
- var retainerEdgeType = containmentEdges[retainerEdgeIndex + edgeTypeOffset];
- if (retainerEdgeType !== edgeWeakType && retainerEdgeType !== edgeShortcutType)
- return false;
- }
- return true;
- },
+ var nodesToVisit = new Uint32Array(this.nodeCount);
+ var nodesToVisitLength = 0;
- // The algorithm is based on the article:
- // K. Cooper, T. Harvey and K. Kennedy "A Simple, Fast Dominance Algorithm"
- // Softw. Pract. Exper. 4 (2001), pp. 1-10.
/**
- * @param {!Array.<number>} postOrderIndex2NodeOrdinal
- * @param {!Array.<number>} nodeOrdinal2PostOrderIndex
+ * @param {number} distance
+ * @param {!WebInspector.HeapSnapshotNode} node
*/
- _buildDominatorTree: function(postOrderIndex2NodeOrdinal, nodeOrdinal2PostOrderIndex)
- {
- var nodeFieldCount = this._nodeFieldCount;
- var firstRetainerIndex = this._firstRetainerIndex;
- var retainingNodes = this._retainingNodes;
- var retainingEdges = this._retainingEdges;
- var edgeFieldsCount = this._edgeFieldsCount;
- var edgeTypeOffset = this._edgeTypeOffset;
- var edgeToNodeOffset = this._edgeToNodeOffset;
- var firstEdgeIndexes = this._firstEdgeIndexes;
- var containmentEdges = this.containmentEdges;
- var rootNodeIndex = this._rootNodeIndex;
-
- var mapAndFlag = this.userObjectsMapAndFlag();
- var flags = mapAndFlag ? mapAndFlag.map : null;
- var flag = mapAndFlag ? mapAndFlag.flag : 0;
-
- var nodesCount = postOrderIndex2NodeOrdinal.length;
- var rootPostOrderedIndex = nodesCount - 1;
- var noEntry = nodesCount;
- var dominators = new Uint32Array(nodesCount);
- for (var i = 0; i < rootPostOrderedIndex; ++i)
- dominators[i] = noEntry;
- dominators[rootPostOrderedIndex] = rootPostOrderedIndex;
-
- // The affected array is used to mark entries which dominators
- // have to be racalculated because of changes in their retainers.
- var affected = new Uint8Array(nodesCount);
- var nodeOrdinal;
-
- { // Mark the root direct children as affected.
- nodeOrdinal = this._rootNodeIndex / nodeFieldCount;
- var endEdgeIndex = firstEdgeIndexes[nodeOrdinal + 1];
- for (var edgeIndex = firstEdgeIndexes[nodeOrdinal]; edgeIndex < endEdgeIndex; edgeIndex += edgeFieldsCount) {
- var edgeType = containmentEdges[edgeIndex + edgeTypeOffset];
- if (!this._isEssentialEdge(this._rootNodeIndex, edgeType))
- continue;
- var childNodeOrdinal = containmentEdges[edgeIndex + edgeToNodeOffset] / nodeFieldCount;
- affected[nodeOrdinal2PostOrderIndex[childNodeOrdinal]] = 1;
- }
- }
+ function enqueueNode(distance, node) {
+ var ordinal = node.ordinal();
+ if (distances[ordinal] !== noDistance)
+ return;
+ distances[ordinal] = distance;
+ nodesToVisit[nodesToVisitLength++] = node.nodeIndex;
+ }
- var changed = true;
- while (changed) {
- changed = false;
- for (var postOrderIndex = rootPostOrderedIndex - 1; postOrderIndex >= 0; --postOrderIndex) {
- if (affected[postOrderIndex] === 0)
- continue;
- affected[postOrderIndex] = 0;
- // If dominator of the entry has already been set to root,
- // then it can't propagate any further.
- if (dominators[postOrderIndex] === rootPostOrderedIndex)
- continue;
- nodeOrdinal = postOrderIndex2NodeOrdinal[postOrderIndex];
- var nodeFlag = !flags || (flags[nodeOrdinal] & flag);
- var newDominatorIndex = noEntry;
- var beginRetainerIndex = firstRetainerIndex[nodeOrdinal];
- var endRetainerIndex = firstRetainerIndex[nodeOrdinal + 1];
- var orphanNode = true;
- for (var retainerIndex = beginRetainerIndex; retainerIndex < endRetainerIndex; ++retainerIndex) {
- var retainerEdgeIndex = retainingEdges[retainerIndex];
- var retainerEdgeType = containmentEdges[retainerEdgeIndex + edgeTypeOffset];
- var retainerNodeIndex = retainingNodes[retainerIndex];
- if (!this._isEssentialEdge(retainerNodeIndex, retainerEdgeType))
- continue;
- orphanNode = false;
- var retainerNodeOrdinal = retainerNodeIndex / nodeFieldCount;
- var retainerNodeFlag = !flags || (flags[retainerNodeOrdinal] & 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 (retainerNodeIndex !== rootNodeIndex && nodeFlag && !retainerNodeFlag)
- continue;
- var retanerPostOrderIndex = nodeOrdinal2PostOrderIndex[retainerNodeOrdinal];
- if (dominators[retanerPostOrderIndex] !== noEntry) {
- if (newDominatorIndex === noEntry)
- newDominatorIndex = retanerPostOrderIndex;
- else {
- while (retanerPostOrderIndex !== newDominatorIndex) {
- while (retanerPostOrderIndex < newDominatorIndex)
- retanerPostOrderIndex = dominators[retanerPostOrderIndex];
- while (newDominatorIndex < retanerPostOrderIndex)
- newDominatorIndex = dominators[newDominatorIndex];
- }
- }
- // If idom has already reached the root, it doesn't make sense
- // to check other retainers.
- if (newDominatorIndex === rootPostOrderedIndex)
- break;
- }
- }
- // Make root dominator of orphans.
- if (orphanNode)
- newDominatorIndex = rootPostOrderedIndex;
- if (newDominatorIndex !== noEntry && dominators[postOrderIndex] !== newDominatorIndex) {
- dominators[postOrderIndex] = newDominatorIndex;
- changed = true;
- nodeOrdinal = postOrderIndex2NodeOrdinal[postOrderIndex];
- var beginEdgeToNodeFieldIndex = firstEdgeIndexes[nodeOrdinal] + edgeToNodeOffset;
- var endEdgeToNodeFieldIndex = firstEdgeIndexes[nodeOrdinal + 1];
- for (var toNodeFieldIndex = beginEdgeToNodeFieldIndex;
- toNodeFieldIndex < endEdgeToNodeFieldIndex;
- toNodeFieldIndex += edgeFieldsCount) {
- var childNodeOrdinal = containmentEdges[toNodeFieldIndex] / nodeFieldCount;
- affected[nodeOrdinal2PostOrderIndex[childNodeOrdinal]] = 1;
- }
- }
- }
- }
+ 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(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(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);
+ }
+ }
- var dominatorsTree = new Uint32Array(nodesCount);
- for (var postOrderIndex = 0, l = dominators.length; postOrderIndex < l; ++postOrderIndex) {
- nodeOrdinal = postOrderIndex2NodeOrdinal[postOrderIndex];
- dominatorsTree[nodeOrdinal] = postOrderIndex2NodeOrdinal[dominators[postOrderIndex]];
- }
- return dominatorsTree;
- },
-
- _calculateRetainedSizes: function(postOrderIndex2NodeOrdinal)
- {
- var nodeCount = this.nodeCount;
- var nodes = this.nodes;
- var nodeSelfSizeOffset = this._nodeSelfSizeOffset;
- var nodeFieldCount = this._nodeFieldCount;
- var dominatorsTree = this._dominatorsTree;
- var retainedSizes = this._retainedSizes;
-
- for (var nodeOrdinal = 0; nodeOrdinal < nodeCount; ++nodeOrdinal)
- retainedSizes[nodeOrdinal] = nodes[nodeOrdinal * nodeFieldCount + nodeSelfSizeOffset];
-
- // Propagate retained sizes for each node excluding root.
- for (var postOrderIndex = 0; postOrderIndex < nodeCount - 1; ++postOrderIndex) {
- var nodeOrdinal = postOrderIndex2NodeOrdinal[postOrderIndex];
- var dominatorOrdinal = dominatorsTree[nodeOrdinal];
- retainedSizes[dominatorOrdinal] += retainedSizes[nodeOrdinal];
- }
- },
-
- _buildDominatedNodes: function()
- {
- // Builds up two arrays:
- // - "dominatedNodes" is a continuous array, where each node owns an
- // interval (can be empty) with corresponding dominated nodes.
- // - "indexArray" is an array of indexes in the "dominatedNodes"
- // with the same positions as in the _nodeIndex.
- var indexArray = this._firstDominatedNodeIndex;
- // All nodes except the root have dominators.
- var dominatedNodes = this._dominatedNodes;
-
- // Count the number of dominated nodes for each node. Skip the root (node at
- // index 0) as it is the only node that dominates itself.
- var nodeFieldCount = this._nodeFieldCount;
- var dominatorsTree = this._dominatorsTree;
-
- var fromNodeOrdinal = 0;
- var toNodeOrdinal = this.nodeCount;
- var rootNodeOrdinal = this._rootNodeIndex / nodeFieldCount;
- if (rootNodeOrdinal === fromNodeOrdinal)
- fromNodeOrdinal = 1;
- else if (rootNodeOrdinal === toNodeOrdinal - 1)
- toNodeOrdinal = toNodeOrdinal - 1;
- else
- throw new Error("Root node is expected to be either first or last");
- for (var nodeOrdinal = fromNodeOrdinal; nodeOrdinal < toNodeOrdinal; ++nodeOrdinal)
- ++indexArray[dominatorsTree[nodeOrdinal]];
- // Put in the first slot of each dominatedNodes slice the count of entries
- // that will be filled.
- var firstDominatedNodeIndex = 0;
- for (var i = 0, l = this.nodeCount; i < l; ++i) {
- var dominatedCount = dominatedNodes[firstDominatedNodeIndex] = indexArray[i];
- indexArray[i] = firstDominatedNodeIndex;
- firstDominatedNodeIndex += dominatedCount;
- }
- indexArray[this.nodeCount] = dominatedNodes.length;
- // Fill up the dominatedNodes array with indexes of dominated nodes. Skip the root (node at
- // index 0) as it is the only node that dominates itself.
- for (var nodeOrdinal = fromNodeOrdinal; nodeOrdinal < toNodeOrdinal; ++nodeOrdinal) {
- var dominatorOrdinal = dominatorsTree[nodeOrdinal];
- var dominatedRefIndex = indexArray[dominatorOrdinal];
- dominatedRefIndex += (--dominatedNodes[dominatedRefIndex]);
- dominatedNodes[dominatedRefIndex] = nodeOrdinal * nodeFieldCount;
- }
- },
-
- _buildSamples: function()
- {
- var samples = this._rawSamples;
- if (!samples || !samples.length)
- return;
- var sampleCount = samples.length / 2;
- var sizeForRange = new Array(sampleCount);
- var timestamps = new Array(sampleCount);
- var lastAssignedIds = new Array(sampleCount);
-
- var timestampOffset = this._metaNode.sample_fields.indexOf("timestamp_us");
- var lastAssignedIdOffset = this._metaNode.sample_fields.indexOf("last_assigned_id");
- for (var i = 0; i < sampleCount; i++) {
- sizeForRange[i] = 0;
- timestamps[i] = (samples[2 * i + timestampOffset]) / 1000;
- lastAssignedIds[i] = samples[2 * i + lastAssignedIdOffset];
+ // 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(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);
}
-
- var nodes = this.nodes;
- var nodesLength = nodes.length;
- var nodeFieldCount = this._nodeFieldCount;
- var node = this.rootNode();
- for (var nodeIndex = 0; nodeIndex < nodesLength; nodeIndex += nodeFieldCount) {
- node.nodeIndex = nodeIndex;
-
- var nodeId = node.id();
- // JS objects have odd ids, skip native objects.
- if (nodeId % 2 === 0)
- continue;
- var rangeIndex = lastAssignedIds.lowerBound(nodeId);
- if (rangeIndex === sampleCount) {
- // TODO: make heap profiler not allocate while taking snapshot
- continue;
- }
- sizeForRange[rangeIndex] += node.selfSize();
+ }
+ 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(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;
+ });
+ }
+
+ /**
+ * The function checks is the edge should be considered during building
+ * postorder iterator and dominator tree.
+ *
+ * @param {number} nodeIndex
+ * @param {number} edgeType
+ * @return {boolean}
+ */
+ _isEssentialEdge(nodeIndex, edgeType) {
+ // Shortcuts at the root node have special meaning of marking user global objects.
+ return edgeType !== this._edgeWeakType &&
+ (edgeType !== this._edgeShortcutType || nodeIndex === this._rootNodeIndex);
+ }
+
+ _buildPostOrderIndex() {
+ 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 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 (!this._isEssentialEdge(nodeOrdinal * nodeFieldCount, edgeType))
+ 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;
}
- this._samples = new WebInspector.HeapSnapshotCommon.Samples(timestamps, lastAssignedIds, sizeForRange);
- },
-
- /**
- * @return {?WebInspector.HeapSnapshotCommon.Samples}
- */
- getSamples: function()
- {
- return this._samples;
- },
-
- _calculateFlags: function()
- {
- throw new Error("Not implemented");
- },
-
- _calculateStatistics: function()
- {
- throw new Error("Not implemented");
- },
-
- userObjectsMapAndFlag: function()
- {
- throw new Error("Not implemented");
- },
+ }
+
+ 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[rootNodeOrdinal + 1]; // no need to reiterate its edges
+ for (var i = 0; i < nodeCount; ++i) {
+ if (visited[i] || !this._hasOnlyWeakRetainers(i))
+ continue;
+
+ // Add all nodes that have only weak retainers to traverse their subgraphs.
+ stackNodes[++stackTop] = i;
+ stackCurrentEdge[stackTop] = firstEdgeIndexes[i];
+ visited[i] = 1;
+
+ dumpNode.nodeIndex = i * nodeFieldCount;
+ var retainers = [];
+ for (var it = dumpNode.retainers(); it.hasNext(); it.next())
+ retainers.push(`${it.item().node().name()}@${it.item().node().id()}.${it.item().name()}`);
+ errors.addError(`${dumpNode.name()} @${dumpNode.id()} weak retainers: ${retainers.join(', ')}`);
+ }
+ console.warn(errors.toString());
+ }
- /**
- * @param {string} baseSnapshotId
- * @param {!Object.<string, !WebInspector.HeapSnapshotCommon.AggregateForDiff>} baseSnapshotAggregates
- * @return {!Object.<string, !WebInspector.HeapSnapshotCommon.Diff>}
- */
- calculateSnapshotDiff: function(baseSnapshotId, baseSnapshotAggregates)
- {
- var snapshotDiff = this._snapshotDiffs[baseSnapshotId];
- if (snapshotDiff)
- return snapshotDiff;
- snapshotDiff = {};
-
- var aggregates = this.aggregates(true, "allObjects");
- for (var className in baseSnapshotAggregates) {
- var baseAggregate = baseSnapshotAggregates[className];
- var diff = this._calculateDiffForClass(baseAggregate, aggregates[className]);
- if (diff)
- snapshotDiff[className] = diff;
- }
- var emptyBaseAggregate = new WebInspector.HeapSnapshotCommon.AggregateForDiff();
- for (var className in aggregates) {
- if (className in baseSnapshotAggregates)
- continue;
- snapshotDiff[className] = this._calculateDiffForClass(emptyBaseAggregate, aggregates[className]);
- }
+ // If we already processed all orphan nodes that have only weak retainers and still have some orphans...
+ if (postOrderIndex !== nodeCount) {
+ var errors = new WebInspector.HeapSnapshotProblemReport(
+ 'Still found ' + (nodeCount - postOrderIndex) + ' unreachable nodes in heap snapshot:');
+ 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;
+ for (var i = 0; i < nodeCount; ++i) {
+ if (visited[i])
+ continue;
+ dumpNode.nodeIndex = i * nodeFieldCount;
+ errors.addError(dumpNode.name() + ' @' + dumpNode.id());
+ // Fix it by giving the node a postorder index anyway.
+ nodeOrdinal2PostOrderIndex[i] = postOrderIndex;
+ postOrderIndex2NodeOrdinal[postOrderIndex++] = i;
+ }
+ nodeOrdinal2PostOrderIndex[rootNodeOrdinal] = postOrderIndex;
+ postOrderIndex2NodeOrdinal[postOrderIndex++] = rootNodeOrdinal;
+ console.warn(errors.toString());
+ }
- this._snapshotDiffs[baseSnapshotId] = snapshotDiff;
- return snapshotDiff;
- },
+ return {
+ postOrderIndex2NodeOrdinal: postOrderIndex2NodeOrdinal,
+ nodeOrdinal2PostOrderIndex: nodeOrdinal2PostOrderIndex
+ };
+ }
+
+ /**
+ * @param {number} nodeOrdinal
+ * @return {boolean}
+ */
+ _hasOnlyWeakRetainers(nodeOrdinal) {
+ var edgeTypeOffset = this._edgeTypeOffset;
+ var edgeWeakType = this._edgeWeakType;
+ var edgeShortcutType = this._edgeShortcutType;
+ var containmentEdges = this.containmentEdges;
+ var retainingEdges = this._retainingEdges;
+ var beginRetainerIndex = this._firstRetainerIndex[nodeOrdinal];
+ var endRetainerIndex = this._firstRetainerIndex[nodeOrdinal + 1];
+ for (var retainerIndex = beginRetainerIndex; retainerIndex < endRetainerIndex; ++retainerIndex) {
+ var retainerEdgeIndex = retainingEdges[retainerIndex];
+ var retainerEdgeType = containmentEdges[retainerEdgeIndex + edgeTypeOffset];
+ if (retainerEdgeType !== edgeWeakType && retainerEdgeType !== edgeShortcutType)
+ return false;
+ }
+ return true;
+ }
+
+ // The algorithm is based on the article:
+ // K. Cooper, T. Harvey and K. Kennedy "A Simple, Fast Dominance Algorithm"
+ // Softw. Pract. Exper. 4 (2001), pp. 1-10.
+ /**
+ * @param {!Array.<number>} postOrderIndex2NodeOrdinal
+ * @param {!Array.<number>} nodeOrdinal2PostOrderIndex
+ */
+ _buildDominatorTree(postOrderIndex2NodeOrdinal, nodeOrdinal2PostOrderIndex) {
+ var nodeFieldCount = this._nodeFieldCount;
+ var firstRetainerIndex = this._firstRetainerIndex;
+ var retainingNodes = this._retainingNodes;
+ var retainingEdges = this._retainingEdges;
+ var edgeFieldsCount = this._edgeFieldsCount;
+ var edgeTypeOffset = this._edgeTypeOffset;
+ var edgeToNodeOffset = this._edgeToNodeOffset;
+ var firstEdgeIndexes = this._firstEdgeIndexes;
+ var containmentEdges = this.containmentEdges;
+ var rootNodeIndex = this._rootNodeIndex;
+
+ var mapAndFlag = this.userObjectsMapAndFlag();
+ var flags = mapAndFlag ? mapAndFlag.map : null;
+ var flag = mapAndFlag ? mapAndFlag.flag : 0;
+
+ var nodesCount = postOrderIndex2NodeOrdinal.length;
+ var rootPostOrderedIndex = nodesCount - 1;
+ var noEntry = nodesCount;
+ var dominators = new Uint32Array(nodesCount);
+ for (var i = 0; i < rootPostOrderedIndex; ++i)
+ dominators[i] = noEntry;
+ dominators[rootPostOrderedIndex] = rootPostOrderedIndex;
+
+ // The affected array is used to mark entries which dominators
+ // have to be racalculated because of changes in their retainers.
+ var affected = new Uint8Array(nodesCount);
+ var nodeOrdinal;
+
+ { // Mark the root direct children as affected.
+ nodeOrdinal = this._rootNodeIndex / nodeFieldCount;
+ var endEdgeIndex = firstEdgeIndexes[nodeOrdinal + 1];
+ for (var edgeIndex = firstEdgeIndexes[nodeOrdinal]; edgeIndex < endEdgeIndex; edgeIndex += edgeFieldsCount) {
+ var edgeType = containmentEdges[edgeIndex + edgeTypeOffset];
+ if (!this._isEssentialEdge(this._rootNodeIndex, edgeType))
+ continue;
+ var childNodeOrdinal = containmentEdges[edgeIndex + edgeToNodeOffset] / nodeFieldCount;
+ affected[nodeOrdinal2PostOrderIndex[childNodeOrdinal]] = 1;
+ }
+ }
- /**
- * @param {!WebInspector.HeapSnapshotCommon.AggregateForDiff} baseAggregate
- * @param {!WebInspector.HeapSnapshotCommon.Aggregate} aggregate
- * @return {?WebInspector.HeapSnapshotCommon.Diff}
- */
- _calculateDiffForClass: function(baseAggregate, aggregate)
- {
- var baseIds = baseAggregate.ids;
- var baseIndexes = baseAggregate.indexes;
- var baseSelfSizes = baseAggregate.selfSizes;
-
- var indexes = aggregate ? aggregate.idxs : [];
-
- var i = 0, l = baseIds.length;
- var j = 0, m = indexes.length;
- var diff = new WebInspector.HeapSnapshotCommon.Diff();
-
- var nodeB = this.createNode(indexes[j]);
- while (i < l && j < m) {
- var nodeAId = baseIds[i];
- if (nodeAId < nodeB.id()) {
- diff.deletedIndexes.push(baseIndexes[i]);
- diff.removedCount++;
- diff.removedSize += baseSelfSizes[i];
- ++i;
- } else if (nodeAId > nodeB.id()) { // Native nodes(e.g. dom groups) may have ids less than max JS object id in the base snapshot
- diff.addedIndexes.push(indexes[j]);
- diff.addedCount++;
- diff.addedSize += nodeB.selfSize();
- nodeB.nodeIndex = indexes[++j];
- } else { // nodeAId === nodeB.id()
- ++i;
- nodeB.nodeIndex = indexes[++j];
+ var changed = true;
+ while (changed) {
+ changed = false;
+ for (var postOrderIndex = rootPostOrderedIndex - 1; postOrderIndex >= 0; --postOrderIndex) {
+ if (affected[postOrderIndex] === 0)
+ continue;
+ affected[postOrderIndex] = 0;
+ // If dominator of the entry has already been set to root,
+ // then it can't propagate any further.
+ if (dominators[postOrderIndex] === rootPostOrderedIndex)
+ continue;
+ nodeOrdinal = postOrderIndex2NodeOrdinal[postOrderIndex];
+ var nodeFlag = !flags || (flags[nodeOrdinal] & flag);
+ var newDominatorIndex = noEntry;
+ var beginRetainerIndex = firstRetainerIndex[nodeOrdinal];
+ var endRetainerIndex = firstRetainerIndex[nodeOrdinal + 1];
+ var orphanNode = true;
+ for (var retainerIndex = beginRetainerIndex; retainerIndex < endRetainerIndex; ++retainerIndex) {
+ var retainerEdgeIndex = retainingEdges[retainerIndex];
+ var retainerEdgeType = containmentEdges[retainerEdgeIndex + edgeTypeOffset];
+ var retainerNodeIndex = retainingNodes[retainerIndex];
+ if (!this._isEssentialEdge(retainerNodeIndex, retainerEdgeType))
+ continue;
+ orphanNode = false;
+ var retainerNodeOrdinal = retainerNodeIndex / nodeFieldCount;
+ var retainerNodeFlag = !flags || (flags[retainerNodeOrdinal] & 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 (retainerNodeIndex !== rootNodeIndex && nodeFlag && !retainerNodeFlag)
+ continue;
+ var retanerPostOrderIndex = nodeOrdinal2PostOrderIndex[retainerNodeOrdinal];
+ if (dominators[retanerPostOrderIndex] !== noEntry) {
+ if (newDominatorIndex === noEntry)
+ newDominatorIndex = retanerPostOrderIndex;
+ else {
+ while (retanerPostOrderIndex !== newDominatorIndex) {
+ while (retanerPostOrderIndex < newDominatorIndex)
+ retanerPostOrderIndex = dominators[retanerPostOrderIndex];
+ while (newDominatorIndex < retanerPostOrderIndex)
+ newDominatorIndex = dominators[newDominatorIndex];
+ }
}
+ // If idom has already reached the root, it doesn't make sense
+ // to check other retainers.
+ if (newDominatorIndex === rootPostOrderedIndex)
+ break;
+ }
}
- while (i < l) {
- diff.deletedIndexes.push(baseIndexes[i]);
- diff.removedCount++;
- diff.removedSize += baseSelfSizes[i];
- ++i;
- }
- while (j < m) {
- diff.addedIndexes.push(indexes[j]);
- diff.addedCount++;
- diff.addedSize += nodeB.selfSize();
- nodeB.nodeIndex = indexes[++j];
+ // Make root dominator of orphans.
+ if (orphanNode)
+ newDominatorIndex = rootPostOrderedIndex;
+ if (newDominatorIndex !== noEntry && dominators[postOrderIndex] !== newDominatorIndex) {
+ dominators[postOrderIndex] = newDominatorIndex;
+ changed = true;
+ nodeOrdinal = postOrderIndex2NodeOrdinal[postOrderIndex];
+ var beginEdgeToNodeFieldIndex = firstEdgeIndexes[nodeOrdinal] + edgeToNodeOffset;
+ var endEdgeToNodeFieldIndex = firstEdgeIndexes[nodeOrdinal + 1];
+ for (var toNodeFieldIndex = beginEdgeToNodeFieldIndex; toNodeFieldIndex < endEdgeToNodeFieldIndex;
+ toNodeFieldIndex += edgeFieldsCount) {
+ var childNodeOrdinal = containmentEdges[toNodeFieldIndex] / nodeFieldCount;
+ affected[nodeOrdinal2PostOrderIndex[childNodeOrdinal]] = 1;
+ }
}
- diff.countDelta = diff.addedCount - diff.removedCount;
- diff.sizeDelta = diff.addedSize - diff.removedSize;
- if (!diff.addedCount && !diff.removedCount)
- return null;
- return diff;
- },
-
- _nodeForSnapshotObjectId: function(snapshotObjectId)
- {
- for (var it = this._allNodes(); it.hasNext(); it.next()) {
- if (it.node.id() === snapshotObjectId)
- return it.node;
- }
- return null;
- },
-
- /**
- * @param {string} snapshotObjectId
- * @return {?string}
- */
- nodeClassName: function(snapshotObjectId)
- {
- var node = this._nodeForSnapshotObjectId(snapshotObjectId);
- if (node)
- return node.className();
- return null;
- },
-
- /**
- * @param {string} name
- * @return {!Array.<number>}
- */
- idsOfObjectsWithName: function(name)
- {
- var ids = [];
- for (var it = this._allNodes(); it.hasNext(); it.next()) {
- if (it.item().name() === name)
- ids.push(it.item().id());
- }
- return ids;
- },
-
- /**
- * @param {number} nodeIndex
- * @return {!WebInspector.HeapSnapshotEdgesProvider}
- */
- createEdgesProvider: function(nodeIndex)
- {
- var node = this.createNode(nodeIndex);
- var filter = this.containmentEdgesFilter();
- var indexProvider = new WebInspector.HeapSnapshotEdgeIndexProvider(this);
- return new WebInspector.HeapSnapshotEdgesProvider(this, filter, node.edges(), indexProvider);
- },
-
- /**
- * @param {number} nodeIndex
- * @param {?function(!WebInspector.HeapSnapshotEdge):boolean} filter
- * @return {!WebInspector.HeapSnapshotEdgesProvider}
- */
- createEdgesProviderForTest: function(nodeIndex, filter)
- {
- var node = this.createNode(nodeIndex);
- var indexProvider = new WebInspector.HeapSnapshotEdgeIndexProvider(this);
- return new WebInspector.HeapSnapshotEdgesProvider(this, filter, node.edges(), indexProvider);
- },
-
- /**
- * @return {?function(!WebInspector.HeapSnapshotEdge):boolean}
- */
- retainingEdgesFilter: function()
- {
- return null;
- },
-
- /**
- * @return {?function(!WebInspector.HeapSnapshotEdge):boolean}
- */
- containmentEdgesFilter: function()
- {
- return null;
- },
-
- /**
- * @param {number} nodeIndex
- * @return {!WebInspector.HeapSnapshotEdgesProvider}
- */
- createRetainingEdgesProvider: function(nodeIndex)
- {
- var node = this.createNode(nodeIndex);
- var filter = this.retainingEdgesFilter();
- var indexProvider = new WebInspector.HeapSnapshotRetainerEdgeIndexProvider(this);
- return new WebInspector.HeapSnapshotEdgesProvider(this, filter, node.retainers(), indexProvider);
- },
-
- /**
- * @param {string} baseSnapshotId
- * @param {string} className
- * @return {!WebInspector.HeapSnapshotNodesProvider}
- */
- createAddedNodesProvider: function(baseSnapshotId, className)
- {
- var snapshotDiff = this._snapshotDiffs[baseSnapshotId];
- var diffForClass = snapshotDiff[className];
- return new WebInspector.HeapSnapshotNodesProvider(this, null, diffForClass.addedIndexes);
- },
+ }
+ }
- /**
- * @param {!Array.<number>} nodeIndexes
- * @return {!WebInspector.HeapSnapshotNodesProvider}
- */
- createDeletedNodesProvider: function(nodeIndexes)
- {
- return new WebInspector.HeapSnapshotNodesProvider(this, null, nodeIndexes);
- },
+ var dominatorsTree = new Uint32Array(nodesCount);
+ for (var postOrderIndex = 0, l = dominators.length; postOrderIndex < l; ++postOrderIndex) {
+ nodeOrdinal = postOrderIndex2NodeOrdinal[postOrderIndex];
+ dominatorsTree[nodeOrdinal] = postOrderIndex2NodeOrdinal[dominators[postOrderIndex]];
+ }
+ return dominatorsTree;
+ }
+
+ _calculateRetainedSizes(postOrderIndex2NodeOrdinal) {
+ var nodeCount = this.nodeCount;
+ var nodes = this.nodes;
+ var nodeSelfSizeOffset = this._nodeSelfSizeOffset;
+ var nodeFieldCount = this._nodeFieldCount;
+ var dominatorsTree = this._dominatorsTree;
+ var retainedSizes = this._retainedSizes;
+
+ for (var nodeOrdinal = 0; nodeOrdinal < nodeCount; ++nodeOrdinal)
+ retainedSizes[nodeOrdinal] = nodes[nodeOrdinal * nodeFieldCount + nodeSelfSizeOffset];
+
+ // Propagate retained sizes for each node excluding root.
+ for (var postOrderIndex = 0; postOrderIndex < nodeCount - 1; ++postOrderIndex) {
+ var nodeOrdinal = postOrderIndex2NodeOrdinal[postOrderIndex];
+ var dominatorOrdinal = dominatorsTree[nodeOrdinal];
+ retainedSizes[dominatorOrdinal] += retainedSizes[nodeOrdinal];
+ }
+ }
+
+ _buildDominatedNodes() {
+ // Builds up two arrays:
+ // - "dominatedNodes" is a continuous array, where each node owns an
+ // interval (can be empty) with corresponding dominated nodes.
+ // - "indexArray" is an array of indexes in the "dominatedNodes"
+ // with the same positions as in the _nodeIndex.
+ var indexArray = this._firstDominatedNodeIndex;
+ // All nodes except the root have dominators.
+ var dominatedNodes = this._dominatedNodes;
+
+ // Count the number of dominated nodes for each node. Skip the root (node at
+ // index 0) as it is the only node that dominates itself.
+ var nodeFieldCount = this._nodeFieldCount;
+ var dominatorsTree = this._dominatorsTree;
+
+ var fromNodeOrdinal = 0;
+ var toNodeOrdinal = this.nodeCount;
+ var rootNodeOrdinal = this._rootNodeIndex / nodeFieldCount;
+ if (rootNodeOrdinal === fromNodeOrdinal)
+ fromNodeOrdinal = 1;
+ else if (rootNodeOrdinal === toNodeOrdinal - 1)
+ toNodeOrdinal = toNodeOrdinal - 1;
+ else
+ throw new Error('Root node is expected to be either first or last');
+ for (var nodeOrdinal = fromNodeOrdinal; nodeOrdinal < toNodeOrdinal; ++nodeOrdinal)
+ ++indexArray[dominatorsTree[nodeOrdinal]];
+ // Put in the first slot of each dominatedNodes slice the count of entries
+ // that will be filled.
+ var firstDominatedNodeIndex = 0;
+ for (var i = 0, l = this.nodeCount; i < l; ++i) {
+ var dominatedCount = dominatedNodes[firstDominatedNodeIndex] = indexArray[i];
+ indexArray[i] = firstDominatedNodeIndex;
+ firstDominatedNodeIndex += dominatedCount;
+ }
+ indexArray[this.nodeCount] = dominatedNodes.length;
+ // Fill up the dominatedNodes array with indexes of dominated nodes. Skip the root (node at
+ // index 0) as it is the only node that dominates itself.
+ for (var nodeOrdinal = fromNodeOrdinal; nodeOrdinal < toNodeOrdinal; ++nodeOrdinal) {
+ var dominatorOrdinal = dominatorsTree[nodeOrdinal];
+ var dominatedRefIndex = indexArray[dominatorOrdinal];
+ dominatedRefIndex += (--dominatedNodes[dominatedRefIndex]);
+ dominatedNodes[dominatedRefIndex] = nodeOrdinal * nodeFieldCount;
+ }
+ }
+
+ _buildSamples() {
+ var samples = this._rawSamples;
+ if (!samples || !samples.length)
+ return;
+ var sampleCount = samples.length / 2;
+ var sizeForRange = new Array(sampleCount);
+ var timestamps = new Array(sampleCount);
+ var lastAssignedIds = new Array(sampleCount);
+
+ var timestampOffset = this._metaNode.sample_fields.indexOf('timestamp_us');
+ var lastAssignedIdOffset = this._metaNode.sample_fields.indexOf('last_assigned_id');
+ for (var i = 0; i < sampleCount; i++) {
+ sizeForRange[i] = 0;
+ timestamps[i] = (samples[2 * i + timestampOffset]) / 1000;
+ lastAssignedIds[i] = samples[2 * i + lastAssignedIdOffset];
+ }
- /**
- * @return {?function(!WebInspector.HeapSnapshotNode):boolean}
- */
- classNodesFilter: function()
- {
- return null;
- },
+ var nodes = this.nodes;
+ var nodesLength = nodes.length;
+ var nodeFieldCount = this._nodeFieldCount;
+ var node = this.rootNode();
+ for (var nodeIndex = 0; nodeIndex < nodesLength; nodeIndex += nodeFieldCount) {
+ node.nodeIndex = nodeIndex;
+
+ var nodeId = node.id();
+ // JS objects have odd ids, skip native objects.
+ if (nodeId % 2 === 0)
+ continue;
+ var rangeIndex = lastAssignedIds.lowerBound(nodeId);
+ if (rangeIndex === sampleCount) {
+ // TODO: make heap profiler not allocate while taking snapshot
+ continue;
+ }
+ sizeForRange[rangeIndex] += node.selfSize();
+ }
+ this._samples = new WebInspector.HeapSnapshotCommon.Samples(timestamps, lastAssignedIds, sizeForRange);
+ }
+
+ /**
+ * @return {?WebInspector.HeapSnapshotCommon.Samples}
+ */
+ getSamples() {
+ return this._samples;
+ }
+
+ /**
+ * @protected
+ */
+ calculateFlags() {
+ throw new Error('Not implemented');
+ }
+
+ /**
+ * @protected
+ */
+ calculateStatistics() {
+ throw new Error('Not implemented');
+ }
+
+ userObjectsMapAndFlag() {
+ throw new Error('Not implemented');
+ }
+
+ /**
+ * @param {string} baseSnapshotId
+ * @param {!Object.<string, !WebInspector.HeapSnapshotCommon.AggregateForDiff>} baseSnapshotAggregates
+ * @return {!Object.<string, !WebInspector.HeapSnapshotCommon.Diff>}
+ */
+ calculateSnapshotDiff(baseSnapshotId, baseSnapshotAggregates) {
+ var snapshotDiff = this._snapshotDiffs[baseSnapshotId];
+ if (snapshotDiff)
+ return snapshotDiff;
+ snapshotDiff = {};
+
+ var aggregates = this.aggregates(true, 'allObjects');
+ for (var className in baseSnapshotAggregates) {
+ var baseAggregate = baseSnapshotAggregates[className];
+ var diff = this._calculateDiffForClass(baseAggregate, aggregates[className]);
+ if (diff)
+ snapshotDiff[className] = diff;
+ }
+ var emptyBaseAggregate = new WebInspector.HeapSnapshotCommon.AggregateForDiff();
+ for (var className in aggregates) {
+ if (className in baseSnapshotAggregates)
+ continue;
+ snapshotDiff[className] = this._calculateDiffForClass(emptyBaseAggregate, aggregates[className]);
+ }
- /**
- * @param {string} className
- * @param {!WebInspector.HeapSnapshotCommon.NodeFilter} nodeFilter
- * @return {!WebInspector.HeapSnapshotNodesProvider}
- */
- createNodesProviderForClass: function(className, nodeFilter)
- {
- return new WebInspector.HeapSnapshotNodesProvider(this, this.classNodesFilter(), this.aggregatesWithFilter(nodeFilter)[className].idxs);
- },
+ this._snapshotDiffs[baseSnapshotId] = snapshotDiff;
+ return snapshotDiff;
+ }
+
+ /**
+ * @param {!WebInspector.HeapSnapshotCommon.AggregateForDiff} baseAggregate
+ * @param {!WebInspector.HeapSnapshotCommon.Aggregate} aggregate
+ * @return {?WebInspector.HeapSnapshotCommon.Diff}
+ */
+ _calculateDiffForClass(baseAggregate, aggregate) {
+ var baseIds = baseAggregate.ids;
+ var baseIndexes = baseAggregate.indexes;
+ var baseSelfSizes = baseAggregate.selfSizes;
+
+ var indexes = aggregate ? aggregate.idxs : [];
+
+ var i = 0, l = baseIds.length;
+ var j = 0, m = indexes.length;
+ var diff = new WebInspector.HeapSnapshotCommon.Diff();
+
+ var nodeB = this.createNode(indexes[j]);
+ while (i < l && j < m) {
+ var nodeAId = baseIds[i];
+ if (nodeAId < nodeB.id()) {
+ diff.deletedIndexes.push(baseIndexes[i]);
+ diff.removedCount++;
+ diff.removedSize += baseSelfSizes[i];
+ ++i;
+ } else if (
+ nodeAId >
+ nodeB.id()) { // Native nodes(e.g. dom groups) may have ids less than max JS object id in the base snapshot
+ diff.addedIndexes.push(indexes[j]);
+ diff.addedCount++;
+ diff.addedSize += nodeB.selfSize();
+ nodeB.nodeIndex = indexes[++j];
+ } else { // nodeAId === nodeB.id()
+ ++i;
+ nodeB.nodeIndex = indexes[++j];
+ }
+ }
+ while (i < l) {
+ diff.deletedIndexes.push(baseIndexes[i]);
+ diff.removedCount++;
+ diff.removedSize += baseSelfSizes[i];
+ ++i;
+ }
+ while (j < m) {
+ diff.addedIndexes.push(indexes[j]);
+ diff.addedCount++;
+ diff.addedSize += nodeB.selfSize();
+ nodeB.nodeIndex = indexes[++j];
+ }
+ diff.countDelta = diff.addedCount - diff.removedCount;
+ diff.sizeDelta = diff.addedSize - diff.removedSize;
+ if (!diff.addedCount && !diff.removedCount)
+ return null;
+ return diff;
+ }
+
+ _nodeForSnapshotObjectId(snapshotObjectId) {
+ for (var it = this._allNodes(); it.hasNext(); it.next()) {
+ if (it.node.id() === snapshotObjectId)
+ return it.node;
+ }
+ return null;
+ }
+
+ /**
+ * @param {string} snapshotObjectId
+ * @return {?string}
+ */
+ nodeClassName(snapshotObjectId) {
+ var node = this._nodeForSnapshotObjectId(snapshotObjectId);
+ if (node)
+ return node.className();
+ return null;
+ }
+
+ /**
+ * @param {string} name
+ * @return {!Array.<number>}
+ */
+ idsOfObjectsWithName(name) {
+ var ids = [];
+ for (var it = this._allNodes(); it.hasNext(); it.next()) {
+ if (it.item().name() === name)
+ ids.push(it.item().id());
+ }
+ return ids;
+ }
+
+ /**
+ * @param {number} nodeIndex
+ * @return {!WebInspector.HeapSnapshotEdgesProvider}
+ */
+ createEdgesProvider(nodeIndex) {
+ var node = this.createNode(nodeIndex);
+ var filter = this.containmentEdgesFilter();
+ var indexProvider = new WebInspector.HeapSnapshotEdgeIndexProvider(this);
+ return new WebInspector.HeapSnapshotEdgesProvider(this, filter, node.edges(), indexProvider);
+ }
+
+ /**
+ * @param {number} nodeIndex
+ * @param {?function(!WebInspector.HeapSnapshotEdge):boolean} filter
+ * @return {!WebInspector.HeapSnapshotEdgesProvider}
+ */
+ createEdgesProviderForTest(nodeIndex, filter) {
+ var node = this.createNode(nodeIndex);
+ var indexProvider = new WebInspector.HeapSnapshotEdgeIndexProvider(this);
+ return new WebInspector.HeapSnapshotEdgesProvider(this, filter, node.edges(), indexProvider);
+ }
+
+ /**
+ * @return {?function(!WebInspector.HeapSnapshotEdge):boolean}
+ */
+ retainingEdgesFilter() {
+ return null;
+ }
+
+ /**
+ * @return {?function(!WebInspector.HeapSnapshotEdge):boolean}
+ */
+ containmentEdgesFilter() {
+ return null;
+ }
+
+ /**
+ * @param {number} nodeIndex
+ * @return {!WebInspector.HeapSnapshotEdgesProvider}
+ */
+ createRetainingEdgesProvider(nodeIndex) {
+ var node = this.createNode(nodeIndex);
+ var filter = this.retainingEdgesFilter();
+ var indexProvider = new WebInspector.HeapSnapshotRetainerEdgeIndexProvider(this);
+ return new WebInspector.HeapSnapshotEdgesProvider(this, filter, node.retainers(), indexProvider);
+ }
+
+ /**
+ * @param {string} baseSnapshotId
+ * @param {string} className
+ * @return {!WebInspector.HeapSnapshotNodesProvider}
+ */
+ createAddedNodesProvider(baseSnapshotId, className) {
+ var snapshotDiff = this._snapshotDiffs[baseSnapshotId];
+ var diffForClass = snapshotDiff[className];
+ return new WebInspector.HeapSnapshotNodesProvider(this, null, diffForClass.addedIndexes);
+ }
+
+ /**
+ * @param {!Array.<number>} nodeIndexes
+ * @return {!WebInspector.HeapSnapshotNodesProvider}
+ */
+ createDeletedNodesProvider(nodeIndexes) {
+ return new WebInspector.HeapSnapshotNodesProvider(this, null, nodeIndexes);
+ }
+
+ /**
+ * @return {?function(!WebInspector.HeapSnapshotNode):boolean}
+ */
+ classNodesFilter() {
+ return null;
+ }
+
+ /**
+ * @param {string} className
+ * @param {!WebInspector.HeapSnapshotCommon.NodeFilter} nodeFilter
+ * @return {!WebInspector.HeapSnapshotNodesProvider}
+ */
+ createNodesProviderForClass(className, nodeFilter) {
+ return new WebInspector.HeapSnapshotNodesProvider(
+ this, this.classNodesFilter(), this.aggregatesWithFilter(nodeFilter)[className].idxs);
+ }
+
+ /**
+ * @return {number}
+ */
+ _maxJsNodeId() {
+ var nodeFieldCount = this._nodeFieldCount;
+ var nodes = this.nodes;
+ var nodesLength = nodes.length;
+ var id = 0;
+ for (var nodeIndex = this._nodeIdOffset; nodeIndex < nodesLength; nodeIndex += nodeFieldCount) {
+ var nextId = nodes[nodeIndex];
+ // JS objects have odd ids, skip native objects.
+ if (nextId % 2 === 0)
+ continue;
+ if (id < nextId)
+ id = nextId;
+ }
+ return id;
+ }
+
+ /**
+ * @return {!WebInspector.HeapSnapshotCommon.StaticData}
+ */
+ updateStaticData() {
+ return new WebInspector.HeapSnapshotCommon.StaticData(
+ this.nodeCount, this._rootNodeIndex, this.totalSize, this._maxJsNodeId());
+ }
+};
- /**
- * @return {number}
- */
- _maxJsNodeId: function()
- {
- var nodeFieldCount = this._nodeFieldCount;
- var nodes = this.nodes;
- var nodesLength = nodes.length;
- var id = 0;
- for (var nodeIndex = this._nodeIdOffset; nodeIndex < nodesLength; nodeIndex += nodeFieldCount) {
- var nextId = nodes[nodeIndex];
- // JS objects have odd ids, skip native objects.
- if (nextId % 2 === 0)
- continue;
- if (id < nextId)
- id = nextId;
- }
- return id;
- },
+/**
+ * @unrestricted
+ */
+var HeapSnapshotMetainfo = class {
+ constructor() {
+ // New format.
+ this.node_fields = [];
+ this.node_types = [];
+ this.edge_fields = [];
+ this.edge_types = [];
+ this.trace_function_info_fields = [];
+ this.trace_node_fields = [];
+ this.sample_fields = [];
+ this.type_strings = {};
+ }
+};
- /**
- * @return {!WebInspector.HeapSnapshotCommon.StaticData}
- */
- updateStaticData: function()
- {
- return new WebInspector.HeapSnapshotCommon.StaticData(this.nodeCount, this._rootNodeIndex, this.totalSize, this._maxJsNodeId());
- }
+/**
+ * @unrestricted
+ */
+var HeapSnapshotHeader = class {
+ constructor() {
+ // New format.
+ this.title = '';
+ this.meta = new HeapSnapshotMetainfo();
+ this.node_count = 0;
+ this.edge_count = 0;
+ this.trace_function_count = 0;
+ }
};
/**
- * @constructor
- * @param {!WebInspector.HeapSnapshotItemIterator} iterator
- * @param {!WebInspector.HeapSnapshotItemIndexProvider} indexProvider
+ * @unrestricted
*/
-WebInspector.HeapSnapshotItemProvider = function(iterator, indexProvider)
-{
+WebInspector.HeapSnapshotItemProvider = class {
+ /**
+ * @param {!WebInspector.HeapSnapshotItemIterator} iterator
+ * @param {!WebInspector.HeapSnapshotItemIndexProvider} indexProvider
+ */
+ constructor(iterator, indexProvider) {
this._iterator = iterator;
this._indexProvider = indexProvider;
this._isEmpty = !iterator.hasNext();
@@ -2255,263 +2148,252 @@ WebInspector.HeapSnapshotItemProvider = function(iterator, indexProvider)
this._currentComparator = null;
this._sortedPrefixLength = 0;
this._sortedSuffixLength = 0;
-};
-
-WebInspector.HeapSnapshotItemProvider.prototype = {
- _createIterationOrder: function()
- {
- if (this._iterationOrder)
- return;
- this._iterationOrder = [];
- for (var iterator = this._iterator; iterator.hasNext(); iterator.next())
- this._iterationOrder.push(iterator.item().itemIndex());
- },
-
- /**
- * @return {boolean}
- */
- isEmpty: function()
- {
- return this._isEmpty;
- },
-
- /**
- * @param {number} begin
- * @param {number} end
- * @return {!WebInspector.HeapSnapshotCommon.ItemsRange}
- */
- serializeItemsRange: function(begin, end)
- {
- this._createIterationOrder();
- if (begin > end)
- throw new Error("Start position > end position: " + begin + " > " + end);
- if (end > this._iterationOrder.length)
- end = this._iterationOrder.length;
- if (this._sortedPrefixLength < end && begin < this._iterationOrder.length - this._sortedSuffixLength) {
- this.sort(this._currentComparator, this._sortedPrefixLength, this._iterationOrder.length - 1 - this._sortedSuffixLength, begin, end - 1);
- if (begin <= this._sortedPrefixLength)
- this._sortedPrefixLength = end;
- if (end >= this._iterationOrder.length - this._sortedSuffixLength)
- this._sortedSuffixLength = this._iterationOrder.length - begin;
- }
- var position = begin;
- var count = end - begin;
- var result = new Array(count);
- for (var i = 0 ; i < count; ++i) {
- var itemIndex = this._iterationOrder[position++];
- var item = this._indexProvider.itemForIndex(itemIndex);
- result[i] = item.serialize();
- }
- return new WebInspector.HeapSnapshotCommon.ItemsRange(begin, end, this._iterationOrder.length, result);
- },
-
- sortAndRewind: function(comparator)
- {
- this._currentComparator = comparator;
- this._sortedPrefixLength = 0;
- this._sortedSuffixLength = 0;
+ }
+
+ _createIterationOrder() {
+ if (this._iterationOrder)
+ return;
+ this._iterationOrder = [];
+ for (var iterator = this._iterator; iterator.hasNext(); iterator.next())
+ this._iterationOrder.push(iterator.item().itemIndex());
+ }
+
+ /**
+ * @return {boolean}
+ */
+ isEmpty() {
+ return this._isEmpty;
+ }
+
+ /**
+ * @param {number} begin
+ * @param {number} end
+ * @return {!WebInspector.HeapSnapshotCommon.ItemsRange}
+ */
+ serializeItemsRange(begin, end) {
+ this._createIterationOrder();
+ if (begin > end)
+ throw new Error('Start position > end position: ' + begin + ' > ' + end);
+ if (end > this._iterationOrder.length)
+ end = this._iterationOrder.length;
+ if (this._sortedPrefixLength < end && begin < this._iterationOrder.length - this._sortedSuffixLength) {
+ this.sort(
+ this._currentComparator, this._sortedPrefixLength, this._iterationOrder.length - 1 - this._sortedSuffixLength,
+ begin, end - 1);
+ if (begin <= this._sortedPrefixLength)
+ this._sortedPrefixLength = end;
+ if (end >= this._iterationOrder.length - this._sortedSuffixLength)
+ this._sortedSuffixLength = this._iterationOrder.length - begin;
}
+ var position = begin;
+ var count = end - begin;
+ var result = new Array(count);
+ for (var i = 0; i < count; ++i) {
+ var itemIndex = this._iterationOrder[position++];
+ var item = this._indexProvider.itemForIndex(itemIndex);
+ result[i] = item.serialize();
+ }
+ return new WebInspector.HeapSnapshotCommon.ItemsRange(begin, end, this._iterationOrder.length, result);
+ }
+
+ sortAndRewind(comparator) {
+ this._currentComparator = comparator;
+ this._sortedPrefixLength = 0;
+ this._sortedSuffixLength = 0;
+ }
};
/**
- * @constructor
- * @extends {WebInspector.HeapSnapshotItemProvider}
- * @param {!WebInspector.HeapSnapshot} snapshot
- * @param {?function(!WebInspector.HeapSnapshotEdge):boolean} filter
- * @param {!WebInspector.HeapSnapshotEdgeIterator} edgesIter
- * @param {!WebInspector.HeapSnapshotItemIndexProvider} indexProvider
+ * @unrestricted
*/
-WebInspector.HeapSnapshotEdgesProvider = function(snapshot, filter, edgesIter, indexProvider)
-{
- var iter = filter ? new WebInspector.HeapSnapshotFilteredIterator(edgesIter, /** @type {function(!WebInspector.HeapSnapshotItem):boolean} */ (filter)) : edgesIter;
- WebInspector.HeapSnapshotItemProvider.call(this, iter, indexProvider);
+WebInspector.HeapSnapshotEdgesProvider = class extends WebInspector.HeapSnapshotItemProvider {
+ /**
+ * @param {!WebInspector.HeapSnapshot} snapshot
+ * @param {?function(!WebInspector.HeapSnapshotEdge):boolean} filter
+ * @param {!WebInspector.HeapSnapshotEdgeIterator} edgesIter
+ * @param {!WebInspector.HeapSnapshotItemIndexProvider} indexProvider
+ */
+ constructor(snapshot, filter, edgesIter, indexProvider) {
+ var iter = filter ?
+ new WebInspector.HeapSnapshotFilteredIterator(
+ edgesIter, /** @type {function(!WebInspector.HeapSnapshotItem):boolean} */ (filter)) :
+ edgesIter;
+ super(iter, indexProvider);
this.snapshot = snapshot;
-};
-
-WebInspector.HeapSnapshotEdgesProvider.prototype = {
- /**
- * @param {!WebInspector.HeapSnapshotCommon.ComparatorConfig} comparator
- * @param {number} leftBound
- * @param {number} rightBound
- * @param {number} windowLeft
- * @param {number} windowRight
- */
- sort: function(comparator, leftBound, rightBound, windowLeft, windowRight)
- {
- 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 = this.snapshot.createNode();
- var nodeB = this.snapshot.createNode();
-
- function compareEdgeFieldName(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;
- }
+ }
+
+ /**
+ * @param {!WebInspector.HeapSnapshotCommon.ComparatorConfig} comparator
+ * @param {number} leftBound
+ * @param {number} rightBound
+ * @param {number} windowLeft
+ * @param {number} windowRight
+ */
+ sort(comparator, leftBound, rightBound, windowLeft, windowRight) {
+ 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 = this.snapshot.createNode();
+ var nodeB = this.snapshot.createNode();
+
+ function compareEdgeFieldName(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 compareNodeField(fieldName, ascending, indexA, indexB)
- {
- edgeA.edgeIndex = indexA;
- nodeA.nodeIndex = edgeA.nodeIndex();
- var valueA = nodeA[fieldName]();
+ function compareNodeField(fieldName, ascending, indexA, indexB) {
+ edgeA.edgeIndex = indexA;
+ nodeA.nodeIndex = edgeA.nodeIndex();
+ var valueA = nodeA[fieldName]();
- edgeB.edgeIndex = indexB;
- nodeB.nodeIndex = edgeB.nodeIndex();
- var valueB = nodeB[fieldName]();
+ edgeB.edgeIndex = indexB;
+ nodeB.nodeIndex = edgeB.nodeIndex();
+ var valueB = nodeB[fieldName]();
- var result = valueA < valueB ? -1 : (valueA > valueB ? 1 : 0);
- return ascending ? result : -result;
- }
-
- function compareEdgeAndNode(indexA, indexB) {
- var result = compareEdgeFieldName(ascending1, indexA, indexB);
- if (result === 0)
- result = compareNodeField(fieldName2, ascending2, indexA, indexB);
- if (result === 0)
- return indexA - indexB;
- return result;
- }
+ var result = valueA < valueB ? -1 : (valueA > valueB ? 1 : 0);
+ return ascending ? result : -result;
+ }
- function compareNodeAndEdge(indexA, indexB) {
- var result = compareNodeField(fieldName1, ascending1, indexA, indexB);
- if (result === 0)
- result = compareEdgeFieldName(ascending2, indexA, indexB);
- if (result === 0)
- return indexA - indexB;
- return result;
- }
+ function compareEdgeAndNode(indexA, indexB) {
+ var result = compareEdgeFieldName(ascending1, indexA, indexB);
+ if (result === 0)
+ result = compareNodeField(fieldName2, ascending2, indexA, indexB);
+ if (result === 0)
+ return indexA - indexB;
+ return result;
+ }
- function compareNodeAndNode(indexA, indexB) {
- var result = compareNodeField(fieldName1, ascending1, indexA, indexB);
- if (result === 0)
- result = compareNodeField(fieldName2, ascending2, indexA, indexB);
- if (result === 0)
- return indexA - indexB;
- return result;
- }
+ function compareNodeAndEdge(indexA, indexB) {
+ var result = compareNodeField(fieldName1, ascending1, indexA, indexB);
+ if (result === 0)
+ result = compareEdgeFieldName(ascending2, indexA, indexB);
+ if (result === 0)
+ return indexA - indexB;
+ return result;
+ }
- if (fieldName1 === "!edgeName")
- this._iterationOrder.sortRange(compareEdgeAndNode, leftBound, rightBound, windowLeft, windowRight);
- else if (fieldName2 === "!edgeName")
- this._iterationOrder.sortRange(compareNodeAndEdge, leftBound, rightBound, windowLeft, windowRight);
- else
- this._iterationOrder.sortRange(compareNodeAndNode, leftBound, rightBound, windowLeft, windowRight);
- },
+ function compareNodeAndNode(indexA, indexB) {
+ var result = compareNodeField(fieldName1, ascending1, indexA, indexB);
+ if (result === 0)
+ result = compareNodeField(fieldName2, ascending2, indexA, indexB);
+ if (result === 0)
+ return indexA - indexB;
+ return result;
+ }
- __proto__: WebInspector.HeapSnapshotItemProvider.prototype
+ if (fieldName1 === '!edgeName')
+ this._iterationOrder.sortRange(compareEdgeAndNode, leftBound, rightBound, windowLeft, windowRight);
+ else if (fieldName2 === '!edgeName')
+ this._iterationOrder.sortRange(compareNodeAndEdge, leftBound, rightBound, windowLeft, windowRight);
+ else
+ this._iterationOrder.sortRange(compareNodeAndNode, leftBound, rightBound, windowLeft, windowRight);
+ }
};
-
/**
- * @constructor
- * @extends {WebInspector.HeapSnapshotItemProvider}
- * @param {!WebInspector.HeapSnapshot} snapshot
- * @param {?function(!WebInspector.HeapSnapshotNode):boolean} filter
- * @param {(!Array.<number>|!Uint32Array)} nodeIndexes
+ * @unrestricted
*/
-WebInspector.HeapSnapshotNodesProvider = function(snapshot, filter, nodeIndexes)
-{
+WebInspector.HeapSnapshotNodesProvider = class extends WebInspector.HeapSnapshotItemProvider {
+ /**
+ * @param {!WebInspector.HeapSnapshot} snapshot
+ * @param {?function(!WebInspector.HeapSnapshotNode):boolean} filter
+ * @param {(!Array.<number>|!Uint32Array)} nodeIndexes
+ */
+ constructor(snapshot, filter, nodeIndexes) {
var indexProvider = new WebInspector.HeapSnapshotNodeIndexProvider(snapshot);
var it = new WebInspector.HeapSnapshotIndexRangeIterator(indexProvider, nodeIndexes);
if (filter)
- it = new WebInspector.HeapSnapshotFilteredIterator(it, /** @type {function(!WebInspector.HeapSnapshotItem):boolean} */ (filter));
- WebInspector.HeapSnapshotItemProvider.call(this, it, indexProvider);
+ it = new WebInspector.HeapSnapshotFilteredIterator(
+ it, /** @type {function(!WebInspector.HeapSnapshotItem):boolean} */ (filter));
+ super(it, indexProvider);
this.snapshot = snapshot;
-};
-
-WebInspector.HeapSnapshotNodesProvider.prototype = {
- /**
- * @param {string} snapshotObjectId
+ }
+
+ /**
+ * @param {string} snapshotObjectId
+ * @return {number}
+ */
+ nodePosition(snapshotObjectId) {
+ this._createIterationOrder();
+ var node = this.snapshot.createNode();
+ for (var i = 0; i < this._iterationOrder.length; i++) {
+ node.nodeIndex = this._iterationOrder[i];
+ if (node.id() === snapshotObjectId)
+ break;
+ }
+ if (i === this._iterationOrder.length)
+ return -1;
+ var targetNodeIndex = this._iterationOrder[i];
+ var smallerCount = 0;
+ var compare = this._buildCompareFunction(this._currentComparator);
+ for (var i = 0; i < this._iterationOrder.length; i++) {
+ if (compare(this._iterationOrder[i], targetNodeIndex) < 0)
+ ++smallerCount;
+ }
+ return smallerCount;
+ }
+
+ /**
+ * @return {function(number,number):number}
+ */
+ _buildCompareFunction(comparator) {
+ var nodeA = this.snapshot.createNode();
+ var nodeB = this.snapshot.createNode();
+ var fieldAccessor1 = nodeA[comparator.fieldName1];
+ var fieldAccessor2 = nodeA[comparator.fieldName2];
+ var ascending1 = comparator.ascending1 ? 1 : -1;
+ var ascending2 = comparator.ascending2 ? 1 : -1;
+
+ /**
+ * @param {function():*} fieldAccessor
+ * @param {number} ascending
* @return {number}
*/
- nodePosition: function(snapshotObjectId)
- {
- this._createIterationOrder();
- var node = this.snapshot.createNode();
- for (var i = 0; i < this._iterationOrder.length; i++) {
- node.nodeIndex = this._iterationOrder[i];
- if (node.id() === snapshotObjectId)
- break;
- }
- if (i === this._iterationOrder.length)
- return -1;
- var targetNodeIndex = this._iterationOrder[i];
- var smallerCount = 0;
- var compare = this._buildCompareFunction(this._currentComparator);
- for (var i = 0; i < this._iterationOrder.length; i++) {
- if (compare(this._iterationOrder[i], targetNodeIndex) < 0)
- ++smallerCount;
- }
- return smallerCount;
- },
-
- /**
- * @return {function(number,number):number}
- */
- _buildCompareFunction: function(comparator)
- {
- var nodeA = this.snapshot.createNode();
- var nodeB = this.snapshot.createNode();
- var fieldAccessor1 = nodeA[comparator.fieldName1];
- var fieldAccessor2 = nodeA[comparator.fieldName2];
- var ascending1 = comparator.ascending1 ? 1 : -1;
- var ascending2 = comparator.ascending2 ? 1 : -1;
-
- /**
- * @param {function():*} fieldAccessor
- * @param {number} ascending
- * @return {number}
- */
- function sortByNodeField(fieldAccessor, ascending)
- {
- var valueA = fieldAccessor.call(nodeA);
- var valueB = fieldAccessor.call(nodeB);
- return valueA < valueB ? -ascending : (valueA > valueB ? ascending : 0);
- }
-
- /**
- * @param {number} indexA
- * @param {number} indexB
- * @return {number}
- */
- function sortByComparator(indexA, indexB)
- {
- nodeA.nodeIndex = indexA;
- nodeB.nodeIndex = indexB;
- var result = sortByNodeField(fieldAccessor1, ascending1);
- if (result === 0)
- result = sortByNodeField(fieldAccessor2, ascending2);
- return result || indexA - indexB;
- }
-
- return sortByComparator;
- },
+ function sortByNodeField(fieldAccessor, ascending) {
+ var valueA = fieldAccessor.call(nodeA);
+ var valueB = fieldAccessor.call(nodeB);
+ return valueA < valueB ? -ascending : (valueA > valueB ? ascending : 0);
+ }
/**
- * @param {!WebInspector.HeapSnapshotCommon.ComparatorConfig} comparator
- * @param {number} leftBound
- * @param {number} rightBound
- * @param {number} windowLeft
- * @param {number} windowRight
+ * @param {number} indexA
+ * @param {number} indexB
+ * @return {number}
*/
- sort: function(comparator, leftBound, rightBound, windowLeft, windowRight)
- {
- this._iterationOrder.sortRange(this._buildCompareFunction(comparator), leftBound, rightBound, windowLeft, windowRight);
- },
+ function sortByComparator(indexA, indexB) {
+ nodeA.nodeIndex = indexA;
+ nodeB.nodeIndex = indexB;
+ var result = sortByNodeField(fieldAccessor1, ascending1);
+ if (result === 0)
+ result = sortByNodeField(fieldAccessor2, ascending2);
+ return result || indexA - indexB;
+ }
- __proto__: WebInspector.HeapSnapshotItemProvider.prototype
+ return sortByComparator;
+ }
+
+ /**
+ * @param {!WebInspector.HeapSnapshotCommon.ComparatorConfig} comparator
+ * @param {number} leftBound
+ * @param {number} rightBound
+ * @param {number} windowLeft
+ * @param {number} windowRight
+ */
+ sort(comparator, leftBound, rightBound, windowLeft, windowRight) {
+ this._iterationOrder.sortRange(
+ this._buildCompareFunction(comparator), leftBound, rightBound, windowLeft, windowRight);
+ }
};
-

Powered by Google App Engine
This is Rietveld 408576698