// Copyright (c) 2012 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
'use strict';
/**
* @fileoverview Provides the AsyncSliceGroup class.
*/
base.require('range');
base.require('model.slice');
base.exportTo('tracing.model', function() {
var Slice = tracing.model.Slice;
/**
* A AsyncSlice represents an interval of time during which an
* asynchronous operation is in progress. An AsyncSlice consumes no CPU time
* itself and so is only associated with Threads at its start and end point.
*
* @constructor
*/
function AsyncSlice(category, title, colorId, start, args) {
Slice.call(this, category, title, colorId, start, args);
};
AsyncSlice.prototype = {
__proto__: Slice.prototype,
toJSON: function() {
var obj = new Object();
var keys = Object.keys(this);
for (var i = 0; i < keys.length; i++) {
var key = keys[i];
if (typeof this[key] == 'function')
continue;
if (key == 'startThread' || key == 'endThread') {
obj[key] = this[key].guid;
continue;
}
obj[key] = this[key];
}
return obj;
},
id: undefined,
startThread: undefined,
endThread: undefined,
subSlices: undefined
};
/**
* A group of AsyncSlices.
* @constructor
*/
function AsyncSliceGroup() {
this.slices = [];
this.bounds = new base.Range();
}
AsyncSliceGroup.prototype = {
__proto__: Object.prototype,
/**
* Helper function that pushes the provided slice onto the slices array.
*/
push: function(slice) {
this.slices.push(slice);
},
/**
* @return {Number} The number of slices in this group.
*/
get length() {
return this.slices.length;
},
/**
* Shifts all the timestamps inside this group forward by the amount
* specified.
*/
shiftTimestampsForward: function(amount) {
for (var sI = 0; sI < this.slices.length; sI++) {
var slice = this.slices[sI];
slice.start = (slice.start + amount);
for (var sJ = 0; sJ < slice.subSlices.length; sJ++)
slice.subSlices[sJ].start += amount;
}
},
/**
* Updates the bounds for this group based on the slices it contains.
*/
updateBounds: function() {
this.bounds.reset();
for (var i = 0; i < this.slices.length; i++) {
this.bounds.addValue(this.slices[i].start);
this.bounds.addValue(this.slices[i].end);
}
},
/**
* Breaks up this group into slices based on start thread.
*
* @return {Array} An array of AsyncSliceGroups where each group has
* slices that started on the same thread.
*/
computeSubGroups: function() {
var subGroupsByGUID = {};
for (var i = 0; i < this.slices.length; ++i) {
var slice = this.slices[i];
var sliceGUID = slice.startThread.guid;
if (!subGroupsByGUID[sliceGUID])
subGroupsByGUID[sliceGUID] = new AsyncSliceGroup();
subGroupsByGUID[sliceGUID].slices.push(slice);
}
var groups = [];
for (var guid in subGroupsByGUID) {
var group = subGroupsByGUID[guid];
group.updateBounds();
groups.push(group);
}
return groups;
}
};
return {
AsyncSlice: AsyncSlice,
AsyncSliceGroup: AsyncSliceGroup
};
});