| /* |
| * 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 |