| // Copyright 2015 the V8 project authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #ifndef V8_INSPECTOR_V8_DEBUGGER_AGENT_IMPL_H_ |
| #define V8_INSPECTOR_V8_DEBUGGER_AGENT_IMPL_H_ |
| |
| #include <deque> |
| #include <unordered_map> |
| #include <vector> |
| |
| #include "src/base/macros.h" |
| #include "src/debug/debug-interface.h" |
| #include "src/debug/interface-types.h" |
| #include "src/inspector/protocol/Debugger.h" |
| #include "src/inspector/protocol/Forward.h" |
| |
| namespace v8_inspector { |
| |
| struct ScriptBreakpoint; |
| class V8Debugger; |
| class V8DebuggerScript; |
| class V8InspectorImpl; |
| class V8InspectorSessionImpl; |
| class V8Regex; |
| |
| using protocol::Maybe; |
| using protocol::Response; |
| |
| class V8DebuggerAgentImpl : public protocol::Debugger::Backend { |
| public: |
| enum BreakpointSource { |
| UserBreakpointSource, |
| DebugCommandBreakpointSource, |
| MonitorCommandBreakpointSource |
| }; |
| |
| V8DebuggerAgentImpl(V8InspectorSessionImpl*, protocol::FrontendChannel*, |
| protocol::DictionaryValue* state); |
| ~V8DebuggerAgentImpl() override; |
| void restore(); |
| |
| // Part of the protocol. |
| Response enable(Maybe<double> maxScriptsCacheSize, |
| String16* outDebuggerId) override; |
| Response disable() override; |
| Response setBreakpointsActive(bool active) override; |
| Response setSkipAllPauses(bool skip) override; |
| Response setBreakpointByUrl( |
| int lineNumber, Maybe<String16> optionalURL, |
| Maybe<String16> optionalURLRegex, Maybe<String16> optionalScriptHash, |
| Maybe<int> optionalColumnNumber, Maybe<String16> optionalCondition, |
| String16*, |
| std::unique_ptr<protocol::Array<protocol::Debugger::Location>>* locations) |
| override; |
| Response setBreakpoint( |
| std::unique_ptr<protocol::Debugger::Location>, |
| Maybe<String16> optionalCondition, String16*, |
| std::unique_ptr<protocol::Debugger::Location>* actualLocation) override; |
| Response setBreakpointOnFunctionCall(const String16& functionObjectId, |
| Maybe<String16> optionalCondition, |
| String16* outBreakpointId) override; |
| Response setInstrumentationBreakpoint(const String16& instrumentation, |
| String16* outBreakpointId) override; |
| Response removeBreakpoint(const String16& breakpointId) override; |
| Response continueToLocation(std::unique_ptr<protocol::Debugger::Location>, |
| Maybe<String16> targetCallFrames) override; |
| Response getStackTrace( |
| std::unique_ptr<protocol::Runtime::StackTraceId> inStackTraceId, |
| std::unique_ptr<protocol::Runtime::StackTrace>* outStackTrace) override; |
| Response searchInContent( |
| const String16& scriptId, const String16& query, |
| Maybe<bool> optionalCaseSensitive, Maybe<bool> optionalIsRegex, |
| std::unique_ptr<protocol::Array<protocol::Debugger::SearchMatch>>*) |
| override; |
| Response getPossibleBreakpoints( |
| std::unique_ptr<protocol::Debugger::Location> start, |
| Maybe<protocol::Debugger::Location> end, Maybe<bool> restrictToFunction, |
| std::unique_ptr<protocol::Array<protocol::Debugger::BreakLocation>>* |
| locations) override; |
| Response setScriptSource( |
| const String16& inScriptId, const String16& inScriptSource, |
| Maybe<bool> dryRun, |
| Maybe<protocol::Array<protocol::Debugger::CallFrame>>* optOutCallFrames, |
| Maybe<bool>* optOutStackChanged, |
| Maybe<protocol::Runtime::StackTrace>* optOutAsyncStackTrace, |
| Maybe<protocol::Runtime::StackTraceId>* optOutAsyncStackTraceId, |
| Maybe<protocol::Runtime::ExceptionDetails>* optOutCompileError) override; |
| Response restartFrame( |
| const String16& callFrameId, |
| std::unique_ptr<protocol::Array<protocol::Debugger::CallFrame>>* |
| newCallFrames, |
| Maybe<protocol::Runtime::StackTrace>* asyncStackTrace, |
| Maybe<protocol::Runtime::StackTraceId>* asyncStackTraceId) override; |
| Response getScriptSource(const String16& scriptId, |
| String16* scriptSource) override; |
| Response pause() override; |
| Response resume() override; |
| Response stepOver() override; |
| Response stepInto(Maybe<bool> inBreakOnAsyncCall) override; |
| Response stepOut() override; |
| Response pauseOnAsyncCall(std::unique_ptr<protocol::Runtime::StackTraceId> |
| inParentStackTraceId) override; |
| Response setPauseOnExceptions(const String16& pauseState) override; |
| Response evaluateOnCallFrame( |
| const String16& callFrameId, const String16& expression, |
| Maybe<String16> objectGroup, Maybe<bool> includeCommandLineAPI, |
| Maybe<bool> silent, Maybe<bool> returnByValue, |
| Maybe<bool> generatePreview, Maybe<bool> throwOnSideEffect, |
| Maybe<double> timeout, |
| std::unique_ptr<protocol::Runtime::RemoteObject>* result, |
| Maybe<protocol::Runtime::ExceptionDetails>*) override; |
| Response setVariableValue( |
| int scopeNumber, const String16& variableName, |
| std::unique_ptr<protocol::Runtime::CallArgument> newValue, |
| const String16& callFrame) override; |
| Response setReturnValue( |
| std::unique_ptr<protocol::Runtime::CallArgument> newValue) override; |
| Response setAsyncCallStackDepth(int depth) override; |
| Response setBlackboxPatterns( |
| std::unique_ptr<protocol::Array<String16>> patterns) override; |
| Response setBlackboxedRanges( |
| const String16& scriptId, |
| std::unique_ptr<protocol::Array<protocol::Debugger::ScriptPosition>> |
| positions) override; |
| |
| bool enabled() const { return m_enabled; } |
| |
| void setBreakpointFor(v8::Local<v8::Function> function, |
| v8::Local<v8::String> condition, |
| BreakpointSource source); |
| void removeBreakpointFor(v8::Local<v8::Function> function, |
| BreakpointSource source); |
| void schedulePauseOnNextStatement( |
| const String16& breakReason, |
| std::unique_ptr<protocol::DictionaryValue> data); |
| void cancelPauseOnNextStatement(); |
| void breakProgram(const String16& breakReason, |
| std::unique_ptr<protocol::DictionaryValue> data); |
| |
| void reset(); |
| |
| // Interface for V8InspectorImpl |
| void didPause(int contextId, v8::Local<v8::Value> exception, |
| const std::vector<v8::debug::BreakpointId>& hitBreakpoints, |
| v8::debug::ExceptionType exceptionType, bool isUncaught, |
| bool isOOMBreak, bool isAssert); |
| void didContinue(); |
| void didParseSource(std::unique_ptr<V8DebuggerScript>, bool success); |
| |
| bool isFunctionBlackboxed(const String16& scriptId, |
| const v8::debug::Location& start, |
| const v8::debug::Location& end); |
| |
| bool acceptsPause(bool isOOMBreak) const; |
| |
| void ScriptCollected(const V8DebuggerScript* script); |
| |
| v8::Isolate* isolate() { return m_isolate; } |
| |
| private: |
| void enableImpl(); |
| |
| Response currentCallFrames( |
| std::unique_ptr<protocol::Array<protocol::Debugger::CallFrame>>*); |
| std::unique_ptr<protocol::Runtime::StackTrace> currentAsyncStackTrace(); |
| std::unique_ptr<protocol::Runtime::StackTraceId> currentExternalStackTrace(); |
| std::unique_ptr<protocol::Runtime::StackTraceId> currentScheduledAsyncCall(); |
| |
| void setPauseOnExceptionsImpl(int); |
| |
| std::unique_ptr<protocol::Debugger::Location> setBreakpointImpl( |
| const String16& breakpointId, const String16& scriptId, |
| const String16& condition, int lineNumber, int columnNumber); |
| void setBreakpointImpl(const String16& breakpointId, |
| v8::Local<v8::Function> function, |
| v8::Local<v8::String> condition); |
| void removeBreakpointImpl(const String16& breakpointId); |
| void clearBreakDetails(); |
| |
| void internalSetAsyncCallStackDepth(int); |
| void increaseCachedSkipStackGeneration(); |
| |
| Response setBlackboxPattern(const String16& pattern); |
| void resetBlackboxedStateCache(); |
| |
| bool isPaused() const; |
| |
| void setScriptInstrumentationBreakpointIfNeeded(V8DebuggerScript* script); |
| |
| using ScriptsMap = |
| std::unordered_map<String16, std::unique_ptr<V8DebuggerScript>>; |
| using BreakpointIdToDebuggerBreakpointIdsMap = |
| std::unordered_map<String16, std::vector<v8::debug::BreakpointId>>; |
| using DebuggerBreakpointIdToBreakpointIdMap = |
| std::unordered_map<v8::debug::BreakpointId, String16>; |
| |
| V8InspectorImpl* m_inspector; |
| V8Debugger* m_debugger; |
| V8InspectorSessionImpl* m_session; |
| bool m_enabled; |
| protocol::DictionaryValue* m_state; |
| protocol::Debugger::Frontend m_frontend; |
| v8::Isolate* m_isolate; |
| ScriptsMap m_scripts; |
| BreakpointIdToDebuggerBreakpointIdsMap m_breakpointIdToDebuggerBreakpointIds; |
| DebuggerBreakpointIdToBreakpointIdMap m_debuggerBreakpointIdToBreakpointId; |
| std::unordered_map<v8::debug::BreakpointId, |
| std::unique_ptr<protocol::DictionaryValue>> |
| m_breakpointsOnScriptRun; |
| |
| size_t m_maxScriptCacheSize = 0; |
| size_t m_cachedScriptSize = 0; |
| std::deque<String16> m_cachedScriptIds; |
| |
| using BreakReason = |
| std::pair<String16, std::unique_ptr<protocol::DictionaryValue>>; |
| std::vector<BreakReason> m_breakReason; |
| |
| void pushBreakDetails( |
| const String16& breakReason, |
| std::unique_ptr<protocol::DictionaryValue> breakAuxData); |
| void popBreakDetails(); |
| |
| bool m_skipAllPauses = false; |
| bool m_breakpointsActive = false; |
| |
| std::unique_ptr<V8Regex> m_blackboxPattern; |
| std::unordered_map<String16, std::vector<std::pair<int, int>>> |
| m_blackboxedPositions; |
| |
| DISALLOW_COPY_AND_ASSIGN(V8DebuggerAgentImpl); |
| }; |
| |
| } // namespace v8_inspector |
| |
| #endif // V8_INSPECTOR_V8_DEBUGGER_AGENT_IMPL_H_ |