blob: 3e03db97824332b980e2ed002cb1a19495a6939e [file] [log] [blame]
/*
* Copyright (C) 2013 Google Inc. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are
* met:
*
* * Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* * Redistributions in binary form must reproduce the above
* copyright notice, this list of conditions and the following disclaimer
* in the documentation and/or other materials provided with the
* distribution.
* * Neither the name of Google Inc. nor the names of its
* contributors may be used to endorse or promote products derived from
* this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#include "config.h"
#include "core/inspector/InspectorTimelineAgent.h"
#include "core/events/Event.h"
#include "core/frame/LocalDOMWindow.h"
#include "core/frame/FrameConsole.h"
#include "core/frame/FrameHost.h"
#include "core/frame/FrameView.h"
#include "core/frame/LocalFrame.h"
#include "core/inspector/ConsoleMessage.h"
#include "core/inspector/IdentifiersFactory.h"
#include "core/inspector/InspectorClient.h"
#include "core/inspector/InspectorCounters.h"
#include "core/inspector/InspectorInstrumentation.h"
#include "core/inspector/InspectorLayerTreeAgent.h"
#include "core/inspector/InspectorNodeIds.h"
#include "core/inspector/InspectorOverlay.h"
#include "core/inspector/InspectorPageAgent.h"
#include "core/inspector/InspectorState.h"
#include "core/inspector/InstrumentingAgents.h"
#include "core/inspector/ScriptCallStack.h"
#include "core/inspector/TimelineRecordFactory.h"
#include "core/inspector/TraceEventDispatcher.h"
#include "core/loader/DocumentLoader.h"
#include "core/page/Page.h"
#include "core/rendering/RenderObject.h"
#include "core/rendering/RenderView.h"
#include "core/xml/XMLHttpRequest.h"
#include "platform/TraceEvent.h"
#include "platform/graphics/DeferredImageDecoder.h"
#include "platform/graphics/GraphicsLayer.h"
#include "platform/network/ResourceRequest.h"
#include "wtf/CurrentTime.h"
#include "wtf/DateMath.h"
namespace blink {
namespace TimelineAgentState {
static const char enabled[] = "enabled";
static const char started[] = "started";
static const char startedFromProtocol[] = "startedFromProtocol";
static const char timelineMaxCallStackDepth[] = "timelineMaxCallStackDepth";
static const char includeCounters[] = "includeCounters";
static const char includeGPUEvents[] = "includeGPUEvents";
static const char bufferEvents[] = "bufferEvents";
static const char liveEvents[] = "liveEvents";
}
// Must be kept in sync with WebInspector.TimelineModel.RecordType in TimelineModel.js
namespace TimelineRecordType {
static const char Program[] = "Program";
static const char EventDispatch[] = "EventDispatch";
static const char ScheduleStyleRecalculation[] = "ScheduleStyleRecalculation";
static const char RecalculateStyles[] = "RecalculateStyles";
static const char InvalidateLayout[] = "InvalidateLayout";
static const char Layout[] = "Layout";
static const char UpdateLayerTree[] = "UpdateLayerTree";
static const char Paint[] = "Paint";
static const char ScrollLayer[] = "ScrollLayer";
static const char ResizeImage[] = "ResizeImage";
static const char CompositeLayers[] = "CompositeLayers";
static const char ParseHTML[] = "ParseHTML";
static const char TimerInstall[] = "TimerInstall";
static const char TimerRemove[] = "TimerRemove";
static const char TimerFire[] = "TimerFire";
static const char EvaluateScript[] = "EvaluateScript";
static const char MarkLoad[] = "MarkLoad";
static const char MarkDOMContent[] = "MarkDOMContent";
static const char MarkFirstPaint[] = "MarkFirstPaint";
static const char TimeStamp[] = "TimeStamp";
static const char ConsoleTime[] = "ConsoleTime";
static const char ResourceSendRequest[] = "ResourceSendRequest";
static const char ResourceReceiveResponse[] = "ResourceReceiveResponse";
static const char ResourceReceivedData[] = "ResourceReceivedData";
static const char ResourceFinish[] = "ResourceFinish";
static const char XHRReadyStateChange[] = "XHRReadyStateChange";
static const char XHRLoad[] = "XHRLoad";
static const char FunctionCall[] = "FunctionCall";
static const char GCEvent[] = "GCEvent";
static const char UpdateCounters[] = "UpdateCounters";
static const char RequestAnimationFrame[] = "RequestAnimationFrame";
static const char CancelAnimationFrame[] = "CancelAnimationFrame";
static const char FireAnimationFrame[] = "FireAnimationFrame";
static const char WebSocketCreate[] = "WebSocketCreate";
static const char WebSocketSendHandshakeRequest[] = "WebSocketSendHandshakeRequest";
static const char WebSocketReceiveHandshakeResponse[] = "WebSocketReceiveHandshakeResponse";
static const char WebSocketDestroy[] = "WebSocketDestroy";
static const char RequestMainThreadFrame[] = "RequestMainThreadFrame";
static const char ActivateLayerTree[] = "ActivateLayerTree";
static const char DrawFrame[] = "DrawFrame";
static const char BeginFrame[] = "BeginFrame";
static const char DecodeImage[] = "DecodeImage";
static const char GPUTask[] = "GPUTask";
static const char Rasterize[] = "Rasterize";
static const char PaintSetup[] = "PaintSetup";
static const char EmbedderCallback[] = "EmbedderCallback";
}
using TypeBuilder::Timeline::TimelineEvent;
class InspectorTimelineAgentTraceEventListener : public TraceEventDispatcher::TraceEventListener {
public:
typedef void (InspectorTimelineAgent::*TraceEventHandlerMethod)(const TraceEventDispatcher::TraceEvent&);
static PassOwnPtrWillBeRawPtr<InspectorTimelineAgentTraceEventListener> create(InspectorTimelineAgent* instance, TraceEventHandlerMethod method)
{
return adoptPtrWillBeNoop(new InspectorTimelineAgentTraceEventListener(instance, method));
}
virtual void call(const TraceEventDispatcher::TraceEvent& event) OVERRIDE
{
(m_instance->*m_method)(event);
}
virtual void* target() OVERRIDE
{
return m_instance;
}
virtual void trace(Visitor* visitor) OVERRIDE
{
visitor->trace(m_instance);
TraceEventDispatcher::TraceEventListener::trace(visitor);
}
private:
InspectorTimelineAgentTraceEventListener(InspectorTimelineAgent* instance, TraceEventHandlerMethod method)
: m_instance(instance)
, m_method(method)
{
}
RawPtrWillBeMember<InspectorTimelineAgent> m_instance;
TraceEventHandlerMethod m_method;
};
struct TimelineRecordEntry {
TimelineRecordEntry(PassRefPtr<TimelineEvent> record, PassRefPtr<JSONObject> data, PassRefPtr<TypeBuilder::Array<TimelineEvent> > children, const String& type)
: record(record)
, data(data)
, children(children)
, type(type)
, skipWhenUnbalanced(false)
{
}
RefPtr<TimelineEvent> record;
RefPtr<JSONObject> data;
RefPtr<TypeBuilder::Array<TimelineEvent> > children;
String type;
bool skipWhenUnbalanced;
};
class TimelineRecordStack {
DISALLOW_ALLOCATION();
private:
struct Entry {
Entry(PassRefPtr<TimelineEvent> record, const String& type)
: record(record)
, children(TypeBuilder::Array<TimelineEvent>::create())
#if ENABLE(ASSERT)
, type(type)
#endif
{
}
RefPtr<TimelineEvent> record;
RefPtr<TypeBuilder::Array<TimelineEvent> > children;
#if ENABLE(ASSERT)
String type;
#endif
};
public:
TimelineRecordStack() : m_timelineAgent(nullptr) { }
explicit TimelineRecordStack(InspectorTimelineAgent*);
void addScopedRecord(PassRefPtr<TimelineEvent> record, const String& type);
void closeScopedRecord(double endTime);
void addInstantRecord(PassRefPtr<TimelineEvent> record);
#if ENABLE(ASSERT)
bool isOpenRecordOfType(const String& type);
#endif
void trace(Visitor*);
private:
void send(PassRefPtr<JSONObject>);
RawPtrWillBeMember<InspectorTimelineAgent> m_timelineAgent;
Vector<Entry> m_stack;
};
struct TimelineThreadState {
ALLOW_ONLY_INLINE_ALLOCATION();
public:
TimelineThreadState() { }
TimelineThreadState(InspectorTimelineAgent* timelineAgent)
: recordStack(timelineAgent)
, inKnownLayerTask(false)
, decodedPixelRefId(0)
{
}
void trace(Visitor*);
TimelineRecordStack recordStack;
bool inKnownLayerTask;
unsigned long long decodedPixelRefId;
};
struct TimelineImageInfo {
int backendNodeId;
String url;
TimelineImageInfo() : backendNodeId(0) { }
TimelineImageInfo(int backendNodeId, String url) : backendNodeId(backendNodeId), url(url) { }
};
static LocalFrame* frameForExecutionContext(ExecutionContext* context)
{
LocalFrame* frame = 0;
if (context->isDocument())
frame = toDocument(context)->frame();
return frame;
}
static bool eventHasListeners(const AtomicString& eventType, LocalDOMWindow* window, Node* node, const EventPath& eventPath)
{
if (window && window->hasEventListeners(eventType))
return true;
if (node->hasEventListeners(eventType))
return true;
for (size_t i = 0; i < eventPath.size(); i++) {
if (eventPath[i].node()->hasEventListeners(eventType))
return true;
}
return false;
}
void InspectorTimelineAgent::didGC(double startTime, double endTime, size_t collectedBytesCount)
{
RefPtr<TimelineEvent> record = TimelineRecordFactory::createGenericRecord(
startTime * msPerSecond,
0,
TimelineRecordType::GCEvent,
TimelineRecordFactory::createGCEventData(collectedBytesCount));
record->setEndTime(endTime * msPerSecond);
double time = timestamp();
addRecordToTimeline(record.release(), time);
if (m_state->getBoolean(TimelineAgentState::includeCounters)) {
addRecordToTimeline(createCountersUpdate(), time);
}
}
InspectorTimelineAgent::~InspectorTimelineAgent()
{
}
void InspectorTimelineAgent::trace(Visitor* visitor)
{
visitor->trace(m_pageAgent);
visitor->trace(m_layerTreeAgent);
#if ENABLE(OILPAN)
visitor->trace(m_threadStates);
#endif
InspectorBaseAgent::trace(visitor);
}
void InspectorTimelineAgent::setFrontend(InspectorFrontend* frontend)
{
m_frontend = frontend->timeline();
}
void InspectorTimelineAgent::clearFrontend()
{
ErrorString error;
stop(&error);
disable(&error);
m_frontend = 0;
}
void InspectorTimelineAgent::restore()
{
if (m_state->getBoolean(TimelineAgentState::startedFromProtocol)) {
if (m_state->getBoolean(TimelineAgentState::bufferEvents))
m_bufferedEvents = TypeBuilder::Array<TimelineEvent>::create();
setLiveEvents(m_state->getString(TimelineAgentState::liveEvents));
innerStart();
} else if (isStarted()) {
// Timeline was started from console.timeline, it is not restored.
// Tell front-end timline is no longer collecting.
m_state->setBoolean(TimelineAgentState::started, false);
bool fromConsole = true;
m_frontend->stopped(&fromConsole, nullptr);
}
}
void InspectorTimelineAgent::enable(ErrorString*)
{
m_state->setBoolean(TimelineAgentState::enabled, true);
}
void InspectorTimelineAgent::disable(ErrorString*)
{
m_state->setBoolean(TimelineAgentState::enabled, false);
}
void InspectorTimelineAgent::start(ErrorString* errorString, const int* maxCallStackDepth, const bool* bufferEvents, const String* liveEvents, const bool* includeCounters, const bool* includeGPUEvents)
{
if (!m_frontend)
return;
m_state->setBoolean(TimelineAgentState::startedFromProtocol, true);
if (isStarted()) {
*errorString = "Timeline is already started";
return;
}
if (maxCallStackDepth && *maxCallStackDepth >= 0)
m_maxCallStackDepth = *maxCallStackDepth;
else
m_maxCallStackDepth = 5;
if (asBool(bufferEvents)) {
m_bufferedEvents = TypeBuilder::Array<TimelineEvent>::create();
m_lastProgressTimestamp = timestamp();
}
if (liveEvents)
setLiveEvents(*liveEvents);
m_state->setLong(TimelineAgentState::timelineMaxCallStackDepth, m_maxCallStackDepth);
m_state->setBoolean(TimelineAgentState::includeCounters, asBool(includeCounters));
m_state->setBoolean(TimelineAgentState::includeGPUEvents, asBool(includeGPUEvents));
m_state->setBoolean(TimelineAgentState::bufferEvents, asBool(bufferEvents));
m_state->setString(TimelineAgentState::liveEvents, liveEvents ? *liveEvents : "");
innerStart();
bool fromConsole = false;
m_frontend->started(&fromConsole);
}
bool InspectorTimelineAgent::isStarted()
{
return m_state->getBoolean(TimelineAgentState::started);
}
void InspectorTimelineAgent::innerStart()
{
if (m_overlay)
m_overlay->startedRecordingProfile();
m_state->setBoolean(TimelineAgentState::started, true);
m_instrumentingAgents->setInspectorTimelineAgent(this);
ScriptGCEvent::addEventListener(this);
if (m_client) {
TraceEventDispatcher* dispatcher = TraceEventDispatcher::instance();
dispatcher->addListener(InstrumentationEvents::BeginFrame, TRACE_EVENT_PHASE_INSTANT, InspectorTimelineAgentTraceEventListener::create(this, &InspectorTimelineAgent::onBeginImplSideFrame), m_client);
dispatcher->addListener(InstrumentationEvents::PaintSetup, TRACE_EVENT_PHASE_BEGIN, InspectorTimelineAgentTraceEventListener::create(this, &InspectorTimelineAgent::onPaintSetupBegin), m_client);
dispatcher->addListener(InstrumentationEvents::PaintSetup, TRACE_EVENT_PHASE_END, InspectorTimelineAgentTraceEventListener::create(this, &InspectorTimelineAgent::onPaintSetupEnd), m_client);
dispatcher->addListener(InstrumentationEvents::RasterTask, TRACE_EVENT_PHASE_BEGIN, InspectorTimelineAgentTraceEventListener::create(this, &InspectorTimelineAgent::onRasterTaskBegin), m_client);
dispatcher->addListener(InstrumentationEvents::RasterTask, TRACE_EVENT_PHASE_END, InspectorTimelineAgentTraceEventListener::create(this, &InspectorTimelineAgent::onRasterTaskEnd), m_client);
dispatcher->addListener(InstrumentationEvents::Layer, TRACE_EVENT_PHASE_DELETE_OBJECT, InspectorTimelineAgentTraceEventListener::create(this, &InspectorTimelineAgent::onLayerDeleted), m_client);
dispatcher->addListener(InstrumentationEvents::RequestMainThreadFrame, TRACE_EVENT_PHASE_INSTANT, InspectorTimelineAgentTraceEventListener::create(this, &InspectorTimelineAgent::onRequestMainThreadFrame), m_client);
dispatcher->addListener(InstrumentationEvents::ActivateLayerTree, TRACE_EVENT_PHASE_INSTANT, InspectorTimelineAgentTraceEventListener::create(this, &InspectorTimelineAgent::onActivateLayerTree), m_client);
dispatcher->addListener(InstrumentationEvents::DrawFrame, TRACE_EVENT_PHASE_INSTANT, InspectorTimelineAgentTraceEventListener::create(this, &InspectorTimelineAgent::onDrawFrame), m_client);
dispatcher->addListener(PlatformInstrumentation::ImageDecodeEvent, TRACE_EVENT_PHASE_BEGIN, InspectorTimelineAgentTraceEventListener::create(this, &InspectorTimelineAgent::onImageDecodeBegin), m_client);
dispatcher->addListener(PlatformInstrumentation::ImageDecodeEvent, TRACE_EVENT_PHASE_END, InspectorTimelineAgentTraceEventListener::create(this, &InspectorTimelineAgent::onImageDecodeEnd), m_client);
dispatcher->addListener(PlatformInstrumentation::DrawLazyPixelRefEvent, TRACE_EVENT_PHASE_INSTANT, InspectorTimelineAgentTraceEventListener::create(this, &InspectorTimelineAgent::onDrawLazyPixelRef), m_client);
dispatcher->addListener(PlatformInstrumentation::DecodeLazyPixelRefEvent, TRACE_EVENT_PHASE_BEGIN, InspectorTimelineAgentTraceEventListener::create(this, &InspectorTimelineAgent::onDecodeLazyPixelRefBegin), m_client);
dispatcher->addListener(PlatformInstrumentation::DecodeLazyPixelRefEvent, TRACE_EVENT_PHASE_END, InspectorTimelineAgentTraceEventListener::create(this, &InspectorTimelineAgent::onDecodeLazyPixelRefEnd), m_client);
dispatcher->addListener(PlatformInstrumentation::LazyPixelRef, TRACE_EVENT_PHASE_DELETE_OBJECT, InspectorTimelineAgentTraceEventListener::create(this, &InspectorTimelineAgent::onLazyPixelRefDeleted), m_client);
dispatcher->addListener(InstrumentationEvents::EmbedderCallback, TRACE_EVENT_PHASE_BEGIN, InspectorTimelineAgentTraceEventListener::create(this, &InspectorTimelineAgent::onEmbedderCallbackBegin), m_client);
dispatcher->addListener(InstrumentationEvents::EmbedderCallback, TRACE_EVENT_PHASE_END, InspectorTimelineAgentTraceEventListener::create(this, &InspectorTimelineAgent::onEmbedderCallbackEnd), m_client);
if (m_state->getBoolean(TimelineAgentState::includeGPUEvents)) {
m_pendingGPURecord.clear();
m_client->startGPUEventsRecording();
}
}
}
void InspectorTimelineAgent::stop(ErrorString* errorString)
{
m_state->setBoolean(TimelineAgentState::startedFromProtocol, false);
m_state->setBoolean(TimelineAgentState::bufferEvents, false);
m_state->setString(TimelineAgentState::liveEvents, "");
if (!isStarted()) {
*errorString = "Timeline was not started";
return;
}
innerStop(false);
m_liveEvents.clear();
}
void InspectorTimelineAgent::innerStop(bool fromConsole)
{
m_state->setBoolean(TimelineAgentState::started, false);
if (m_client) {
TraceEventDispatcher::instance()->removeAllListeners(this, m_client);
if (m_state->getBoolean(TimelineAgentState::includeGPUEvents))
m_client->stopGPUEventsRecording();
}
m_instrumentingAgents->setInspectorTimelineAgent(0);
ScriptGCEvent::removeEventListener(this);
clearRecordStack();
m_threadStates.clear();
m_gpuTask.clear();
m_layerToNodeMap.clear();
m_pixelRefToImageInfo.clear();
m_imageBeingPainted = 0;
m_paintSetupStart = 0;
m_mayEmitFirstPaint = false;
for (size_t i = 0; i < m_consoleTimelines.size(); ++i) {
String message = String::format("Timeline '%s' terminated.", m_consoleTimelines[i].utf8().data());
mainFrame()->console().addMessage(ConsoleMessage::create(JSMessageSource, DebugMessageLevel, message));
}
m_consoleTimelines.clear();
m_frontend->stopped(&fromConsole, m_bufferedEvents.release());
if (m_overlay)
m_overlay->finishedRecordingProfile();
}
void InspectorTimelineAgent::didBeginFrame(int frameId)
{
TraceEventDispatcher::instance()->processBackgroundEvents();
m_pendingFrameRecord = TimelineRecordFactory::createGenericRecord(timestamp(), 0, TimelineRecordType::BeginFrame, TimelineRecordFactory::createFrameData(frameId));
}
void InspectorTimelineAgent::didCancelFrame()
{
m_pendingFrameRecord.clear();
}
bool InspectorTimelineAgent::willCallFunction(ExecutionContext* context, int scriptId, const String& scriptName, int scriptLine)
{
pushCurrentRecord(TimelineRecordFactory::createFunctionCallData(scriptId, scriptName, scriptLine), TimelineRecordType::FunctionCall, true, frameForExecutionContext(context));
return true;
}
void InspectorTimelineAgent::didCallFunction()
{
didCompleteCurrentRecord(TimelineRecordType::FunctionCall);
}
bool InspectorTimelineAgent::willDispatchEvent(Document* document, const Event& event, LocalDOMWindow* window, Node* node, const EventPath& eventPath)
{
if (!eventHasListeners(event.type(), window, node, eventPath))
return false;
pushCurrentRecord(TimelineRecordFactory::createEventDispatchData(event), TimelineRecordType::EventDispatch, false, document->frame());
return true;
}
bool InspectorTimelineAgent::willDispatchEventOnWindow(const Event& event, LocalDOMWindow* window)
{
if (!window->hasEventListeners(event.type()))
return false;
pushCurrentRecord(TimelineRecordFactory::createEventDispatchData(event), TimelineRecordType::EventDispatch, false, window->frame());
return true;
}
void InspectorTimelineAgent::didDispatchEvent()
{
didCompleteCurrentRecord(TimelineRecordType::EventDispatch);
}
void InspectorTimelineAgent::didDispatchEventOnWindow()
{
didDispatchEvent();
}
void InspectorTimelineAgent::didInvalidateLayout(LocalFrame* frame)
{
appendRecord(JSONObject::create(), TimelineRecordType::InvalidateLayout, true, frame);
}
bool InspectorTimelineAgent::willLayout(LocalFrame* frame)
{
bool isPartial;
unsigned needsLayoutObjects;
unsigned totalObjects;
frame->countObjectsNeedingLayout(needsLayoutObjects, totalObjects, isPartial);
pushCurrentRecord(TimelineRecordFactory::createLayoutData(needsLayoutObjects, totalObjects, isPartial), TimelineRecordType::Layout, true, frame);
return true;
}
void InspectorTimelineAgent::didLayout(RenderObject* root)
{
if (m_recordStack.isEmpty())
return;
TimelineRecordEntry& entry = m_recordStack.last();
ASSERT(entry.type == TimelineRecordType::Layout);
Vector<FloatQuad> quads;
root->absoluteQuads(quads);
if (quads.size() >= 1)
TimelineRecordFactory::setLayoutRoot(entry.data.get(), quads[0], nodeId(root));
else
ASSERT_NOT_REACHED();
didCompleteCurrentRecord(TimelineRecordType::Layout);
}
void InspectorTimelineAgent::layerTreeDidChange()
{
ASSERT(!m_pendingLayerTreeData);
m_pendingLayerTreeData = m_layerTreeAgent->buildLayerTree();
}
void InspectorTimelineAgent::willUpdateLayerTree()
{
pushCurrentRecord(JSONObject::create(), TimelineRecordType::UpdateLayerTree, false, 0);
}
void InspectorTimelineAgent::didUpdateLayerTree()
{
if (m_recordStack.isEmpty())
return;
TimelineRecordEntry& entry = m_recordStack.last();
ASSERT(entry.type == TimelineRecordType::UpdateLayerTree);
if (m_pendingLayerTreeData)
TimelineRecordFactory::setLayerTreeData(entry.data.get(), m_pendingLayerTreeData.release());
didCompleteCurrentRecord(TimelineRecordType::UpdateLayerTree);
}
void InspectorTimelineAgent::didScheduleStyleRecalculation(Document* document)
{
appendRecord(JSONObject::create(), TimelineRecordType::ScheduleStyleRecalculation, true, document->frame());
}
bool InspectorTimelineAgent::willRecalculateStyle(Document* document)
{
pushCurrentRecord(JSONObject::create(), TimelineRecordType::RecalculateStyles, true, document->frame());
return true;
}
void InspectorTimelineAgent::didRecalculateStyle(int elementCount)
{
if (m_recordStack.isEmpty())
return;
TimelineRecordEntry& entry = m_recordStack.last();
ASSERT(entry.type == TimelineRecordType::RecalculateStyles);
TimelineRecordFactory::setStyleRecalcDetails(entry.data.get(), elementCount);
didCompleteCurrentRecord(TimelineRecordType::RecalculateStyles);
}
void InspectorTimelineAgent::willPaint(RenderObject* renderer, const GraphicsLayer* graphicsLayer)
{
LocalFrame* frame = renderer->frame();
TraceEventDispatcher::instance()->processBackgroundEvents();
double paintSetupStart = m_paintSetupStart;
m_paintSetupStart = 0;
if (graphicsLayer) {
int layerIdentifier = graphicsLayer->platformLayer()->id();
int nodeIdentifier = nodeId(renderer);
ASSERT(layerIdentifier && nodeIdentifier);
m_layerToNodeMap.set(layerIdentifier, nodeIdentifier);
if (paintSetupStart) {
RefPtr<TimelineEvent> paintSetupRecord = TimelineRecordFactory::createGenericRecord(paintSetupStart, 0, TimelineRecordType::PaintSetup, TimelineRecordFactory::createLayerData(nodeIdentifier));
paintSetupRecord->setEndTime(m_paintSetupEnd);
addRecordToTimeline(paintSetupRecord, paintSetupStart);
}
}
pushCurrentRecord(JSONObject::create(), TimelineRecordType::Paint, true, frame, true);
}
void InspectorTimelineAgent::didPaint(RenderObject* renderer, const GraphicsLayer* graphicsLayer, GraphicsContext*, const LayoutRect& clipRect)
{
TimelineRecordEntry& entry = m_recordStack.last();
ASSERT(entry.type == TimelineRecordType::Paint);
FloatQuad quad;
localToPageQuad(*renderer, clipRect, &quad);
int graphicsLayerId = graphicsLayer ? graphicsLayer->platformLayer()->id() : 0;
TimelineRecordFactory::setPaintData(entry.data.get(), quad, nodeId(renderer), graphicsLayerId);
didCompleteCurrentRecord(TimelineRecordType::Paint);
if (m_mayEmitFirstPaint && !graphicsLayer) {
m_mayEmitFirstPaint = false;
appendRecord(JSONObject::create(), TimelineRecordType::MarkFirstPaint, false, 0);
}
}
void InspectorTimelineAgent::willPaintImage(RenderImage* renderImage)
{
ASSERT(!m_imageBeingPainted);
m_imageBeingPainted = renderImage;
}
void InspectorTimelineAgent::didPaintImage()
{
m_imageBeingPainted = 0;
}
void InspectorTimelineAgent::willScrollLayer(RenderObject* renderer)
{
pushCurrentRecord(TimelineRecordFactory::createLayerData(nodeId(renderer)), TimelineRecordType::ScrollLayer, false, renderer->frame());
}
void InspectorTimelineAgent::didScrollLayer()
{
didCompleteCurrentRecord(TimelineRecordType::ScrollLayer);
}
void InspectorTimelineAgent::willDecodeImage(const String& imageType)
{
RefPtr<JSONObject> data = TimelineRecordFactory::createDecodeImageData(imageType);
if (m_imageBeingPainted)
populateImageDetails(data.get(), *m_imageBeingPainted);
pushCurrentRecord(data, TimelineRecordType::DecodeImage, true, 0);
}
void InspectorTimelineAgent::didDecodeImage()
{
didCompleteCurrentRecord(TimelineRecordType::DecodeImage);
}
void InspectorTimelineAgent::willResizeImage(bool shouldCache)
{
RefPtr<JSONObject> data = TimelineRecordFactory::createResizeImageData(shouldCache);
if (m_imageBeingPainted)
populateImageDetails(data.get(), *m_imageBeingPainted);
pushCurrentRecord(data, TimelineRecordType::ResizeImage, true, 0);
}
void InspectorTimelineAgent::didResizeImage()
{
didCompleteCurrentRecord(TimelineRecordType::ResizeImage);
}
void InspectorTimelineAgent::willComposite()
{
pushCurrentRecord(JSONObject::create(), TimelineRecordType::CompositeLayers, false, 0);
}
void InspectorTimelineAgent::didComposite()
{
didCompleteCurrentRecord(TimelineRecordType::CompositeLayers);
if (m_mayEmitFirstPaint) {
m_mayEmitFirstPaint = false;
appendRecord(JSONObject::create(), TimelineRecordType::MarkFirstPaint, false, 0);
}
}
bool InspectorTimelineAgent::willWriteHTML(Document* document, unsigned startLine)
{
pushCurrentRecord(TimelineRecordFactory::createParseHTMLData(startLine), TimelineRecordType::ParseHTML, true, document->frame());
return true;
}
void InspectorTimelineAgent::didWriteHTML(unsigned endLine)
{
if (!m_recordStack.isEmpty()) {
TimelineRecordEntry& entry = m_recordStack.last();
entry.data->setNumber("endLine", endLine);
didCompleteCurrentRecord(TimelineRecordType::ParseHTML);
}
}
void InspectorTimelineAgent::didInstallTimer(ExecutionContext* context, int timerId, int timeout, bool singleShot)
{
appendRecord(TimelineRecordFactory::createTimerInstallData(timerId, timeout, singleShot), TimelineRecordType::TimerInstall, true, frameForExecutionContext(context));
}
void InspectorTimelineAgent::didRemoveTimer(ExecutionContext* context, int timerId)
{
appendRecord(TimelineRecordFactory::createGenericTimerData(timerId), TimelineRecordType::TimerRemove, true, frameForExecutionContext(context));
}
bool InspectorTimelineAgent::willFireTimer(ExecutionContext* context, int timerId)
{
pushCurrentRecord(TimelineRecordFactory::createGenericTimerData(timerId), TimelineRecordType::TimerFire, false, frameForExecutionContext(context));
return true;
}
void InspectorTimelineAgent::didFireTimer()
{
didCompleteCurrentRecord(TimelineRecordType::TimerFire);
}
bool InspectorTimelineAgent::willDispatchXHRReadyStateChangeEvent(ExecutionContext* context, XMLHttpRequest* request)
{
if (!request->hasEventListeners(EventTypeNames::readystatechange))
return false;
pushCurrentRecord(TimelineRecordFactory::createXHRReadyStateChangeData(request->url().string(), request->readyState()), TimelineRecordType::XHRReadyStateChange, false, frameForExecutionContext(context));
return true;
}
void InspectorTimelineAgent::didDispatchXHRReadyStateChangeEvent()
{
didCompleteCurrentRecord(TimelineRecordType::XHRReadyStateChange);
}
bool InspectorTimelineAgent::willDispatchXHRLoadEvent(ExecutionContext* context, XMLHttpRequest* request)
{
if (!request->hasEventListeners(EventTypeNames::load))
return false;
pushCurrentRecord(TimelineRecordFactory::createXHRLoadData(request->url().string()), TimelineRecordType::XHRLoad, true, frameForExecutionContext(context));
return true;
}
void InspectorTimelineAgent::didDispatchXHRLoadEvent()
{
didCompleteCurrentRecord(TimelineRecordType::XHRLoad);
}
bool InspectorTimelineAgent::willEvaluateScript(LocalFrame* frame, const String& url, int lineNumber)
{
pushCurrentRecord(TimelineRecordFactory::createEvaluateScriptData(url, lineNumber), TimelineRecordType::EvaluateScript, true, frame);
return true;
}
void InspectorTimelineAgent::didEvaluateScript()
{
didCompleteCurrentRecord(TimelineRecordType::EvaluateScript);
}
void InspectorTimelineAgent::willSendRequest(unsigned long identifier, DocumentLoader* loader, const ResourceRequest& request, const ResourceResponse&, const FetchInitiatorInfo&)
{
String requestId = IdentifiersFactory::requestId(identifier);
appendRecord(TimelineRecordFactory::createResourceSendRequestData(requestId, request), TimelineRecordType::ResourceSendRequest, true, loader->frame());
}
void InspectorTimelineAgent::didReceiveData(LocalFrame* frame, unsigned long identifier, const char*, int, int encodedDataLength)
{
String requestId = IdentifiersFactory::requestId(identifier);
appendRecord(TimelineRecordFactory::createReceiveResourceData(requestId, encodedDataLength), TimelineRecordType::ResourceReceivedData, false, frame);
}
void InspectorTimelineAgent::didReceiveResourceResponse(LocalFrame* frame, unsigned long identifier, DocumentLoader* loader, const ResourceResponse& response, ResourceLoader* resourceLoader)
{
String requestId = IdentifiersFactory::requestId(identifier);
appendRecord(TimelineRecordFactory::createResourceReceiveResponseData(requestId, response), TimelineRecordType::ResourceReceiveResponse, false, 0);
}
void InspectorTimelineAgent::didFinishLoadingResource(unsigned long identifier, bool didFail, double finishTime)
{
appendRecord(TimelineRecordFactory::createResourceFinishData(IdentifiersFactory::requestId(identifier), didFail, finishTime), TimelineRecordType::ResourceFinish, false, 0);
}
void InspectorTimelineAgent::didFinishLoading(unsigned long identifier, DocumentLoader* loader, double monotonicFinishTime, int64_t)
{
didFinishLoadingResource(identifier, false, monotonicFinishTime * msPerSecond);
}
void InspectorTimelineAgent::didFailLoading(unsigned long identifier, const ResourceError& error)
{
didFinishLoadingResource(identifier, true, 0);
}
void InspectorTimelineAgent::consoleTimeStamp(ExecutionContext* context, const String& title)
{
appendRecord(TimelineRecordFactory::createTimeStampData(title), TimelineRecordType::TimeStamp, true, frameForExecutionContext(context));
}
void InspectorTimelineAgent::consoleTime(ExecutionContext* context, const String& message)
{
pushCurrentRecord(TimelineRecordFactory::createConsoleTimeData(message), TimelineRecordType::ConsoleTime, false, frameForExecutionContext(context));
m_recordStack.last().skipWhenUnbalanced = true;
}
void InspectorTimelineAgent::consoleTimeEnd(ExecutionContext* context, const String& message, ScriptState*)
{
if (m_recordStack.last().type != TimelineRecordType::ConsoleTime)
return;
String originalMessage;
if (m_recordStack.last().data->getString("message", &originalMessage) && message != originalMessage)
return;
// Only complete console.time that is balanced.
didCompleteCurrentRecord(TimelineRecordType::ConsoleTime);
}
void InspectorTimelineAgent::consoleTimeline(ExecutionContext* context, const String& title, ScriptState* scriptState)
{
if (!m_state->getBoolean(TimelineAgentState::enabled))
return;
String message = String::format("Timeline '%s' started.", title.utf8().data());
RefPtrWillBeRawPtr<ConsoleMessage> consoleMessage = ConsoleMessage::create(JSMessageSource, DebugMessageLevel, message);
consoleMessage->setScriptState(scriptState);
mainFrame()->console().addMessage(consoleMessage.release());
m_consoleTimelines.append(title);
if (!isStarted()) {
m_state->setBoolean(TimelineAgentState::bufferEvents, true);
m_bufferedEvents = TypeBuilder::Array<TimelineEvent>::create();
innerStart();
bool fromConsole = true;
m_frontend->started(&fromConsole);
}
appendRecord(TimelineRecordFactory::createTimeStampData(message), TimelineRecordType::TimeStamp, true, frameForExecutionContext(context));
}
void InspectorTimelineAgent::consoleTimelineEnd(ExecutionContext* context, const String& title, ScriptState* scriptState)
{
if (!m_state->getBoolean(TimelineAgentState::enabled))
return;
size_t index = m_consoleTimelines.find(title);
if (index == kNotFound) {
String message = String::format("Timeline '%s' was not started.", title.utf8().data());
RefPtrWillBeRawPtr<ConsoleMessage> consoleMessage = ConsoleMessage::create(JSMessageSource, DebugMessageLevel, message);
consoleMessage->setScriptState(scriptState);
mainFrame()->console().addMessage(consoleMessage.release());
return;
}
String message = String::format("Timeline '%s' finished.", title.utf8().data());
appendRecord(TimelineRecordFactory::createTimeStampData(message), TimelineRecordType::TimeStamp, true, frameForExecutionContext(context));
m_consoleTimelines.remove(index);
if (!m_consoleTimelines.size() && isStarted() && !m_state->getBoolean(TimelineAgentState::startedFromProtocol)) {
unwindRecordStack();
innerStop(true);
}
RefPtrWillBeRawPtr<ConsoleMessage> consoleMessage = ConsoleMessage::create(JSMessageSource, DebugMessageLevel, message);
consoleMessage->setScriptState(scriptState);
mainFrame()->console().addMessage(consoleMessage.release());
}
void InspectorTimelineAgent::domContentLoadedEventFired(LocalFrame* frame)
{
bool isMainFrame = frame && m_pageAgent && (frame == m_pageAgent->mainFrame());
appendRecord(TimelineRecordFactory::createMarkData(isMainFrame), TimelineRecordType::MarkDOMContent, false, frame);
if (isMainFrame)
m_mayEmitFirstPaint = true;
}
void InspectorTimelineAgent::loadEventFired(LocalFrame* frame)
{
bool isMainFrame = frame && m_pageAgent && (frame == m_pageAgent->mainFrame());
appendRecord(TimelineRecordFactory::createMarkData(isMainFrame), TimelineRecordType::MarkLoad, false, frame);
}
void InspectorTimelineAgent::didCommitLoad()
{
clearRecordStack();
}
void InspectorTimelineAgent::didRequestAnimationFrame(Document* document, int callbackId)
{
appendRecord(TimelineRecordFactory::createAnimationFrameData(callbackId), TimelineRecordType::RequestAnimationFrame, true, document->frame());
}
void InspectorTimelineAgent::didCancelAnimationFrame(Document* document, int callbackId)
{
appendRecord(TimelineRecordFactory::createAnimationFrameData(callbackId), TimelineRecordType::CancelAnimationFrame, true, document->frame());
}
bool InspectorTimelineAgent::willFireAnimationFrame(Document* document, int callbackId)
{
pushCurrentRecord(TimelineRecordFactory::createAnimationFrameData(callbackId), TimelineRecordType::FireAnimationFrame, false, document->frame());
return true;
}
void InspectorTimelineAgent::didFireAnimationFrame()
{
didCompleteCurrentRecord(TimelineRecordType::FireAnimationFrame);
}
void InspectorTimelineAgent::willProcessTask()
{
pushCurrentRecord(JSONObject::create(), TimelineRecordType::Program, false, 0);
}
void InspectorTimelineAgent::didProcessTask()
{
didCompleteCurrentRecord(TimelineRecordType::Program);
}
void InspectorTimelineAgent::didCreateWebSocket(Document* document, unsigned long identifier, const KURL& url, const String& protocol)
{
appendRecord(TimelineRecordFactory::createWebSocketCreateData(identifier, url, protocol), TimelineRecordType::WebSocketCreate, true, document->frame());
}
void InspectorTimelineAgent::willSendWebSocketHandshakeRequest(Document* document, unsigned long identifier, const WebSocketHandshakeRequest*)
{
appendRecord(TimelineRecordFactory::createGenericWebSocketData(identifier), TimelineRecordType::WebSocketSendHandshakeRequest, true, document->frame());
}
void InspectorTimelineAgent::didReceiveWebSocketHandshakeResponse(Document* document, unsigned long identifier, const WebSocketHandshakeRequest*, const WebSocketHandshakeResponse*)
{
appendRecord(TimelineRecordFactory::createGenericWebSocketData(identifier), TimelineRecordType::WebSocketReceiveHandshakeResponse, false, document->frame());
}
void InspectorTimelineAgent::didCloseWebSocket(Document* document, unsigned long identifier)
{
appendRecord(TimelineRecordFactory::createGenericWebSocketData(identifier), TimelineRecordType::WebSocketDestroy, true, document->frame());
}
void InspectorTimelineAgent::onBeginImplSideFrame(const TraceEventDispatcher::TraceEvent& event)
{
unsigned long long layerTreeId = event.asUInt(InstrumentationEventArguments::LayerTreeId);
if (layerTreeId != m_layerTreeId)
return;
TimelineThreadState& state = threadState(event.threadIdentifier());
state.recordStack.addInstantRecord(createRecordForEvent(event, TimelineRecordType::BeginFrame, JSONObject::create()));
}
void InspectorTimelineAgent::onPaintSetupBegin(const TraceEventDispatcher::TraceEvent& event)
{
ASSERT(!m_paintSetupStart);
m_paintSetupStart = event.timestamp() * msPerSecond;
}
void InspectorTimelineAgent::onPaintSetupEnd(const TraceEventDispatcher::TraceEvent& event)
{
ASSERT(m_paintSetupStart);
m_paintSetupEnd = event.timestamp() * msPerSecond;
}
void InspectorTimelineAgent::onRasterTaskBegin(const TraceEventDispatcher::TraceEvent& event)
{
TimelineThreadState& state = threadState(event.threadIdentifier());
unsigned long long layerId = event.asUInt(InstrumentationEventArguments::LayerId);
ASSERT(layerId);
if (!m_layerToNodeMap.contains(layerId))
return;
ASSERT(!state.inKnownLayerTask);
state.inKnownLayerTask = true;
double timestamp = event.timestamp() * msPerSecond;
RefPtr<JSONObject> data = TimelineRecordFactory::createLayerData(m_layerToNodeMap.get(layerId));
RefPtr<TimelineEvent> record = TimelineRecordFactory::createBackgroundRecord(timestamp, String::number(event.threadIdentifier()), TimelineRecordType::Rasterize, data);
state.recordStack.addScopedRecord(record, TimelineRecordType::Rasterize);
}
void InspectorTimelineAgent::onRasterTaskEnd(const TraceEventDispatcher::TraceEvent& event)
{
TimelineThreadState& state = threadState(event.threadIdentifier());
if (!state.inKnownLayerTask)
return;
ASSERT(state.recordStack.isOpenRecordOfType(TimelineRecordType::Rasterize));
state.recordStack.closeScopedRecord(event.timestamp() * msPerSecond);
state.inKnownLayerTask = false;
}
void InspectorTimelineAgent::onImageDecodeBegin(const TraceEventDispatcher::TraceEvent& event)
{
TimelineThreadState& state = threadState(event.threadIdentifier());
if (!state.decodedPixelRefId && !state.inKnownLayerTask)
return;
TimelineImageInfo imageInfo;
if (state.decodedPixelRefId) {
PixelRefToImageInfoMap::const_iterator it = m_pixelRefToImageInfo.find(state.decodedPixelRefId);
if (it != m_pixelRefToImageInfo.end())
imageInfo = it->value;
else
ASSERT_NOT_REACHED();
}
RefPtr<JSONObject> data = JSONObject::create();
TimelineRecordFactory::setImageDetails(data.get(), imageInfo.backendNodeId, imageInfo.url);
double timeestamp = event.timestamp() * msPerSecond;
state.recordStack.addScopedRecord(TimelineRecordFactory::createBackgroundRecord(timeestamp, String::number(event.threadIdentifier()), TimelineRecordType::DecodeImage, data), TimelineRecordType::DecodeImage);
}
void InspectorTimelineAgent::onImageDecodeEnd(const TraceEventDispatcher::TraceEvent& event)
{
TimelineThreadState& state = threadState(event.threadIdentifier());
if (!state.decodedPixelRefId)
return;
ASSERT(state.recordStack.isOpenRecordOfType(TimelineRecordType::DecodeImage));
state.recordStack.closeScopedRecord(event.timestamp() * msPerSecond);
}
void InspectorTimelineAgent::onRequestMainThreadFrame(const TraceEventDispatcher::TraceEvent& event)
{
unsigned long long layerTreeId = event.asUInt(InstrumentationEventArguments::LayerTreeId);
if (layerTreeId != m_layerTreeId)
return;
TimelineThreadState& state = threadState(event.threadIdentifier());
state.recordStack.addInstantRecord(createRecordForEvent(event, TimelineRecordType::RequestMainThreadFrame, JSONObject::create()));
}
void InspectorTimelineAgent::onActivateLayerTree(const TraceEventDispatcher::TraceEvent& event)
{
unsigned long long layerTreeId = event.asUInt(InstrumentationEventArguments::LayerTreeId);
if (layerTreeId != m_layerTreeId)
return;
unsigned long long frameId = event.asUInt(InstrumentationEventArguments::FrameId);
TimelineThreadState& state = threadState(event.threadIdentifier());
state.recordStack.addInstantRecord(createRecordForEvent(event, TimelineRecordType::ActivateLayerTree, TimelineRecordFactory::createFrameData(frameId)));
}
void InspectorTimelineAgent::onDrawFrame(const TraceEventDispatcher::TraceEvent& event)
{
unsigned long long layerTreeId = event.asUInt(InstrumentationEventArguments::LayerTreeId);
if (layerTreeId != m_layerTreeId)
return;
TimelineThreadState& state = threadState(event.threadIdentifier());
state.recordStack.addInstantRecord(createRecordForEvent(event, TimelineRecordType::DrawFrame, JSONObject::create()));
}
void InspectorTimelineAgent::onLayerDeleted(const TraceEventDispatcher::TraceEvent& event)
{
unsigned long long id = event.id();
ASSERT(id);
m_layerToNodeMap.remove(id);
}
void InspectorTimelineAgent::onDecodeLazyPixelRefBegin(const TraceEventDispatcher::TraceEvent& event)
{
TimelineThreadState& state = threadState(event.threadIdentifier());
ASSERT(!state.decodedPixelRefId);
unsigned long long pixelRefId = event.asUInt(PlatformInstrumentation::LazyPixelRef);
ASSERT(pixelRefId);
if (m_pixelRefToImageInfo.contains(pixelRefId))
state.decodedPixelRefId = pixelRefId;
}
void InspectorTimelineAgent::onDecodeLazyPixelRefEnd(const TraceEventDispatcher::TraceEvent& event)
{
threadState(event.threadIdentifier()).decodedPixelRefId = 0;
}
void InspectorTimelineAgent::onDrawLazyPixelRef(const TraceEventDispatcher::TraceEvent& event)
{
unsigned long long pixelRefId = event.asUInt(PlatformInstrumentation::LazyPixelRef);
ASSERT(pixelRefId);
if (!m_imageBeingPainted)
return;
String url;
if (const ImageResource* resource = m_imageBeingPainted->cachedImage())
url = resource->url().string();
m_pixelRefToImageInfo.set(pixelRefId, TimelineImageInfo(nodeId(m_imageBeingPainted->generatingNode()), url));
}
void InspectorTimelineAgent::onLazyPixelRefDeleted(const TraceEventDispatcher::TraceEvent& event)
{
m_pixelRefToImageInfo.remove(event.id());
}
void InspectorTimelineAgent::processGPUEvent(const GPUEvent& event)
{
double timelineTimestamp = event.timestamp * msPerSecond;
if (event.phase == GPUEvent::PhaseBegin) {
m_pendingGPURecord = TimelineRecordFactory::createBackgroundRecord(timelineTimestamp, "gpu", TimelineRecordType::GPUTask, TimelineRecordFactory::createGPUTaskData(event.foreign));
} else if (m_pendingGPURecord) {
m_pendingGPURecord->setEndTime(timelineTimestamp);
sendEvent(m_pendingGPURecord.release());
if (!event.foreign && m_state->getBoolean(TimelineAgentState::includeCounters)) {
RefPtr<TypeBuilder::Timeline::Counters> counters = TypeBuilder::Timeline::Counters::create();
counters->setGpuMemoryUsedKB(static_cast<double>(event.usedGPUMemoryBytes / 1024));
counters->setGpuMemoryLimitKB(static_cast<double>(event.limitGPUMemoryBytes / 1024));
sendEvent(TimelineRecordFactory::createBackgroundRecord(timelineTimestamp, "gpu", TimelineRecordType::UpdateCounters, counters.release()->asObject()));
}
}
}
void InspectorTimelineAgent::onEmbedderCallbackBegin(const TraceEventDispatcher::TraceEvent& event)
{
TimelineThreadState& state = threadState(event.threadIdentifier());
double timestamp = event.timestamp() * msPerSecond;
RefPtr<JSONObject> data = TimelineRecordFactory::createEmbedderCallbackData(event.asString(InstrumentationEventArguments::CallbackName));
RefPtr<TimelineEvent> record = TimelineRecordFactory::createGenericRecord(timestamp, 0, TimelineRecordType::EmbedderCallback, data);
state.recordStack.addScopedRecord(record, TimelineRecordType::EmbedderCallback);
}
void InspectorTimelineAgent::onEmbedderCallbackEnd(const TraceEventDispatcher::TraceEvent& event)
{
TimelineThreadState& state = threadState(event.threadIdentifier());
state.recordStack.closeScopedRecord(event.timestamp() * msPerSecond);
}
void InspectorTimelineAgent::addRecordToTimeline(PassRefPtr<TimelineEvent> record, double ts)
{
commitFrameRecord();
innerAddRecordToTimeline(record);
if (m_bufferedEvents && ts - m_lastProgressTimestamp > 300) {
m_lastProgressTimestamp = ts;
m_frontend->progress(m_bufferedEvents->length());
}
}
void InspectorTimelineAgent::innerAddRecordToTimeline(PassRefPtr<TimelineEvent> record)
{
if (m_recordStack.isEmpty()) {
TraceEventDispatcher::instance()->processBackgroundEvents();
sendEvent(record);
} else {
TimelineRecordEntry& parent = m_recordStack.last();
parent.children->addItem(record);
if (m_state->getBoolean(TimelineAgentState::includeCounters))
parent.children->addItem(createCountersUpdate());
}
}
static size_t getUsedHeapSize()
{
HeapInfo info;
ScriptGCEvent::getHeapSize(info);
return info.usedJSHeapSize;
}
PassRefPtr<TypeBuilder::Timeline::TimelineEvent> InspectorTimelineAgent::createCountersUpdate()
{
RefPtr<TypeBuilder::Timeline::Counters> counters = TypeBuilder::Timeline::Counters::create();
if (m_inspectorType == PageInspector) {
counters->setDocuments(InspectorCounters::counterValue(InspectorCounters::DocumentCounter));
counters->setNodes(InspectorCounters::counterValue(InspectorCounters::NodeCounter));
counters->setJsEventListeners(InspectorCounters::counterValue(InspectorCounters::JSEventListenerCounter));
}
counters->setJsHeapSizeUsed(static_cast<double>(getUsedHeapSize()));
return TimelineRecordFactory::createGenericRecord(timestamp(), 0, TimelineRecordType::UpdateCounters, counters.release()->asObject());
}
void InspectorTimelineAgent::setFrameIdentifier(TimelineEvent* record, LocalFrame* frame)
{
if (!frame || !m_pageAgent)
return;
String frameId;
if (frame && m_pageAgent)
frameId = m_pageAgent->frameId(frame);
record->setFrameId(frameId);
}
void InspectorTimelineAgent::populateImageDetails(JSONObject* data, const RenderImage& renderImage)
{
const ImageResource* resource = renderImage.cachedImage();
TimelineRecordFactory::setImageDetails(data, nodeId(renderImage.generatingNode()), resource ? resource->url().string() : "");
}
void InspectorTimelineAgent::didCompleteCurrentRecord(const String& type)
{
// An empty stack could merely mean that the timeline agent was turned on in the middle of
// an event. Don't treat as an error.
if (!m_recordStack.isEmpty()) {
if (m_platformInstrumentationClientInstalledAtStackDepth == m_recordStack.size()) {
m_platformInstrumentationClientInstalledAtStackDepth = 0;
PlatformInstrumentation::setClient(0);
}
TimelineRecordEntry entry = m_recordStack.last();
m_recordStack.removeLast();
while (entry.type != type && entry.skipWhenUnbalanced && !m_recordStack.isEmpty()) {
// Discard pending skippable entry, paste its children inplace.
if (entry.children)
m_recordStack.last().children->concat(entry.children);
entry = m_recordStack.last();
m_recordStack.removeLast();
}
ASSERT(entry.type == type);
entry.record->setChildren(entry.children);
double ts = timestamp();
entry.record->setEndTime(ts);
addRecordToTimeline(entry.record, ts);
}
}
void InspectorTimelineAgent::unwindRecordStack()
{
while (!m_recordStack.isEmpty()) {
TimelineRecordEntry& entry = m_recordStack.last();
didCompleteCurrentRecord(entry.type);
}
}
InspectorTimelineAgent::InspectorTimelineAgent(InspectorPageAgent* pageAgent, InspectorLayerTreeAgent* layerTreeAgent,
InspectorOverlay* overlay, InspectorType type, InspectorClient* client)
: InspectorBaseAgent<InspectorTimelineAgent>("Timeline")
, m_pageAgent(pageAgent)
, m_layerTreeAgent(layerTreeAgent)
, m_frontend(0)
, m_client(client)
, m_overlay(overlay)
, m_inspectorType(type)
, m_id(1)
, m_layerTreeId(0)
, m_maxCallStackDepth(5)
, m_platformInstrumentationClientInstalledAtStackDepth(0)
, m_imageBeingPainted(0)
, m_paintSetupStart(0)
, m_mayEmitFirstPaint(false)
, m_lastProgressTimestamp(0)
{
}
void InspectorTimelineAgent::appendRecord(PassRefPtr<JSONObject> data, const String& type, bool captureCallStack, LocalFrame* frame)
{
double ts = timestamp();
RefPtr<TimelineEvent> record = TimelineRecordFactory::createGenericRecord(ts, captureCallStack ? m_maxCallStackDepth : 0, type, data);
setFrameIdentifier(record.get(), frame);
addRecordToTimeline(record.release(), ts);
}
void InspectorTimelineAgent::sendEvent(PassRefPtr<TimelineEvent> record)
{
RefPtr<TimelineEvent> retain = record;
if (m_bufferedEvents) {
m_bufferedEvents->addItem(retain);
if (!m_liveEvents.contains(TimelineRecordFactory::type(retain.get())))
return;
}
m_frontend->eventRecorded(retain.release());
}
void InspectorTimelineAgent::pushCurrentRecord(PassRefPtr<JSONObject> data, const String& type, bool captureCallStack, LocalFrame* frame, bool hasLowLevelDetails)
{
commitFrameRecord();
RefPtr<TimelineEvent> record = TimelineRecordFactory::createGenericRecord(timestamp(), captureCallStack ? m_maxCallStackDepth : 0, type, data.get());
setFrameIdentifier(record.get(), frame);
m_recordStack.append(TimelineRecordEntry(record.release(), data, TypeBuilder::Array<TimelineEvent>::create(), type));
if (hasLowLevelDetails && !m_platformInstrumentationClientInstalledAtStackDepth && !PlatformInstrumentation::hasClient()) {
m_platformInstrumentationClientInstalledAtStackDepth = m_recordStack.size();
PlatformInstrumentation::setClient(this);
}
}
TimelineThreadState& InspectorTimelineAgent::threadState(ThreadIdentifier thread)
{
ThreadStateMap::iterator it = m_threadStates.find(thread);
if (it != m_threadStates.end())
return it->value;
return m_threadStates.add(thread, TimelineThreadState(this)).storedValue->value;
}
void InspectorTimelineAgent::commitFrameRecord()
{
if (!m_pendingFrameRecord)
return;
innerAddRecordToTimeline(m_pendingFrameRecord.release());
}
void InspectorTimelineAgent::clearRecordStack()
{
if (m_platformInstrumentationClientInstalledAtStackDepth) {
m_platformInstrumentationClientInstalledAtStackDepth = 0;
PlatformInstrumentation::setClient(0);
}
m_pendingFrameRecord.clear();
m_recordStack.clear();
m_id++;
}
void InspectorTimelineAgent::localToPageQuad(const RenderObject& renderer, const LayoutRect& rect, FloatQuad* quad)
{
LocalFrame* frame = renderer.frame();
FrameView* view = frame->view();
FloatQuad absolute = renderer.localToAbsoluteQuad(FloatQuad(rect));
quad->setP1(view->contentsToRootView(roundedIntPoint(absolute.p1())));
quad->setP2(view->contentsToRootView(roundedIntPoint(absolute.p2())));
quad->setP3(view->contentsToRootView(roundedIntPoint(absolute.p3())));
quad->setP4(view->contentsToRootView(roundedIntPoint(absolute.p4())));
}
long long InspectorTimelineAgent::nodeId(Node* node)
{
return node ? InspectorNodeIds::idForNode(node) : 0;
}
long long InspectorTimelineAgent::nodeId(RenderObject* renderer)
{
return InspectorNodeIds::idForNode(renderer->generatingNode());
}
double InspectorTimelineAgent::timestamp()
{
return WTF::monotonicallyIncreasingTime() * msPerSecond;
}
LocalFrame* InspectorTimelineAgent::mainFrame() const
{
if (!m_pageAgent)
return 0;
return m_pageAgent->mainFrame();
}
PassRefPtr<TimelineEvent> InspectorTimelineAgent::createRecordForEvent(const TraceEventDispatcher::TraceEvent& event, const String& type, PassRefPtr<JSONObject> data)
{
double timeestamp = event.timestamp() * msPerSecond;
return TimelineRecordFactory::createBackgroundRecord(timeestamp, String::number(event.threadIdentifier()), type, data);
}
void InspectorTimelineAgent::setLiveEvents(const String& liveEvents)
{
m_liveEvents.clear();
if (liveEvents.isNull() || liveEvents.isEmpty())
return;
Vector<String> eventList;
liveEvents.split(',', eventList);
for (Vector<String>::iterator it = eventList.begin(); it != eventList.end(); ++it)
m_liveEvents.add(*it);
}
TimelineRecordStack::TimelineRecordStack(InspectorTimelineAgent* timelineAgent)
: m_timelineAgent(timelineAgent)
{
}
void TimelineRecordStack::addScopedRecord(PassRefPtr<TimelineEvent> record, const String& type)
{
m_stack.append(Entry(record, type));
}
void TimelineRecordStack::closeScopedRecord(double endTime)
{
if (m_stack.isEmpty())
return;
Entry last = m_stack.last();
m_stack.removeLast();
last.record->setEndTime(endTime);
if (last.children->length())
last.record->setChildren(last.children);
addInstantRecord(last.record);
}
void TimelineRecordStack::addInstantRecord(PassRefPtr<TimelineEvent> record)
{
if (m_stack.isEmpty())
m_timelineAgent->sendEvent(record);
else
m_stack.last().children->addItem(record);
}
#if ENABLE(ASSERT)
bool TimelineRecordStack::isOpenRecordOfType(const String& type)
{
return !m_stack.isEmpty() && m_stack.last().type == type;
}
#endif
void TimelineRecordStack::trace(Visitor* visitor)
{
visitor->trace(m_timelineAgent);
}
void TimelineThreadState::trace(Visitor* visitor)
{
visitor->trace(recordStack);
}
} // namespace blink