2
2
// Use of this source code is governed by a BSD-style license that can be
3
3
// found in the LICENSE file.
4
4
5
- /**
6
- * @constructor
7
- * @extends {WebInspector.ProfileNode }
8
- * @param {!ProfilerAgent.CPUProfileNode } sourceNode
9
- * @param {number } sampleTime
10
- */
11
- WebInspector . CPUProfileNode = function ( sourceNode , sampleTime )
12
- {
13
- WebInspector . ProfileNode . call ( this , sourceNode . functionName , sourceNode . scriptId , sourceNode . url , sourceNode . lineNumber , sourceNode . columnNumber ) ;
14
- this . id = sourceNode . id ;
15
- this . self = sourceNode . hitCount * sampleTime ;
16
- this . callUID = sourceNode . callUID ;
17
- this . positionTicks = sourceNode . positionTicks ;
18
- this . deoptReason = sourceNode . deoptReason ;
19
- // TODO: Remove the following field in favor of this.self
20
- this . selfTime = this . self ;
21
- }
22
-
23
- WebInspector . CPUProfileNode . prototype = {
24
- __proto__ : WebInspector . ProfileNode . prototype
25
- }
26
5
27
6
/**
28
7
* @constructor
29
- * @extends {WebInspector.ProfileTreeModel }
30
8
* @param {!ProfilerAgent.CPUProfile } profile
31
9
*/
32
10
WebInspector . CPUProfileDataModel = function ( profile )
33
11
{
12
+ this . profileHead = profile . head ;
34
13
this . samples = profile . samples ;
35
14
this . timestamps = profile . timestamps ;
36
- // Convert times from sec to msec.
37
15
this . profileStartTime = profile . startTime * 1000 ;
38
16
this . profileEndTime = profile . endTime * 1000 ;
39
- this . totalHitCount = 0 ;
40
- if ( ! WebInspector . moduleSetting ( "showNativeFunctionsInJSProfile" ) . get ( ) )
41
- this . _filterNativeFrames ( profile . head ) ;
42
- this . profileHead = this . _translateProfileTree ( profile . head ) ;
43
- WebInspector . ProfileTreeModel . call ( this , this . profileHead , this . profileStartTime , this . profileEndTime ) ;
44
- this . _extractMetaNodes ( ) ;
17
+ this . _assignParentsInProfile ( ) ;
45
18
if ( this . samples ) {
46
- this . _buildIdToNodeMap ( ) ;
47
19
this . _sortSamples ( ) ;
48
20
this . _normalizeTimestamps ( ) ;
21
+ this . _buildIdToNodeMap ( ) ;
49
22
this . _fixMissingSamples ( ) ;
50
23
}
51
- this . _assignTotalTimes ( this . profileHead ) ;
24
+ if ( ! WebInspector . moduleSetting ( "showNativeFunctionsInJSProfile" ) . get ( ) )
25
+ this . _filterNativeFrames ( ) ;
26
+ this . _assignDepthsInProfile ( ) ;
27
+ this . _calculateTimes ( profile ) ;
52
28
}
53
29
54
30
WebInspector . CPUProfileDataModel . prototype = {
55
31
/**
56
- * @param {!ProfilerAgent.CPUProfileNode } root
32
+ * @param {!ProfilerAgent.CPUProfile } profile
57
33
*/
58
- _filterNativeFrames : function ( root )
34
+ _calculateTimes : function ( profile )
35
+ {
36
+ function totalHitCount ( node ) {
37
+ var result = node . hitCount ;
38
+ for ( var i = 0 ; i < node . children . length ; i ++ )
39
+ result += totalHitCount ( node . children [ i ] ) ;
40
+ return result ;
41
+ }
42
+ profile . totalHitCount = totalHitCount ( profile . head ) ;
43
+ this . totalHitCount = profile . totalHitCount ;
44
+
45
+ var duration = this . profileEndTime - this . profileStartTime ;
46
+ var samplingInterval = duration / profile . totalHitCount ;
47
+ this . samplingInterval = samplingInterval ;
48
+
49
+ function calculateTimesForNode ( node ) {
50
+ node . selfTime = node . hitCount * samplingInterval ;
51
+ var totalHitCount = node . hitCount ;
52
+ for ( var i = 0 ; i < node . children . length ; i ++ )
53
+ totalHitCount += calculateTimesForNode ( node . children [ i ] ) ;
54
+ node . totalTime = totalHitCount * samplingInterval ;
55
+ return totalHitCount ;
56
+ }
57
+ calculateTimesForNode ( profile . head ) ;
58
+ } ,
59
+
60
+ _filterNativeFrames : function ( )
59
61
{
60
- // TODO: get rid of this function and do the filtering while _translateProfileTree
61
62
if ( this . samples ) {
62
- /** @type {!Map<number, !ProfilerAgent.CPUProfileNode> } */
63
- var idToNode = new Map ( ) ;
64
- var stack = [ root ] ;
65
- while ( stack . length ) {
66
- var node = stack . pop ( ) ;
67
- idToNode . set ( node . id , node ) ;
68
- for ( var i = 0 ; i < node . children . length ; i ++ ) {
69
- node . children [ i ] . parent = node ;
70
- stack . push ( node . children [ i ] ) ;
71
- }
72
- }
73
63
for ( var i = 0 ; i < this . samples . length ; ++ i ) {
74
- var node = idToNode . get ( this . samples [ i ] ) ;
64
+ var node = this . nodeByIndex ( i ) ;
75
65
while ( isNativeNode ( node ) )
76
66
node = node . parent ;
77
67
this . samples [ i ] = node . id ;
78
68
}
79
69
}
80
- processSubtree ( root ) ;
70
+ processSubtree ( this . profileHead ) ;
81
71
82
72
/**
83
73
* @param {!ProfilerAgent.CPUProfileNode } node
@@ -128,43 +118,44 @@ WebInspector.CPUProfileDataModel.prototype = {
128
118
}
129
119
} ,
130
120
131
- /**
132
- * @param {!ProfilerAgent.CPUProfileNode } root
133
- * @return {!WebInspector.CPUProfileNode }
134
- */
135
- _translateProfileTree : function ( root )
121
+ _assignParentsInProfile : function ( )
136
122
{
137
- /**
138
- * @param {!ProfilerAgent.CPUProfileNode } node
139
- * @return {number }
140
- */
141
- function computeHitCountForSubtree ( node )
142
- {
143
- return node . children . reduce ( ( acc , node ) => acc + computeHitCountForSubtree ( node ) , node . hitCount ) ;
144
- }
145
- this . totalHitCount = computeHitCountForSubtree ( root ) ;
146
- var sampleTime = ( this . profileEndTime - this . profileStartTime ) / this . totalHitCount ;
147
- var resultRoot = new WebInspector . CPUProfileNode ( root , sampleTime ) ;
148
- var targetNodeStack = [ resultRoot ] ;
149
- var sourceNodeStack = [ root ] ;
150
- while ( sourceNodeStack . length ) {
151
- var sourceNode = sourceNodeStack . pop ( ) ;
152
- var parentNode = targetNodeStack . pop ( ) ;
153
- parentNode . children = sourceNode . children . map ( child => new WebInspector . CPUProfileNode ( child , sampleTime ) ) ;
154
- sourceNodeStack . push . apply ( sourceNodeStack , sourceNode . children ) ;
155
- targetNodeStack . push . apply ( targetNodeStack , parentNode . children ) ;
123
+ var head = this . profileHead ;
124
+ head . parent = null ;
125
+ var nodesToTraverse = [ head ] ;
126
+ while ( nodesToTraverse . length ) {
127
+ var parent = nodesToTraverse . pop ( ) ;
128
+ var children = parent . children ;
129
+ var length = children . length ;
130
+ for ( var i = 0 ; i < length ; ++ i ) {
131
+ var child = children [ i ] ;
132
+ child . parent = parent ;
133
+ if ( child . children . length )
134
+ nodesToTraverse . push ( child ) ;
135
+ }
156
136
}
157
- return resultRoot ;
158
137
} ,
159
138
160
- /**
161
- * @param {!WebInspector.ProfileNode } node
162
- */
163
- _assignTotalTimes : function ( node )
139
+ _assignDepthsInProfile : function ( )
164
140
{
165
- // TODO: get rid of this field in favor of this.total
166
- node . totalTime = node . total ;
167
- node . children . forEach ( this . _assignTotalTimes , this ) ;
141
+ var head = this . profileHead ;
142
+ head . depth = - 1 ;
143
+ this . maxDepth = 0 ;
144
+ var nodesToTraverse = [ head ] ;
145
+ while ( nodesToTraverse . length ) {
146
+ var parent = nodesToTraverse . pop ( ) ;
147
+ var depth = parent . depth + 1 ;
148
+ if ( depth > this . maxDepth )
149
+ this . maxDepth = depth ;
150
+ var children = parent . children ;
151
+ var length = children . length ;
152
+ for ( var i = 0 ; i < length ; ++ i ) {
153
+ var child = children [ i ] ;
154
+ child . depth = depth ;
155
+ if ( child . children . length )
156
+ nodesToTraverse . push ( child ) ;
157
+ }
158
+ }
168
159
} ,
169
160
170
161
_sortSamples : function ( )
@@ -222,7 +213,7 @@ WebInspector.CPUProfileDataModel.prototype = {
222
213
223
214
_buildIdToNodeMap : function ( )
224
215
{
225
- /** @type {!Object<number, !WebInspector .CPUProfileNode> } */
216
+ /** @type {!Object. <number, !ProfilerAgent .CPUProfileNode> } */
226
217
this . _idToNode = { } ;
227
218
var idToNode = this . _idToNode ;
228
219
var stack = [ this . profileHead ] ;
@@ -232,10 +223,7 @@ WebInspector.CPUProfileDataModel.prototype = {
232
223
for ( var i = 0 ; i < node . children . length ; i ++ )
233
224
stack . push ( node . children [ i ] ) ;
234
225
}
235
- } ,
236
226
237
- _extractMetaNodes : function ( )
238
- {
239
227
var topLevelNodes = this . profileHead . children ;
240
228
for ( var i = 0 ; i < topLevelNodes . length && ! ( this . gcNode && this . programNode && this . idleNode ) ; i ++ ) {
241
229
var node = topLevelNodes [ i ] ;
@@ -277,8 +265,8 @@ WebInspector.CPUProfileDataModel.prototype = {
277
265
}
278
266
279
267
/**
280
- * @param {!WebInspector.ProfileNode } node
281
- * @return {!WebInspector.ProfileNode }
268
+ * @param {!ProfilerAgent.CPUProfileNode } node
269
+ * @return {!ProfilerAgent.CPUProfileNode }
282
270
*/
283
271
function bottomNode ( node )
284
272
{
@@ -298,8 +286,8 @@ WebInspector.CPUProfileDataModel.prototype = {
298
286
} ,
299
287
300
288
/**
301
- * @param {function(number, !WebInspector .CPUProfileNode, number) } openFrameCallback
302
- * @param {function(number, !WebInspector .CPUProfileNode, number, number, number) } closeFrameCallback
289
+ * @param {function(number, !ProfilerAgent .CPUProfileNode, number) } openFrameCallback
290
+ * @param {function(number, !ProfilerAgent .CPUProfileNode, number, number, number) } closeFrameCallback
303
291
* @param {number= } startTime
304
292
* @param {number= } stopTime
305
293
*/
@@ -370,7 +358,7 @@ WebInspector.CPUProfileDataModel.prototype = {
370
358
var start = stackStartTimes [ stackTop ] ;
371
359
var duration = sampleTime - start ;
372
360
stackChildrenDuration [ stackTop - 1 ] += duration ;
373
- closeFrameCallback ( prevNode . depth , /** @type { !WebInspector.CPUProfileNode } */ ( prevNode ) , start , duration , duration - stackChildrenDuration [ stackTop ] ) ;
361
+ closeFrameCallback ( prevNode . depth , prevNode , start , duration , duration - stackChildrenDuration [ stackTop ] ) ;
374
362
-- stackTop ;
375
363
if ( node . depth === prevNode . depth ) {
376
364
stackNodes . push ( node ) ;
@@ -402,19 +390,18 @@ WebInspector.CPUProfileDataModel.prototype = {
402
390
var start = stackStartTimes [ stackTop ] ;
403
391
var duration = sampleTime - start ;
404
392
stackChildrenDuration [ stackTop - 1 ] += duration ;
405
- closeFrameCallback ( node . depth , /** @type { !WebInspector.CPUProfileNode } */ ( node ) , start , duration , duration - stackChildrenDuration [ stackTop ] ) ;
393
+ closeFrameCallback ( node . depth , node , start , duration , duration - stackChildrenDuration [ stackTop ] ) ;
406
394
-- stackTop ;
407
395
}
408
396
} ,
409
397
410
398
/**
411
399
* @param {number } index
412
- * @return {!WebInspector .CPUProfileNode }
400
+ * @return {!ProfilerAgent .CPUProfileNode }
413
401
*/
414
402
nodeByIndex : function ( index )
415
403
{
416
404
return this . _idToNode [ this . samples [ index ] ] ;
417
- } ,
405
+ }
418
406
419
- __proto__ : WebInspector . ProfileTreeModel . prototype
420
407
}
0 commit comments