blob: c3f4e34a2cad945315f855c1676c007771490956 [file] [log] [blame]
/*
* Copyright (C) 2013 Google Inc. All rights reserved.
* Copyright (C) 2014 University of Washington.
* Copyright (C) 2015 Apple 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 "InspectorTimelineAgent.h"
#include "DOMWindow.h"
#include "Event.h"
#include "Frame.h"
#include "FrameSnapshotting.h"
#include "InspectorAnimationAgent.h"
#include "InspectorCPUProfilerAgent.h"
#include "InspectorClient.h"
#include "InspectorController.h"
#include "InspectorMemoryAgent.h"
#include "InspectorPageAgent.h"
#include "InstrumentingAgents.h"
#include "JSDOMWindow.h"
#include "JSExecState.h"
#include "PageDebugger.h"
#include "PageHeapAgent.h"
#include "RenderView.h"
#include "TimelineRecordFactory.h"
#include "WebConsoleAgent.h"
#include "WebDebuggerAgent.h"
#include <JavaScriptCore/Breakpoint.h>
#include <JavaScriptCore/ConsoleMessage.h>
#include <JavaScriptCore/InspectorScriptProfilerAgent.h>
#include <JavaScriptCore/ScriptArguments.h>
#include <wtf/SetForScope.h>
#include <wtf/Stopwatch.h>
#if PLATFORM(IOS_FAMILY)
#include "RuntimeApplicationChecks.h"
#include "WebCoreThreadInternal.h"
#endif
#if PLATFORM(COCOA)
#include "RunLoopObserver.h"
#endif
namespace WebCore {
using namespace Inspector;
#if PLATFORM(COCOA)
static CFRunLoopRef currentRunLoop()
{
#if PLATFORM(IOS_FAMILY)
// A race condition during WebView deallocation can lead to a crash if the layer sync run loop
// observer is added to the main run loop <rdar://problem/9798550>. However, for responsiveness,
// we still allow this, see <rdar://problem/7403328>. Since the race condition and subsequent
// crash are especially troublesome for iBooks, we never allow the observer to be added to the
// main run loop in iBooks.
if (CocoaApplication::isIBooks())
return WebThreadRunLoop();
#endif
return CFRunLoopGetCurrent();
}
#endif
InspectorTimelineAgent::InspectorTimelineAgent(PageAgentContext& context)
: InspectorAgentBase("Timeline"_s, context)
, m_frontendDispatcher(makeUnique<Inspector::TimelineFrontendDispatcher>(context.frontendRouter))
, m_backendDispatcher(Inspector::TimelineBackendDispatcher::create(context.backendDispatcher, this))
, m_inspectedPage(context.inspectedPage)
{
}
InspectorTimelineAgent::~InspectorTimelineAgent() = default;
void InspectorTimelineAgent::didCreateFrontendAndBackend(Inspector::FrontendRouter*, Inspector::BackendDispatcher*)
{
}
void InspectorTimelineAgent::willDestroyFrontendAndBackend(Inspector::DisconnectReason)
{
disable();
}
Protocol::ErrorStringOr<void> InspectorTimelineAgent::enable()
{
if (m_instrumentingAgents.enabledTimelineAgent() == this)
return makeUnexpected("Timeline domain already enabled"_s);
m_instrumentingAgents.setEnabledTimelineAgent(this);
return { };
}
Protocol::ErrorStringOr<void> InspectorTimelineAgent::disable()
{
if (m_instrumentingAgents.enabledTimelineAgent() != this)
return makeUnexpected("Timeline domain already disabled"_s);
m_instrumentingAgents.setEnabledTimelineAgent(nullptr);
stop();
m_autoCaptureEnabled = false;
m_instruments.clear();
return { };
}
Protocol::ErrorStringOr<void> InspectorTimelineAgent::start(std::optional<int>&& maxCallStackDepth)
{
m_trackingFromFrontend = true;
internalStart(WTFMove(maxCallStackDepth));
return { };
}
Protocol::ErrorStringOr<void> InspectorTimelineAgent::stop()
{
internalStop();
m_trackingFromFrontend = false;
return { };
}
Protocol::ErrorStringOr<void> InspectorTimelineAgent::setAutoCaptureEnabled(bool enabled)
{
m_autoCaptureEnabled = enabled;
return { };
}
Protocol::ErrorStringOr<void> InspectorTimelineAgent::setInstruments(Ref<JSON::Array>&& instruments)
{
Vector<Protocol::Timeline::Instrument> newInstruments;
newInstruments.reserveCapacity(instruments->length());
for (const auto& instrumentValue : instruments.get()) {
auto instrumentString = instrumentValue->asString();
if (!instrumentString)
return makeUnexpected("Unexpected non-string value in given instruments"_s);
auto instrument = Protocol::Helpers::parseEnumValueFromString<Protocol::Timeline::Instrument>(instrumentString);
if (!instrument)
return makeUnexpected(makeString("Unknown instrument: ", instrumentString));
newInstruments.uncheckedAppend(*instrument);
}
m_instruments.swap(newInstruments);
return { };
}
void InspectorTimelineAgent::internalStart(std::optional<int>&& maxCallStackDepth)
{
if (m_tracking)
return;
if (maxCallStackDepth && *maxCallStackDepth > 0)
m_maxCallStackDepth = *maxCallStackDepth;
else
m_maxCallStackDepth = 5;
m_instrumentingAgents.setTrackingTimelineAgent(this);
m_environment.debugger().addObserver(*this);
m_tracking = true;
// FIXME: Abstract away platform-specific code once https://bugs.webkit.org/show_bug.cgi?id=142748 is fixed.
#if PLATFORM(COCOA)
m_frameStartObserver = makeUnique<RunLoopObserver>(static_cast<CFIndex>(RunLoopObserver::WellKnownRunLoopOrders::InspectorFrameBegin), [this]() {
if (!m_tracking || m_environment.debugger().isPaused())
return;
if (!m_runLoopNestingLevel)
pushCurrentRecord(JSON::Object::create(), TimelineRecordType::RenderingFrame, false, nullptr);
m_runLoopNestingLevel++;
});
m_frameStopObserver = makeUnique<RunLoopObserver>(static_cast<CFIndex>(RunLoopObserver::WellKnownRunLoopOrders::InspectorFrameEnd), [this]() {
if (!m_tracking || m_environment.debugger().isPaused())
return;
ASSERT(m_runLoopNestingLevel > 0);
m_runLoopNestingLevel--;
if (m_runLoopNestingLevel)
return;
if (m_startedComposite)
didComposite();
didCompleteCurrentRecord(TimelineRecordType::RenderingFrame);
});
m_frameStartObserver->schedule(currentRunLoop(), kCFRunLoopEntry | kCFRunLoopAfterWaiting);
m_frameStopObserver->schedule(currentRunLoop(), kCFRunLoopExit | kCFRunLoopBeforeWaiting);
// Create a runloop record and increment the runloop nesting level, to capture the current turn of the main runloop
// (which is the outer runloop if recording started while paused in the debugger).
pushCurrentRecord(JSON::Object::create(), TimelineRecordType::RenderingFrame, false, nullptr);
m_runLoopNestingLevel = 1;
#elif USE(GLIB_EVENT_LOOP)
m_runLoopObserver = makeUnique<RunLoop::Observer>([this](RunLoop::Event event, const String& name) {
if (!m_tracking || m_environment.debugger().isPaused())
return;
switch (event) {
case RunLoop::Event::WillDispatch:
pushCurrentRecord(TimelineRecordFactory::createRenderingFrameData(name), TimelineRecordType::RenderingFrame, false, nullptr);
break;
case RunLoop::Event::DidDispatch:
didCompleteCurrentRecord(TimelineRecordType::RenderingFrame);
break;
}
});
RunLoop::current().observe(*m_runLoopObserver);
#endif
m_frontendDispatcher->recordingStarted(timestamp());
if (auto* client = m_inspectedPage.inspectorController().inspectorClient())
client->timelineRecordingChanged(true);
}
void InspectorTimelineAgent::internalStop()
{
if (!m_tracking)
return;
m_instrumentingAgents.setTrackingTimelineAgent(nullptr);
m_environment.debugger().removeObserver(*this, true);
#if PLATFORM(COCOA)
m_frameStartObserver = nullptr;
m_frameStopObserver = nullptr;
m_runLoopNestingLevel = 0;
#elif USE(GLIB_EVENT_LOOP)
m_runLoopObserver = nullptr;
#endif
// Complete all pending records to prevent discarding events that are currently in progress.
while (!m_recordStack.isEmpty())
didCompleteCurrentRecord(m_recordStack.last().type);
m_recordStack.clear();
m_tracking = false;
m_startedComposite = false;
m_autoCapturePhase = AutoCapturePhase::None;
m_frontendDispatcher->recordingStopped(timestamp());
if (auto* client = m_inspectedPage.inspectorController().inspectorClient())
client->timelineRecordingChanged(false);
}
double InspectorTimelineAgent::timestamp()
{
return m_environment.executionStopwatch().elapsedTime().seconds();
}
static Frame* frame(JSC::JSGlobalObject* globalObject)
{
auto context = executionContext(globalObject);
return is<Document>(context) ? downcast<Document>(*context).frame() : nullptr;
}
void InspectorTimelineAgent::startFromConsole(JSC::JSGlobalObject* globalObject, const String& title)
{
// Allow duplicate unnamed profiles. Disallow duplicate named profiles.
if (!title.isEmpty()) {
for (const TimelineRecordEntry& record : m_pendingConsoleProfileRecords) {
auto recordTitle = record.data->getString("title"_s);
if (recordTitle == title) {
if (auto* consoleAgent = m_instrumentingAgents.webConsoleAgent()) {
// FIXME: Send an enum to the frontend for localization?
String warning = title.isEmpty() ? "Unnamed Profile already exists"_s : makeString("Profile \"", ScriptArguments::truncateStringForConsoleMessage(title), "\" already exists");
consoleAgent->addMessageToConsole(makeUnique<ConsoleMessage>(MessageSource::ConsoleAPI, MessageType::Profile, MessageLevel::Warning, warning));
}
return;
}
}
}
if (!m_tracking && m_pendingConsoleProfileRecords.isEmpty())
startProgrammaticCapture();
m_pendingConsoleProfileRecords.append(createRecordEntry(TimelineRecordFactory::createConsoleProfileData(title), TimelineRecordType::ConsoleProfile, true, frame(globalObject)));
}
void InspectorTimelineAgent::stopFromConsole(JSC::JSGlobalObject*, const String& title)
{
// Stop profiles in reverse order. If the title is empty, then stop the last profile.
// Otherwise, match the title of the profile to stop.
for (int i = m_pendingConsoleProfileRecords.size() - 1; i >= 0; --i) {
const TimelineRecordEntry& record = m_pendingConsoleProfileRecords[i];
auto recordTitle = record.data->getString("title"_s);
if (title.isEmpty() || recordTitle == title) {
didCompleteRecordEntry(record);
m_pendingConsoleProfileRecords.remove(i);
if (!m_trackingFromFrontend && m_pendingConsoleProfileRecords.isEmpty())
stopProgrammaticCapture();
return;
}
}
if (auto* consoleAgent = m_instrumentingAgents.webConsoleAgent()) {
// FIXME: Send an enum to the frontend for localization?
String warning = title.isEmpty() ? "No profiles exist"_s : makeString("Profile \"", ScriptArguments::truncateStringForConsoleMessage(title), "\" does not exist");
consoleAgent->addMessageToConsole(makeUnique<ConsoleMessage>(MessageSource::ConsoleAPI, MessageType::ProfileEnd, MessageLevel::Warning, warning));
}
}
void InspectorTimelineAgent::willCallFunction(const String& scriptName, int scriptLine, int scriptColumn, Frame* frame)
{
pushCurrentRecord(TimelineRecordFactory::createFunctionCallData(scriptName, scriptLine, scriptColumn), TimelineRecordType::FunctionCall, true, frame);
}
void InspectorTimelineAgent::didCallFunction(Frame*)
{
didCompleteCurrentRecord(TimelineRecordType::FunctionCall);
}
void InspectorTimelineAgent::willDispatchEvent(const Event& event, Frame* frame)
{
pushCurrentRecord(TimelineRecordFactory::createEventDispatchData(event), TimelineRecordType::EventDispatch, false, frame);
}
void InspectorTimelineAgent::didDispatchEvent(bool defaultPrevented)
{
if (m_recordStack.isEmpty())
return;
auto& entry = m_recordStack.last();
ASSERT(entry.type == TimelineRecordType::EventDispatch);
entry.data->setBoolean("defaultPrevented"_s, defaultPrevented);
didCompleteCurrentRecord(TimelineRecordType::EventDispatch);
}
void InspectorTimelineAgent::didInvalidateLayout(Frame& frame)
{
appendRecord(JSON::Object::create(), TimelineRecordType::InvalidateLayout, true, &frame);
}
void InspectorTimelineAgent::willLayout(Frame& frame)
{
pushCurrentRecord(JSON::Object::create(), TimelineRecordType::Layout, true, &frame);
}
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::appendLayoutRoot(entry.data.get(), quads[0]);
else
ASSERT_NOT_REACHED();
didCompleteCurrentRecord(TimelineRecordType::Layout);
}
void InspectorTimelineAgent::didScheduleStyleRecalculation(Frame* frame)
{
appendRecord(JSON::Object::create(), TimelineRecordType::ScheduleStyleRecalculation, true, frame);
}
void InspectorTimelineAgent::willRecalculateStyle(Frame* frame)
{
pushCurrentRecord(JSON::Object::create(), TimelineRecordType::RecalculateStyles, true, frame);
}
void InspectorTimelineAgent::didRecalculateStyle()
{
didCompleteCurrentRecord(TimelineRecordType::RecalculateStyles);
}
void InspectorTimelineAgent::willComposite(Frame& frame)
{
ASSERT(!m_startedComposite);
pushCurrentRecord(JSON::Object::create(), TimelineRecordType::Composite, true, &frame);
m_startedComposite = true;
}
void InspectorTimelineAgent::didComposite()
{
if (m_startedComposite)
didCompleteCurrentRecord(TimelineRecordType::Composite);
m_startedComposite = false;
if (m_instruments.contains(Protocol::Timeline::Instrument::Screenshot))
captureScreenshot();
}
void InspectorTimelineAgent::willPaint(Frame& frame)
{
if (m_isCapturingScreenshot)
return;
pushCurrentRecord(JSON::Object::create(), TimelineRecordType::Paint, true, &frame);
}
void InspectorTimelineAgent::didPaint(RenderObject& renderer, const LayoutRect& clipRect)
{
if (m_isCapturingScreenshot)
return;
if (m_recordStack.isEmpty())
return;
TimelineRecordEntry& entry = m_recordStack.last();
ASSERT(entry.type == TimelineRecordType::Paint);
auto clipQuadInRootView = renderer.view().frameView().contentsToRootView(renderer.localToAbsoluteQuad({ clipRect }));
entry.data = TimelineRecordFactory::createPaintData(clipQuadInRootView);
didCompleteCurrentRecord(TimelineRecordType::Paint);
}
void InspectorTimelineAgent::didInstallTimer(int timerId, Seconds timeout, bool singleShot, Frame* frame)
{
appendRecord(TimelineRecordFactory::createTimerInstallData(timerId, timeout, singleShot), TimelineRecordType::TimerInstall, true, frame);
}
void InspectorTimelineAgent::didRemoveTimer(int timerId, Frame* frame)
{
appendRecord(TimelineRecordFactory::createGenericTimerData(timerId), TimelineRecordType::TimerRemove, true, frame);
}
void InspectorTimelineAgent::willFireTimer(int timerId, Frame* frame)
{
pushCurrentRecord(TimelineRecordFactory::createGenericTimerData(timerId), TimelineRecordType::TimerFire, false, frame);
}
void InspectorTimelineAgent::didFireTimer()
{
didCompleteCurrentRecord(TimelineRecordType::TimerFire);
}
void InspectorTimelineAgent::willEvaluateScript(const String& url, int lineNumber, int columnNumber, Frame& frame)
{
pushCurrentRecord(TimelineRecordFactory::createEvaluateScriptData(url, lineNumber, columnNumber), TimelineRecordType::EvaluateScript, true, &frame);
}
void InspectorTimelineAgent::didEvaluateScript(Frame&)
{
didCompleteCurrentRecord(TimelineRecordType::EvaluateScript);
}
void InspectorTimelineAgent::didTimeStamp(Frame& frame, const String& message)
{
appendRecord(TimelineRecordFactory::createTimeStampData(message), TimelineRecordType::TimeStamp, true, &frame);
}
void InspectorTimelineAgent::time(Frame& frame, const String& message)
{
appendRecord(TimelineRecordFactory::createTimeStampData(message), TimelineRecordType::Time, true, &frame);
}
void InspectorTimelineAgent::timeEnd(Frame& frame, const String& message)
{
appendRecord(TimelineRecordFactory::createTimeStampData(message), TimelineRecordType::TimeEnd, true, &frame);
}
void InspectorTimelineAgent::mainFrameStartedLoading()
{
if (m_tracking)
return;
if (!m_autoCaptureEnabled)
return;
if (m_instruments.isEmpty())
return;
m_autoCapturePhase = AutoCapturePhase::BeforeLoad;
// Pre-emptively disable breakpoints. The frontend must re-enable them.
if (auto* webDebuggerAgent = m_instrumentingAgents.enabledWebDebuggerAgent())
webDebuggerAgent->setBreakpointsActive(false);
// Inform the frontend we started an auto capture. The frontend must stop capture.
m_frontendDispatcher->autoCaptureStarted();
toggleInstruments(InstrumentState::Start);
}
void InspectorTimelineAgent::mainFrameNavigated()
{
if (m_autoCapturePhase == AutoCapturePhase::BeforeLoad) {
m_autoCapturePhase = AutoCapturePhase::FirstNavigation;
toggleInstruments(InstrumentState::Start);
m_autoCapturePhase = AutoCapturePhase::AfterFirstNavigation;
}
}
void InspectorTimelineAgent::startProgrammaticCapture()
{
ASSERT(!m_tracking);
// Disable breakpoints during programmatic capture.
if (auto* webDebuggerAgent = m_instrumentingAgents.enabledWebDebuggerAgent()) {
m_programmaticCaptureRestoreBreakpointActiveValue = webDebuggerAgent->breakpointsActive();
if (m_programmaticCaptureRestoreBreakpointActiveValue)
webDebuggerAgent->setBreakpointsActive(false);
} else
m_programmaticCaptureRestoreBreakpointActiveValue = false;
toggleScriptProfilerInstrument(InstrumentState::Start); // Ensure JavaScript samping data.
toggleTimelineInstrument(InstrumentState::Start); // Ensure Console Profile event records.
toggleInstruments(InstrumentState::Start); // Any other instruments the frontend wants us to record.
}
void InspectorTimelineAgent::stopProgrammaticCapture()
{
ASSERT(m_tracking);
ASSERT(!m_trackingFromFrontend);
toggleInstruments(InstrumentState::Stop);
toggleTimelineInstrument(InstrumentState::Stop);
toggleScriptProfilerInstrument(InstrumentState::Stop);
// Re-enable breakpoints if they were enabled.
if (m_programmaticCaptureRestoreBreakpointActiveValue) {
if (auto* webDebuggerAgent = m_instrumentingAgents.enabledWebDebuggerAgent())
webDebuggerAgent->setBreakpointsActive(true);
}
}
void InspectorTimelineAgent::toggleInstruments(InstrumentState state)
{
for (auto instrumentType : m_instruments) {
switch (instrumentType) {
case Protocol::Timeline::Instrument::ScriptProfiler: {
toggleScriptProfilerInstrument(state);
break;
}
case Protocol::Timeline::Instrument::Heap: {
toggleHeapInstrument(state);
break;
}
case Protocol::Timeline::Instrument::CPU: {
toggleCPUInstrument(state);
break;
}
case Protocol::Timeline::Instrument::Memory: {
toggleMemoryInstrument(state);
break;
}
case Protocol::Timeline::Instrument::Timeline:
toggleTimelineInstrument(state);
break;
case Protocol::Timeline::Instrument::Animation:
toggleAnimationInstrument(state);
break;
case Protocol::Timeline::Instrument::Screenshot: {
break;
}
}
}
}
void InspectorTimelineAgent::toggleScriptProfilerInstrument(InstrumentState state)
{
if (auto* scriptProfilerAgent = m_instrumentingAgents.persistentScriptProfilerAgent()) {
if (state == InstrumentState::Start)
scriptProfilerAgent->startTracking(true);
else
scriptProfilerAgent->stopTracking();
}
}
void InspectorTimelineAgent::toggleHeapInstrument(InstrumentState state)
{
if (auto* heapAgent = m_instrumentingAgents.enabledPageHeapAgent()) {
if (state == InstrumentState::Start) {
if (m_autoCapturePhase == AutoCapturePhase::None || m_autoCapturePhase == AutoCapturePhase::FirstNavigation)
heapAgent->startTracking();
} else
heapAgent->stopTracking();
}
}
void InspectorTimelineAgent::toggleCPUInstrument(InstrumentState state)
{
#if ENABLE(RESOURCE_USAGE)
if (auto* cpuProfilerAgent = m_instrumentingAgents.persistentCPUProfilerAgent()) {
if (state == InstrumentState::Start)
cpuProfilerAgent->startTracking();
else
cpuProfilerAgent->stopTracking();
}
#else
UNUSED_PARAM(state);
#endif
}
void InspectorTimelineAgent::toggleMemoryInstrument(InstrumentState state)
{
#if ENABLE(RESOURCE_USAGE)
if (auto* memoryAgent = m_instrumentingAgents.persistentMemoryAgent()) {
if (state == InstrumentState::Start)
memoryAgent->startTracking();
else
memoryAgent->stopTracking();
}
#else
UNUSED_PARAM(state);
#endif
}
void InspectorTimelineAgent::toggleTimelineInstrument(InstrumentState state)
{
if (state == InstrumentState::Start)
internalStart(std::nullopt);
else
internalStop();
}
void InspectorTimelineAgent::toggleAnimationInstrument(InstrumentState state)
{
if (auto* animationAgent = m_instrumentingAgents.persistentAnimationAgent()) {
if (state == InstrumentState::Start)
animationAgent->startTracking();
else
animationAgent->stopTracking();
}
}
void InspectorTimelineAgent::captureScreenshot()
{
SetForScope isTakingScreenshot(m_isCapturingScreenshot, true);
auto snapshotStartTime = timestamp();
auto& frame = m_inspectedPage.mainFrame();
auto viewportRect = m_inspectedPage.mainFrame().view()->unobscuredContentRect();
if (auto snapshot = snapshotFrameRect(frame, viewportRect, { { }, PixelFormat::BGRA8, DestinationColorSpace::SRGB() })) {
auto snapshotRecord = TimelineRecordFactory::createScreenshotData(snapshot->toDataURL("image/png"_s));
pushCurrentRecord(WTFMove(snapshotRecord), TimelineRecordType::Screenshot, false, &frame, snapshotStartTime);
didCompleteCurrentRecord(TimelineRecordType::Screenshot);
}
}
void InspectorTimelineAgent::didRequestAnimationFrame(int callbackId, Frame* frame)
{
appendRecord(TimelineRecordFactory::createAnimationFrameData(callbackId), TimelineRecordType::RequestAnimationFrame, true, frame);
}
void InspectorTimelineAgent::didCancelAnimationFrame(int callbackId, Frame* frame)
{
appendRecord(TimelineRecordFactory::createAnimationFrameData(callbackId), TimelineRecordType::CancelAnimationFrame, true, frame);
}
void InspectorTimelineAgent::willFireAnimationFrame(int callbackId, Frame* frame)
{
pushCurrentRecord(TimelineRecordFactory::createAnimationFrameData(callbackId), TimelineRecordType::FireAnimationFrame, false, frame);
}
void InspectorTimelineAgent::didFireAnimationFrame()
{
didCompleteCurrentRecord(TimelineRecordType::FireAnimationFrame);
}
void InspectorTimelineAgent::willFireObserverCallback(const String& callbackType, Frame* frame)
{
pushCurrentRecord(TimelineRecordFactory::createObserverCallbackData(callbackType), TimelineRecordType::ObserverCallback, false, frame);
}
void InspectorTimelineAgent::didFireObserverCallback()
{
didCompleteCurrentRecord(TimelineRecordType::ObserverCallback);
}
void InspectorTimelineAgent::breakpointActionProbe(JSC::JSGlobalObject* lexicalGlobalObject, JSC::BreakpointActionID actionID, unsigned /*batchId*/, unsigned sampleId, JSC::JSValue)
{
appendRecord(TimelineRecordFactory::createProbeSampleData(actionID, sampleId), TimelineRecordType::ProbeSample, false, frame(lexicalGlobalObject));
}
static Protocol::Timeline::EventType toProtocol(TimelineRecordType type)
{
switch (type) {
case TimelineRecordType::EventDispatch:
return Protocol::Timeline::EventType::EventDispatch;
case TimelineRecordType::ScheduleStyleRecalculation:
return Protocol::Timeline::EventType::ScheduleStyleRecalculation;
case TimelineRecordType::RecalculateStyles:
return Protocol::Timeline::EventType::RecalculateStyles;
case TimelineRecordType::InvalidateLayout:
return Protocol::Timeline::EventType::InvalidateLayout;
case TimelineRecordType::Layout:
return Protocol::Timeline::EventType::Layout;
case TimelineRecordType::Paint:
return Protocol::Timeline::EventType::Paint;
case TimelineRecordType::Composite:
return Protocol::Timeline::EventType::Composite;
case TimelineRecordType::RenderingFrame:
return Protocol::Timeline::EventType::RenderingFrame;
case TimelineRecordType::TimerInstall:
return Protocol::Timeline::EventType::TimerInstall;
case TimelineRecordType::TimerRemove:
return Protocol::Timeline::EventType::TimerRemove;
case TimelineRecordType::TimerFire:
return Protocol::Timeline::EventType::TimerFire;
case TimelineRecordType::EvaluateScript:
return Protocol::Timeline::EventType::EvaluateScript;
case TimelineRecordType::TimeStamp:
return Protocol::Timeline::EventType::TimeStamp;
case TimelineRecordType::Time:
return Protocol::Timeline::EventType::Time;
case TimelineRecordType::TimeEnd:
return Protocol::Timeline::EventType::TimeEnd;
case TimelineRecordType::FunctionCall:
return Protocol::Timeline::EventType::FunctionCall;
case TimelineRecordType::ProbeSample:
return Protocol::Timeline::EventType::ProbeSample;
case TimelineRecordType::ConsoleProfile:
return Protocol::Timeline::EventType::ConsoleProfile;
case TimelineRecordType::RequestAnimationFrame:
return Protocol::Timeline::EventType::RequestAnimationFrame;
case TimelineRecordType::CancelAnimationFrame:
return Protocol::Timeline::EventType::CancelAnimationFrame;
case TimelineRecordType::FireAnimationFrame:
return Protocol::Timeline::EventType::FireAnimationFrame;
case TimelineRecordType::ObserverCallback:
return Protocol::Timeline::EventType::ObserverCallback;
case TimelineRecordType::Screenshot:
return Protocol::Timeline::EventType::Screenshot;
}
return Protocol::Timeline::EventType::TimeStamp;
}
void InspectorTimelineAgent::addRecordToTimeline(Ref<JSON::Object>&& record, TimelineRecordType type)
{
record->setString(Protocol::Timeline::TimelineEvent::typeKey, Protocol::Helpers::getEnumConstantValue(toProtocol(type)));
if (m_recordStack.isEmpty()) {
// FIXME: runtimeCast is a hack. We do it because we can't build TimelineEvent directly now.
auto recordObject = Protocol::BindingTraits<Protocol::Timeline::TimelineEvent>::runtimeCast(WTFMove(record));
sendEvent(WTFMove(recordObject));
} else {
const TimelineRecordEntry& parent = m_recordStack.last();
// Nested paint records are an implementation detail and add no information not already contained in the parent.
if (type == TimelineRecordType::Paint && parent.type == type)
return;
parent.children->pushObject(WTFMove(record));
}
}
void InspectorTimelineAgent::setFrameIdentifier(JSON::Object* record, Frame* frame)
{
if (!frame)
return;
auto* pageAgent = m_instrumentingAgents.enabledPageAgent();
if (!pageAgent)
return;
record->setString("frameId"_s, pageAgent->frameId(frame));
}
void InspectorTimelineAgent::didCompleteRecordEntry(const TimelineRecordEntry& entry)
{
entry.record->setObject(Protocol::Timeline::TimelineEvent::dataKey, entry.data.copyRef());
if (entry.children)
entry.record->setArray(Protocol::Timeline::TimelineEvent::childrenKey, *entry.children);
entry.record->setDouble("endTime"_s, timestamp());
addRecordToTimeline(entry.record.copyRef(), entry.type);
}
void InspectorTimelineAgent::didCompleteCurrentRecord(TimelineRecordType 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()) {
TimelineRecordEntry entry = m_recordStack.last();
m_recordStack.removeLast();
ASSERT_UNUSED(type, entry.type == type);
// Don't send RenderingFrame records that have no children to reduce noise.
if (entry.type == TimelineRecordType::RenderingFrame && !entry.children->length())
return;
didCompleteRecordEntry(entry);
}
}
void InspectorTimelineAgent::appendRecord(Ref<JSON::Object>&& data, TimelineRecordType type, bool captureCallStack, Frame* frame)
{
Ref<JSON::Object> record = TimelineRecordFactory::createGenericRecord(timestamp(), captureCallStack ? m_maxCallStackDepth : 0);
record->setObject(Protocol::Timeline::TimelineEvent::dataKey, WTFMove(data));
setFrameIdentifier(&record.get(), frame);
addRecordToTimeline(WTFMove(record), type);
}
void InspectorTimelineAgent::sendEvent(Ref<JSON::Object>&& event)
{
// FIXME: runtimeCast is a hack. We do it because we can't build TimelineEvent directly now.
auto recordChecked = Protocol::BindingTraits<Protocol::Timeline::TimelineEvent>::runtimeCast(WTFMove(event));
m_frontendDispatcher->eventRecorded(WTFMove(recordChecked));
}
InspectorTimelineAgent::TimelineRecordEntry InspectorTimelineAgent::createRecordEntry(Ref<JSON::Object>&& data, TimelineRecordType type, bool captureCallStack, Frame* frame, std::optional<double> startTime)
{
Ref<JSON::Object> record = TimelineRecordFactory::createGenericRecord(startTime.value_or(timestamp()), captureCallStack ? m_maxCallStackDepth : 0);
setFrameIdentifier(&record.get(), frame);
return TimelineRecordEntry(WTFMove(record), WTFMove(data), JSON::Array::create(), type);
}
void InspectorTimelineAgent::pushCurrentRecord(Ref<JSON::Object>&& data, TimelineRecordType type, bool captureCallStack, Frame* frame, std::optional<double> startTime)
{
pushCurrentRecord(createRecordEntry(WTFMove(data), type, captureCallStack, frame, startTime));
}
} // namespace WebCore