// 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.
/**
* @fileoverview TraceEventImporter imports TraceEvent-formatted data
* into the provided timeline model.
*/
base.require('timeline_model');
base.require('timeline_color_scheme');
base.exportTo('tracing', function() {
function TraceEventImporter(model, eventData) {
this.importPriority = 1;
this.model_ = model;
if (typeof(eventData) === 'string' || eventData instanceof String) {
// If the event data begins with a [, then we know it should end with a ].
// The reason we check for this is because some tracing implementations
// cannot guarantee that a ']' gets written to the trace file. So, we are
// forgiving and if this is obviously the case, we fix it up before
// throwing the string at JSON.parse.
if (eventData[0] == '[') {
n = eventData.length;
if (eventData[n - 1] == '\n') {
eventData = eventData.substring(0, n - 1);
n--;
if (eventData[n - 1] == '\r') {
eventData = eventData.substring(0, n - 1);
n--;
}
}
if (eventData[n - 1] == ',')
eventData = eventData.substring(0, n - 1);
if (eventData[n - 1] != ']')
eventData = eventData + ']';
}
this.events_ = JSON.parse(eventData);
} else {
this.events_ = eventData;
}
// Some trace_event implementations put the actual trace events
// inside a container. E.g { ... , traceEvents: [ ] }
// If we see that, just pull out the trace events.
if (this.events_.traceEvents) {
this.events_ = this.events_.traceEvents;
for (fieldName in this.events_) {
if (fieldName == 'traceEvents')
continue;
this.model_.metadata.push({name: fieldName,
value: this.events_[fieldName]});
}
}
// Async events need to be processed durign finalizeEvents
this.allAsyncEvents_ = [];
}
/**
* @return {boolean} Whether obj is a TraceEvent array.
*/
TraceEventImporter.canImport = function(eventData) {
// May be encoded JSON. But we dont want to parse it fully yet.
// Use a simple heuristic:
// - eventData that starts with [ are probably trace_event
// - eventData that starts with { are probably trace_event
// May be encoded JSON. Treat files that start with { as importable by us.
if (typeof(eventData) === 'string' || eventData instanceof String) {
return eventData[0] == '{' || eventData[0] == '[';
}
// Might just be an array of events
if (eventData instanceof Array && eventData.length && eventData[0].ph)
return true;
// Might be an object with a traceEvents field in it.
if (eventData.traceEvents)
return eventData.traceEvents instanceof Array &&
eventData.traceEvents[0].ph;
return false;
};
TraceEventImporter.prototype = {
__proto__: Object.prototype,
/**
* Helper to process an 'async finish' event, which will close an open slice
* on a TimelineAsyncSliceGroup object.
*/
processAsyncEvent: function(index, event) {
var thread = this.model_.getOrCreateProcess(event.pid).
getOrCreateThread(event.tid);
this.allAsyncEvents_.push({
event: event,
thread: thread});
},
/**
* Helper that creates and adds samples to a TimelineCounter object based on
* 'C' phase events.
*/
processCounterEvent: function(event) {
var ctr_name;
if (event.id !== undefined)
ctr_name = event.name + '[' + event.id + ']';
else
ctr_name = event.name;
var ctr = this.model_.getOrCreateProcess(event.pid)
.getOrCreateCounter(event.cat, ctr_name);
// Initialize the counter's series fields if needed.
if (ctr.numSeries == 0) {
for (var seriesName in event.args) {
ctr.seriesNames.push(seriesName);
ctr.seriesColors.push(
tracing.getStringColorId(ctr.name + '.' + seriesName));
}
if (ctr.numSeries == 0) {
this.model_.importErrors.push('Expected counter ' + event.name +
' to have at least one argument to use as a value.');
// Drop the counter.
delete ctr.parent.counters[ctr.name];
return;
}
}
// Add the sample values.
ctr.timestamps.push(event.ts / 1000);
for (var i = 0; i < ctr.numSeries; i++) {
var seriesName = ctr.seriesNames[i];
if (event.args[seriesName] === undefined) {
ctr.samples.push(0);
continue;
}
ctr.samples.push(event.args[seriesName]);
}
},
/**
* Walks through the events_ list and outputs the structures discovered to
* model_.
*/
importEvents: function() {
// Walk through events
var events = this.events_;
// Some events cannot be handled until we have done a first pass over the
// data set. So, accumulate them into a temporary data structure.
var second_pass_events = [];
for (var eI = 0; eI < events.length; eI++) {
var event = events[eI];
if (event.ph == 'B') {
var thread = this.model_.getOrCreateProcess(event.pid)
.getOrCreateThread(event.tid);
if (!thread.isTimestampValidForBeginOrEnd(event.ts / 1000)) {
this.model_.importErrors.push(
'Timestamps are moving backward.');
continue;
}
thread.beginSlice(event.cat, event.name, event.ts / 1000, event.args);
} else if (event.ph == 'E') {
var thread = this.model_.getOrCreateProcess(event.pid)
.getOrCreateThread(event.tid);
if (!thread.isTimestampValidForBeginOrEnd(event.ts / 1000)) {
this.model_.importErrors.push(
'Timestamps are moving backward.');
continue;
}
if (!thread.openSliceCount) {
this.model_.importErrors.push(
'E phase event without a matching B phase event.');
continue;
}
var slice = thread.endSlice(event.ts / 1000);
for (var arg in event.args) {
if (slice.args[arg] !== undefined) {
this.model_.importErrors.push(
'Both the B and E phases of ' + slice.name +
'provided values for argument ' + arg + '. ' +
'The value of the E phase event will be used.');
}
slice.args[arg] = event.args[arg];
}
} else if (event.ph == 'S') {
this.processAsyncEvent(eI, event);
} else if (event.ph == 'F') {
this.processAsyncEvent(eI, event);
} else if (event.ph == 'T') {
this.processAsyncEvent(eI, event);
} else if (event.ph == 'I') {
// Treat an Instant event as a duration 0 slice.
// TimelineSliceTrack's redraw() knows how to handle this.
var thread = this.model_.getOrCreateProcess(event.pid)
.getOrCreateThread(event.tid);
thread.beginSlice(event.cat, event.name, event.ts / 1000, event.args);
thread.endSlice(event.ts / 1000);
} else if (event.ph == 'C') {
this.processCounterEvent(event);
} else if (event.ph == 'M') {
if (event.name == 'thread_name') {
var thread = this.model_.getOrCreateProcess(event.pid)
.getOrCreateThread(event.tid);
thread.name = event.args.name;
} else {
this.model_.importErrors.push(
'Unrecognized metadata name: ' + event.name);
}
} else if (event.ph == 's') {
// NB: toss until there's proper support
} else if (event.ph == 't') {
// NB: toss until there's proper support
} else if (event.ph == 'f') {
// NB: toss until there's proper support
} else {
this.model_.importErrors.push(
'Unrecognized event phase: ' + event.ph +
'(' + event.name + ')');
}
}
},
/**
* Called by the TimelineModel after all other importers have imported their
* events.
*/
finalizeImport: function() {
this.createAsyncSlices_();
},
createAsyncSlices_: function() {
if (this.allAsyncEvents_.length == 0)
return;
this.allAsyncEvents_.sort(function(x, y) {
return x.event.ts - y.event.ts;
});
var asyncEventStatesByNameThenID = {};
var allAsyncEvents = this.allAsyncEvents_;
for (var i = 0; i < allAsyncEvents.length; i++) {
var asyncEventState = allAsyncEvents[i];
var event = asyncEventState.event;
var name = event.name;
if (name === undefined) {
this.model_.importErrors.push(
'Async events (ph: S, T or F) require an name parameter.');
continue;
}
var id = event.id;
if (id === undefined) {
this.model_.importErrors.push(
'Async events (ph: S, T or F) require an id parameter.');
continue;
}
// TODO(simonjam): Add a synchronous tick on the appropriate thread.
if (event.ph == 'S') {
if (asyncEventStatesByNameThenID[name] === undefined)
asyncEventStatesByNameThenID[name] = {};
if (asyncEventStatesByNameThenID[name][id]) {
this.model_.importErrors.push(
'At ' + event.ts + ', a slice of the same id ' + id +
' was alrady open.');
continue;
}
asyncEventStatesByNameThenID[name][id] = [];
asyncEventStatesByNameThenID[name][id].push(asyncEventState);
} else {
if (asyncEventStatesByNameThenID[name] === undefined) {
this.model_.importErrors.push(
'At ' + event.ts + ', no slice named ' + name +
' was open.');
continue;
}
if (asyncEventStatesByNameThenID[name][id] === undefined) {
this.model_.importErrors.push(
'At ' + event.ts + ', no slice named ' + name +
' with id=' + id + ' was open.');
continue;
}
var events = asyncEventStatesByNameThenID[name][id];
events.push(asyncEventState);
if (event.ph == 'F') {
// Create a slice from start to end.
var slice = new tracing.TimelineAsyncSlice(
events[0].event.cat,
name,
tracing.getStringColorId(name),
events[0].event.ts / 1000);
slice.duration = (event.ts / 1000) - (events[0].event.ts / 1000);
slice.startThread = events[0].thread;
slice.endThread = asyncEventState.thread;
slice.id = id;
slice.args = events[0].event.args;
slice.subSlices = [];
// Create subSlices for each step.
for (var j = 1; j < events.length; ++j) {
var subName = name;
if (events[j - 1].event.ph == 'T')
subName = name + ':' + events[j - 1].event.args.step;
var subSlice = new tracing.TimelineAsyncSlice(
events[0].event.cat,
subName,
tracing.getStringColorId(name + j),
events[j - 1].event.ts / 1000);
subSlice.duration =
(events[j].event.ts / 1000) - (events[j - 1].event.ts / 1000);
subSlice.startThread = events[j - 1].thread;
subSlice.endThread = events[j].thread;
subSlice.id = id;
subSlice.args = events[j - 1].event.args;
slice.subSlices.push(subSlice);
}
// The args for the finish event go in the last subSlice.
var lastSlice = slice.subSlices[slice.subSlices.length - 1];
for (var arg in event.args)
lastSlice.args[arg] = event.args[arg];
// Add |slice| to the start-thread's asyncSlices.
slice.startThread.asyncSlices.push(slice);
delete asyncEventStatesByNameThenID[name][id];
}
}
}
}
};
tracing.TimelineModel.registerImporter(TraceEventImporter);
return {
TraceEventImporter: TraceEventImporter
};
});