| // 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 { |
| 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 |
| }; |
| }); |