blob: 7e8727666b16401555c07166aa60fe740caf59a2 [file] [log] [blame]
/*
* Copyright (C) 2010, 2013, 2015-2016 Apple Inc. All rights reserved.
* Copyright (C) 2010, 2011 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:
*
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. 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.
* 3. Neither the name of Apple Inc. ("Apple") 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 APPLE AND ITS 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 APPLE OR ITS 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.
*/
#pragma once
#include "Debugger.h"
#include "InspectorAgentBase.h"
#include "InspectorBackendDispatchers.h"
#include "InspectorFrontendDispatchers.h"
#include "ScriptBreakpoint.h"
#include "ScriptDebugListener.h"
#include <wtf/Forward.h>
#include <wtf/HashMap.h>
#include <wtf/HashSet.h>
#include <wtf/Noncopyable.h>
#include <wtf/Vector.h>
namespace Inspector {
class AsyncStackTrace;
class InjectedScript;
class InjectedScriptManager;
class ScriptDebugServer;
typedef String ErrorString;
class JS_EXPORT_PRIVATE InspectorDebuggerAgent : public InspectorAgentBase, public DebuggerBackendDispatcherHandler, public ScriptDebugListener {
WTF_MAKE_NONCOPYABLE(InspectorDebuggerAgent);
WTF_MAKE_FAST_ALLOCATED;
public:
~InspectorDebuggerAgent() override;
static const char* backtraceObjectGroup;
// InspectorAgentBase
void didCreateFrontendAndBackend(FrontendRouter*, BackendDispatcher*) final;
void willDestroyFrontendAndBackend(DisconnectReason) final;
virtual bool enabled() const { return m_enabled; }
// DebuggerBackendDispatcherHandler
void enable(ErrorString&) final;
void disable(ErrorString&) final;
void setAsyncStackTraceDepth(ErrorString&, int depth) final;
void setBreakpointsActive(ErrorString&, bool active) final;
void setBreakpointByUrl(ErrorString&, int lineNumber, const String* optionalURL, const String* optionalURLRegex, const int* optionalColumnNumber, const JSON::Object* options, Protocol::Debugger::BreakpointId*, RefPtr<JSON::ArrayOf<Protocol::Debugger::Location>>& locations) final;
void setBreakpoint(ErrorString&, const JSON::Object& location, const JSON::Object* options, Protocol::Debugger::BreakpointId*, RefPtr<Protocol::Debugger::Location>& actualLocation) final;
void removeBreakpoint(ErrorString&, const String& breakpointIdentifier) final;
void continueUntilNextRunLoop(ErrorString&) final;
void continueToLocation(ErrorString&, const JSON::Object& location) final;
void stepOver(ErrorString&) final;
void stepInto(ErrorString&) final;
void stepOut(ErrorString&) final;
void pause(ErrorString&) final;
void resume(ErrorString&) final;
void searchInContent(ErrorString&, const String& scriptID, const String& query, const bool* optionalCaseSensitive, const bool* optionalIsRegex, RefPtr<JSON::ArrayOf<Protocol::GenericTypes::SearchMatch>>&) final;
void getScriptSource(ErrorString&, const String& scriptID, String* scriptSource) final;
void getFunctionDetails(ErrorString&, const String& functionId, RefPtr<Protocol::Debugger::FunctionDetails>&) final;
void setPauseOnExceptions(ErrorString&, const String& pauseState) final;
void setPauseOnAssertions(ErrorString&, bool enabled) final;
void setPauseOnMicrotasks(ErrorString&, bool enabled) final;
void setPauseForInternalScripts(ErrorString&, bool shouldPause) final;
void evaluateOnCallFrame(ErrorString&, const String& callFrameId, const String& expression, const String* objectGroup, const bool* includeCommandLineAPI, const bool* doNotPauseOnExceptionsAndMuteConsole, const bool* returnByValue, const bool* generatePreview, const bool* saveResult, const bool* emulateUserGesture, RefPtr<Protocol::Runtime::RemoteObject>& result, Optional<bool>& wasThrown, Optional<int>& savedResultIndex) override;
void setShouldBlackboxURL(ErrorString&, const String& url, bool shouldBlackbox, const bool* caseSensitive, const bool* isRegex) final;
// ScriptDebugListener
void didParseSource(JSC::SourceID, const Script&) final;
void failedToParseSource(const String& url, const String& data, int firstLine, int errorLine, const String& errorMessage) final;
void willRunMicrotask() final;
void didRunMicrotask() final;
void didPause(JSC::ExecState&, JSC::JSValue callFrames, JSC::JSValue exceptionOrCaughtValue) final;
void didContinue() final;
void breakpointActionSound(int breakpointActionIdentifier) final;
void breakpointActionProbe(JSC::ExecState&, const ScriptBreakpointAction&, unsigned batchId, unsigned sampleId, JSC::JSValue sample) final;
bool isPaused() const;
bool breakpointsActive() const;
void setSuppressAllPauses(bool);
void handleConsoleAssert(const String& message);
enum class AsyncCallType {
DOMTimer,
EventListener,
PostMessage,
RequestAnimationFrame,
};
void didScheduleAsyncCall(JSC::ExecState*, AsyncCallType, int callbackId, bool singleShot);
void didCancelAsyncCall(AsyncCallType, int callbackId);
void willDispatchAsyncCall(AsyncCallType, int callbackId);
void didDispatchAsyncCall();
void schedulePauseOnNextStatement(DebuggerFrontendDispatcher::Reason, RefPtr<JSON::Object>&& data);
void cancelPauseOnNextStatement();
bool pauseOnNextStatementEnabled() const { return m_javaScriptPauseScheduled; }
void breakProgram(DebuggerFrontendDispatcher::Reason, RefPtr<JSON::Object>&& data);
void scriptExecutionBlockedByCSP(const String& directiveText);
class Listener {
public:
virtual ~Listener() { }
virtual void debuggerWasEnabled() = 0;
virtual void debuggerWasDisabled() = 0;
};
void addListener(Listener& listener) { m_listeners.add(&listener); }
void removeListener(Listener& listener) { m_listeners.remove(&listener); }
protected:
InspectorDebuggerAgent(AgentContext&);
virtual void enable();
virtual void disable(bool isBeingDestroyed);
InjectedScriptManager& injectedScriptManager() const { return m_injectedScriptManager; }
virtual InjectedScript injectedScriptForEval(ErrorString&, const int* executionContextId) = 0;
ScriptDebugServer& scriptDebugServer() { return m_scriptDebugServer; }
virtual void muteConsole() = 0;
virtual void unmuteConsole() = 0;
virtual String sourceMapURLForScript(const Script&);
void didClearGlobalObject();
virtual void didClearAsyncStackTraceData() { }
private:
bool shouldBlackboxURL(const String&) const;
Ref<JSON::ArrayOf<Protocol::Debugger::CallFrame>> currentCallFrames(const InjectedScript&);
void resolveBreakpoint(const Script&, JSC::Breakpoint&);
void setBreakpoint(JSC::Breakpoint&, bool& existing);
void didSetBreakpoint(const JSC::Breakpoint&, const String&, const ScriptBreakpoint&);
bool assertPaused(ErrorString&);
void clearDebuggerBreakpointState();
void clearInspectorBreakpointState();
void clearPauseDetails();
void clearExceptionValue();
void clearAsyncStackTraceData();
enum class ShouldDispatchResumed { No, WhenIdle, WhenContinued };
void registerIdleHandler();
void willStepAndMayBecomeIdle();
void didBecomeIdle();
void updatePauseReasonAndData(DebuggerFrontendDispatcher::Reason, RefPtr<JSON::Object>&& data);
RefPtr<JSON::Object> buildBreakpointPauseReason(JSC::BreakpointID);
RefPtr<JSON::Object> buildExceptionPauseReason(JSC::JSValue exception, const InjectedScript&);
bool breakpointActionsFromProtocol(ErrorString&, RefPtr<JSON::Array>& actions, BreakpointActions* result);
typedef std::pair<unsigned, int> AsyncCallIdentifier;
static AsyncCallIdentifier asyncCallIdentifier(AsyncCallType, int callbackId);
std::unique_ptr<DebuggerFrontendDispatcher> m_frontendDispatcher;
RefPtr<DebuggerBackendDispatcher> m_backendDispatcher;
ScriptDebugServer& m_scriptDebugServer;
InjectedScriptManager& m_injectedScriptManager;
HashMap<JSC::SourceID, Script> m_scripts;
struct BlackboxConfig {
String url;
bool caseSensitive { false };
bool isRegex { false };
};
Vector<BlackboxConfig> m_blackboxedURLs;
HashSet<Listener*> m_listeners;
JSC::ExecState* m_pausedScriptState { nullptr };
JSC::Strong<JSC::Unknown> m_currentCallStack;
HashMap<String, Vector<JSC::BreakpointID>> m_breakpointIdentifierToDebugServerBreakpointIDs;
HashMap<String, RefPtr<JSON::Object>> m_javaScriptBreakpoints;
HashMap<JSC::BreakpointID, String> m_debuggerBreakpointIdentifierToInspectorBreakpointIdentifier;
JSC::BreakpointID m_continueToLocationBreakpointID { JSC::noBreakpointID };
ShouldDispatchResumed m_conditionToDispatchResumed { ShouldDispatchResumed::No };
DebuggerFrontendDispatcher::Reason m_pauseReason;
RefPtr<JSON::Object> m_pauseData;
DebuggerFrontendDispatcher::Reason m_preBlackboxPauseReason;
RefPtr<JSON::Object> m_preBlackboxPauseData;
HashMap<AsyncCallIdentifier, RefPtr<AsyncStackTrace>> m_pendingAsyncCalls;
Optional<AsyncCallIdentifier> m_currentAsyncCallIdentifier;
int m_asyncStackTraceDepth { 0 };
bool m_enabled { false };
bool m_enablePauseWhenIdle { false };
bool m_pauseOnAssertionFailures { false };
bool m_pauseOnMicrotasks { false };
bool m_pauseForInternalScripts { false };
bool m_javaScriptPauseScheduled { false };
bool m_didPauseStopwatch { false };
bool m_hasExceptionValue { false };
bool m_registeredIdleCallback { false };
};
} // namespace Inspector