| /* |
| * Copyright (C) 2010 Apple Inc. All rights reserved. |
| * 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: |
| * |
| * 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 Computer, 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. |
| */ |
| |
| #include "config.h" |
| #include "core/inspector/InspectorDebuggerAgent.h" |
| |
| #include "bindings/core/dart/DartInspectorDebuggerAgent.h" |
| |
| #include "bindings/core/v8/ScriptCallStackFactory.h" |
| #include "bindings/core/v8/ScriptRegexp.h" |
| #include "bindings/core/v8/ScriptValue.h" |
| #include "bindings/core/v8/V8Binding.h" |
| #include "bindings/core/v8/V8RecursionScope.h" |
| #include "bindings/core/v8/V8ScriptRunner.h" |
| #include "core/dom/Microtask.h" |
| #include "core/inspector/AsyncCallChain.h" |
| #include "core/inspector/ContentSearchUtils.h" |
| #include "core/inspector/InjectedScript.h" |
| #include "core/inspector/InjectedScriptManager.h" |
| #include "core/inspector/InspectorState.h" |
| #include "core/inspector/InstrumentingAgents.h" |
| #include "core/inspector/JavaScriptCallFrame.h" |
| #include "core/inspector/ScriptAsyncCallStack.h" |
| #include "core/inspector/ScriptCallFrame.h" |
| #include "core/inspector/ScriptCallStack.h" |
| #include "core/inspector/V8AsyncCallTracker.h" |
| #include "core/inspector/V8Debugger.h" |
| #include "platform/JSONValues.h" |
| #include "wtf/text/StringBuilder.h" |
| #include "wtf/text/WTFString.h" |
| |
| using blink::TypeBuilder::Array; |
| using blink::TypeBuilder::Console::AsyncStackTrace; |
| using blink::TypeBuilder::Debugger::AsyncOperation; |
| using blink::TypeBuilder::Debugger::BreakpointId; |
| using blink::TypeBuilder::Debugger::CallFrame; |
| using blink::TypeBuilder::Debugger::CollectionEntry; |
| using blink::TypeBuilder::Debugger::ExceptionDetails; |
| using blink::TypeBuilder::Debugger::FunctionDetails; |
| using blink::TypeBuilder::Debugger::GeneratorObjectDetails; |
| using blink::TypeBuilder::Debugger::PromiseDetails; |
| using blink::TypeBuilder::Debugger::ScriptId; |
| using blink::TypeBuilder::Debugger::StackTrace; |
| using blink::TypeBuilder::Runtime::RemoteObject; |
| |
| namespace blink { |
| |
| namespace DebuggerAgentState { |
| static const char debuggerEnabled[] = "debuggerEnabled"; |
| static const char javaScriptBreakpoints[] = "javaScriptBreakopints"; |
| static const char pauseOnExceptionsState[] = "pauseOnExceptionsState"; |
| static const char asyncCallStackDepth[] = "asyncCallStackDepth"; |
| static const char promiseTrackerEnabled[] = "promiseTrackerEnabled"; |
| static const char promiseTrackerCaptureStacks[] = "promiseTrackerCaptureStacks"; |
| |
| // Breakpoint properties. |
| static const char url[] = "url"; |
| static const char isRegex[] = "isRegex"; |
| static const char lineNumber[] = "lineNumber"; |
| static const char columnNumber[] = "columnNumber"; |
| static const char condition[] = "condition"; |
| static const char skipStackPattern[] = "skipStackPattern"; |
| static const char skipContentScripts[] = "skipContentScripts"; |
| static const char skipAllPauses[] = "skipAllPauses"; |
| |
| }; |
| |
| static const int maxSkipStepFrameCount = 128; |
| |
| const char InspectorDebuggerAgent::backtraceObjectGroup[] = "backtrace"; |
| |
| const int InspectorDebuggerAgent::unknownAsyncOperationId = 0; |
| |
| static String breakpointIdSuffix(InspectorDebuggerAgent::BreakpointSource source) |
| { |
| switch (source) { |
| case InspectorDebuggerAgent::UserBreakpointSource: |
| break; |
| case InspectorDebuggerAgent::DebugCommandBreakpointSource: |
| return ":debug"; |
| case InspectorDebuggerAgent::MonitorCommandBreakpointSource: |
| return ":monitor"; |
| } |
| return String(); |
| } |
| |
| static String generateBreakpointId(const String& scriptId, int lineNumber, int columnNumber, InspectorDebuggerAgent::BreakpointSource source) |
| { |
| return scriptId + ':' + String::number(lineNumber) + ':' + String::number(columnNumber) + breakpointIdSuffix(source); |
| } |
| |
| static ScriptCallFrame toScriptCallFrame(JavaScriptCallFrame* callFrame) |
| { |
| String scriptId = String::number(callFrame->sourceID()); |
| // FIXME(WK62725): Debugger line/column are 0-based, while console ones are 1-based. |
| int line = callFrame->line() + 1; |
| int column = callFrame->column() + 1; |
| return ScriptCallFrame(callFrame->functionName(), scriptId, callFrame->scriptName(), line, column); |
| } |
| |
| static PassRefPtrWillBeRawPtr<ScriptCallStack> toScriptCallStack(JavaScriptCallFrame* callFrame) |
| { |
| Vector<ScriptCallFrame> frames; |
| for (; callFrame; callFrame = callFrame->caller()) |
| frames.append(toScriptCallFrame(callFrame)); |
| return ScriptCallStack::create(frames); |
| } |
| |
| static PassRefPtrWillBeRawPtr<ScriptCallStack> toScriptCallStack(const ScriptValue& callFrames) |
| { |
| RefPtr<JavaScriptCallFrame> jsCallFrame = V8Debugger::toJavaScriptCallFrameUnsafe(callFrames); |
| return jsCallFrame ? toScriptCallStack(jsCallFrame.get()) : nullptr; |
| } |
| |
| InspectorDebuggerAgent::InspectorDebuggerAgent(InjectedScriptManager* injectedScriptManager, v8::Isolate* isolate) |
| : InspectorBaseAgent<InspectorDebuggerAgent, InspectorFrontend::Debugger>("Debugger") |
| , m_injectedScriptManager(injectedScriptManager) |
| , m_pausedScriptState(nullptr) |
| , m_breakReason(InspectorFrontend::Debugger::Reason::Other) |
| , m_scheduledDebuggerStep(NoStep) |
| , m_skipNextDebuggerStepOut(false) |
| , m_javaScriptPauseScheduled(false) |
| , m_steppingFromFramework(false) |
| , m_pausingOnNativeEvent(false) |
| , m_pausingOnAsyncOperation(false) |
| , m_listener(nullptr) |
| , m_skippedStepFrameCount(0) |
| , m_recursionLevelForStepOut(0) |
| , m_recursionLevelForStepFrame(0) |
| , m_skipAllPauses(false) |
| , m_skipContentScripts(false) |
| , m_cachedSkipStackGeneration(0) |
| , m_lastAsyncOperationId(0) |
| , m_maxAsyncCallStackDepth(0) |
| , m_currentAsyncCallChain(nullptr) |
| , m_nestedAsyncCallCount(0) |
| , m_currentAsyncOperationId(unknownAsyncOperationId) |
| , m_pendingTraceAsyncOperationCompleted(false) |
| , m_startingStepIntoAsync(false) |
| , m_compiledScripts(isolate) |
| { |
| m_v8AsyncCallTracker = V8AsyncCallTracker::create(this); |
| } |
| |
| InspectorDebuggerAgent::~InspectorDebuggerAgent() |
| { |
| #if !ENABLE(OILPAN) |
| ASSERT(!m_instrumentingAgents->inspectorDebuggerAgent()); |
| #endif |
| } |
| |
| void InspectorDebuggerAgent::init() |
| { |
| if (dartAgent()) |
| dartAgent()->init(); |
| |
| m_promiseTracker = PromiseTracker::create(this, debugger().isolate()); |
| // FIXME: make breakReason optional so that there was no need to init it with "other". |
| clearBreakDetails(); |
| m_state->setLong(DebuggerAgentState::pauseOnExceptionsState, V8Debugger::DontPauseOnExceptions); |
| } |
| |
| bool InspectorDebuggerAgent::checkEnabled(ErrorString* errorString) |
| { |
| if (enabled()) |
| return true; |
| *errorString = "Debugger agent is not enabled"; |
| return false; |
| } |
| |
| void InspectorDebuggerAgent::enable() |
| { |
| if (dartAgent()) |
| dartAgent()->enable(0); |
| |
| m_instrumentingAgents->setInspectorDebuggerAgent(this); |
| // startListeningV8Debugger may result in reporting all parsed scripts to |
| // the agent so it should already be in enabled state by then. |
| m_state->setBoolean(DebuggerAgentState::debuggerEnabled, true); |
| startListeningV8Debugger(); |
| // FIXME(WK44513): breakpoints activated flag should be synchronized between all front-ends |
| debugger().setBreakpointsActivated(true); |
| if (m_listener) |
| m_listener->debuggerWasEnabled(); |
| } |
| |
| void InspectorDebuggerAgent::disable() |
| { |
| if (dartAgent()) |
| dartAgent()->disable(0); |
| |
| m_state->setObject(DebuggerAgentState::javaScriptBreakpoints, JSONObject::create()); |
| m_state->setLong(DebuggerAgentState::pauseOnExceptionsState, V8Debugger::DontPauseOnExceptions); |
| m_state->setString(DebuggerAgentState::skipStackPattern, ""); |
| m_state->setBoolean(DebuggerAgentState::skipContentScripts, false); |
| m_state->setLong(DebuggerAgentState::asyncCallStackDepth, 0); |
| m_state->setBoolean(DebuggerAgentState::promiseTrackerEnabled, false); |
| m_instrumentingAgents->setInspectorDebuggerAgent(0); |
| |
| stopListeningV8Debugger(); |
| clear(); |
| |
| if (m_listener) |
| m_listener->debuggerWasDisabled(); |
| |
| m_skipAllPauses = false; |
| } |
| |
| bool InspectorDebuggerAgent::enabled() |
| { |
| return m_state->getBoolean(DebuggerAgentState::debuggerEnabled); |
| } |
| |
| void InspectorDebuggerAgent::enable(ErrorString*) |
| { |
| // TODO(jacobr): why weren't we passing errorString in 39? |
| if (dartAgent()) |
| dartAgent()->enable(0); |
| |
| if (enabled()) |
| return; |
| |
| enable(); |
| |
| ASSERT(frontend()); |
| } |
| |
| void InspectorDebuggerAgent::disable(ErrorString*) |
| { |
| // TODO(jacobr): why weren't we passing errorString in 39? |
| if (dartAgent()) |
| dartAgent()->disable(0); |
| |
| if (!enabled()) |
| return; |
| |
| disable(); |
| m_state->setBoolean(DebuggerAgentState::debuggerEnabled, false); |
| } |
| |
| static PassOwnPtr<ScriptRegexp> compileSkipCallFramePattern(String patternText) |
| { |
| if (patternText.isEmpty()) |
| return nullptr; |
| OwnPtr<ScriptRegexp> result = adoptPtr(new ScriptRegexp(patternText, TextCaseSensitive)); |
| if (!result->isValid()) |
| result.clear(); |
| return result.release(); |
| } |
| |
| void InspectorDebuggerAgent::increaseCachedSkipStackGeneration() |
| { |
| ++m_cachedSkipStackGeneration; |
| if (!m_cachedSkipStackGeneration) |
| m_cachedSkipStackGeneration = 1; |
| } |
| |
| void InspectorDebuggerAgent::internalSetAsyncCallStackDepth(int depth) |
| { |
| if (depth <= 0) { |
| m_maxAsyncCallStackDepth = 0; |
| resetAsyncCallTracker(); |
| } else { |
| m_maxAsyncCallStackDepth = depth; |
| } |
| for (auto& listener: m_asyncCallTrackingListeners) |
| listener->asyncCallTrackingStateChanged(m_maxAsyncCallStackDepth); |
| } |
| |
| void InspectorDebuggerAgent::restore() |
| { |
| if (dartAgent()) |
| dartAgent()->restore(); |
| |
| if (enabled()) { |
| frontend()->globalObjectCleared(); |
| enable(); |
| long pauseState = m_state->getLong(DebuggerAgentState::pauseOnExceptionsState); |
| String error; |
| setPauseOnExceptionsImpl(&error, pauseState); |
| m_cachedSkipStackRegExp = compileSkipCallFramePattern(m_state->getString(DebuggerAgentState::skipStackPattern)); |
| increaseCachedSkipStackGeneration(); |
| m_skipContentScripts = m_state->getBoolean(DebuggerAgentState::skipContentScripts); |
| m_skipAllPauses = m_state->getBoolean(DebuggerAgentState::skipAllPauses); |
| internalSetAsyncCallStackDepth(m_state->getLong(DebuggerAgentState::asyncCallStackDepth)); |
| promiseTracker().setEnabled(m_state->getBoolean(DebuggerAgentState::promiseTrackerEnabled), m_state->getBoolean(DebuggerAgentState::promiseTrackerCaptureStacks)); |
| } |
| } |
| |
| void InspectorDebuggerAgent::setBreakpointsActive(ErrorString* errorString, bool active) |
| { |
| // TODO(jacobr): why weren't we passing errorString in 39? |
| if (dartAgent()) |
| dartAgent()->setBreakpointsActive(0, active); |
| |
| if (!checkEnabled(errorString)) |
| return; |
| debugger().setBreakpointsActivated(active); |
| } |
| |
| void InspectorDebuggerAgent::setSkipAllPauses(ErrorString*, bool skipped) |
| { |
| if (dartAgent()) |
| dartAgent()->setSkipAllPauses(0, skipped, 0 /* was untilReload */); |
| |
| m_skipAllPauses = skipped; |
| m_state->setBoolean(DebuggerAgentState::skipAllPauses, m_skipAllPauses); |
| } |
| |
| bool InspectorDebuggerAgent::isPaused() |
| { |
| if (dartAgent() && dartAgent()->isPaused()) |
| return true; |
| |
| return debugger().isPaused(); |
| } |
| |
| static PassRefPtr<JSONObject> buildObjectForBreakpointCookie(const String& url, int lineNumber, int columnNumber, const String& condition, bool isRegex) |
| { |
| RefPtr<JSONObject> breakpointObject = JSONObject::create(); |
| breakpointObject->setString(DebuggerAgentState::url, url); |
| breakpointObject->setNumber(DebuggerAgentState::lineNumber, lineNumber); |
| breakpointObject->setNumber(DebuggerAgentState::columnNumber, columnNumber); |
| breakpointObject->setString(DebuggerAgentState::condition, condition); |
| breakpointObject->setBoolean(DebuggerAgentState::isRegex, isRegex); |
| return breakpointObject.release(); |
| } |
| |
| static bool matches(const String& url, const String& pattern, bool isRegex) |
| { |
| if (isRegex) { |
| ScriptRegexp regex(pattern, TextCaseSensitive); |
| return regex.match(url) != -1; |
| } |
| return url == pattern; |
| } |
| |
| void InspectorDebuggerAgent::setBreakpointByUrl(ErrorString* errorString, int lineNumber, const String* const optionalURL, const String* const optionalURLRegex, const int* const optionalColumnNumber, const String* const optionalCondition, const String* inLanguage, BreakpointId* outBreakpointId, RefPtr<Array<TypeBuilder::Debugger::Location>>& locations) |
| { |
| if (dartAgent() && (dartAgent()->isDartURL(optionalURL, optionalURLRegex) || (inLanguage && *inLanguage == "dart"))) { |
| dartAgent()->setBreakpointByUrl(errorString, lineNumber, optionalURL, optionalURLRegex, optionalColumnNumber, optionalCondition, 0 /*isAntiBreakpoint*/, outBreakpointId, locations); |
| // TODO(jacobr): why are we asserting on errorstring? |
| ASSERT(errorString->isEmpty()); |
| return; |
| } |
| |
| locations = Array<TypeBuilder::Debugger::Location>::create(); |
| if (!optionalURL == !optionalURLRegex) { |
| *errorString = "Either url or urlRegex must be specified."; |
| return; |
| } |
| |
| String url = optionalURL ? *optionalURL : *optionalURLRegex; |
| int columnNumber = 0; |
| if (optionalColumnNumber) { |
| columnNumber = *optionalColumnNumber; |
| if (columnNumber < 0) { |
| *errorString = "Incorrect column number"; |
| return; |
| } |
| } |
| String condition = optionalCondition ? *optionalCondition : ""; |
| bool isRegex = optionalURLRegex; |
| |
| String breakpointId = (isRegex ? "/" + url + "/" : url) + ':' + String::number(lineNumber) + ':' + String::number(columnNumber); |
| RefPtr<JSONObject> breakpointsCookie = m_state->getObject(DebuggerAgentState::javaScriptBreakpoints); |
| if (breakpointsCookie->find(breakpointId) != breakpointsCookie->end()) { |
| *errorString = "Breakpoint at specified location already exists."; |
| return; |
| } |
| |
| breakpointsCookie->setObject(breakpointId, buildObjectForBreakpointCookie(url, lineNumber, columnNumber, condition, isRegex)); |
| m_state->setObject(DebuggerAgentState::javaScriptBreakpoints, breakpointsCookie); |
| |
| ScriptBreakpoint breakpoint(lineNumber, columnNumber, condition); |
| for (auto& script : m_scripts) { |
| if (!matches(script.value.sourceURL(), url, isRegex)) |
| continue; |
| RefPtr<TypeBuilder::Debugger::Location> location = resolveBreakpoint(breakpointId, script.key, breakpoint, UserBreakpointSource); |
| if (location) |
| locations->addItem(location); |
| } |
| |
| *outBreakpointId = breakpointId; |
| } |
| |
| static bool parseLocation(ErrorString* errorString, PassRefPtr<JSONObject> location, String* scriptId, int* lineNumber, int* columnNumber) |
| { |
| if (!location->getString("scriptId", scriptId) || !location->getNumber("lineNumber", lineNumber)) { |
| // FIXME: replace with input validation. |
| *errorString = "scriptId and lineNumber are required."; |
| return false; |
| } |
| *columnNumber = 0; |
| location->getNumber("columnNumber", columnNumber); |
| return true; |
| } |
| |
| void InspectorDebuggerAgent::setBreakpoint(ErrorString* errorString, const RefPtr<JSONObject>& location, const String* const optionalCondition, BreakpointId* outBreakpointId, RefPtr<TypeBuilder::Debugger::Location>& actualLocation) |
| { |
| // TODO(jacobr): why are we ignoring dartErrorString? Seems like we just |
| // need to update dartAgent()->setBreakpoint so it behaves better. |
| ErrorString dartErrorString; |
| if (dartAgent()) { |
| dartAgent()->setBreakpoint(&dartErrorString, location, optionalCondition, outBreakpointId, actualLocation); |
| if (dartErrorString.isEmpty()) { |
| // This was a dart breakpoint... Done. |
| // TODO(JACOBR): why in the world was there an assert zero here? |
| // ASSERT(0); |
| return; |
| } |
| } |
| |
| String scriptId; |
| int lineNumber; |
| int columnNumber; |
| |
| if (!parseLocation(errorString, location, &scriptId, &lineNumber, &columnNumber)) |
| return; |
| |
| String condition = optionalCondition ? *optionalCondition : emptyString(); |
| |
| String breakpointId = generateBreakpointId(scriptId, lineNumber, columnNumber, UserBreakpointSource); |
| if (m_breakpointIdToDebuggerBreakpointIds.find(breakpointId) != m_breakpointIdToDebuggerBreakpointIds.end()) { |
| *errorString = "Breakpoint at specified location already exists."; |
| return; |
| } |
| ScriptBreakpoint breakpoint(lineNumber, columnNumber, condition); |
| actualLocation = resolveBreakpoint(breakpointId, scriptId, breakpoint, UserBreakpointSource); |
| if (actualLocation) |
| *outBreakpointId = breakpointId; |
| else |
| *errorString = "Could not resolve breakpoint"; |
| } |
| |
| void InspectorDebuggerAgent::removeBreakpoint(ErrorString* errorString, const String& breakpointId) |
| { |
| // TODO(jacobr): why are we ignoring errorString for the Dart case? |
| if (dartAgent()) |
| dartAgent()->removeBreakpoint(0, breakpointId); |
| |
| if (!checkEnabled(errorString)) |
| return; |
| RefPtr<JSONObject> breakpointsCookie = m_state->getObject(DebuggerAgentState::javaScriptBreakpoints); |
| breakpointsCookie->remove(breakpointId); |
| m_state->setObject(DebuggerAgentState::javaScriptBreakpoints, breakpointsCookie); |
| removeBreakpoint(breakpointId); |
| } |
| |
| void InspectorDebuggerAgent::removeBreakpoint(const String& breakpointId) |
| { |
| ASSERT(enabled()); |
| BreakpointIdToDebuggerBreakpointIdsMap::iterator debuggerBreakpointIdsIterator = m_breakpointIdToDebuggerBreakpointIds.find(breakpointId); |
| if (debuggerBreakpointIdsIterator == m_breakpointIdToDebuggerBreakpointIds.end()) |
| return; |
| for (size_t i = 0; i < debuggerBreakpointIdsIterator->value.size(); ++i) { |
| const String& debuggerBreakpointId = debuggerBreakpointIdsIterator->value[i]; |
| debugger().removeBreakpoint(debuggerBreakpointId); |
| m_serverBreakpoints.remove(debuggerBreakpointId); |
| } |
| m_breakpointIdToDebuggerBreakpointIds.remove(debuggerBreakpointIdsIterator); |
| } |
| |
| void InspectorDebuggerAgent::continueToLocation(ErrorString* errorString, const RefPtr<JSONObject>& location, const bool* interstateLocationOpt) |
| { |
| if (dartAgent()) |
| dartAgent()->continueToLocation(errorString, location, interstateLocationOpt); |
| |
| if (!checkEnabled(errorString)) |
| return; |
| if (!m_continueToLocationBreakpointId.isEmpty()) { |
| debugger().removeBreakpoint(m_continueToLocationBreakpointId); |
| m_continueToLocationBreakpointId = ""; |
| } |
| |
| String scriptId; |
| int lineNumber; |
| int columnNumber; |
| |
| if (!parseLocation(errorString, location, &scriptId, &lineNumber, &columnNumber)) |
| return; |
| |
| ScriptBreakpoint breakpoint(lineNumber, columnNumber, ""); |
| m_continueToLocationBreakpointId = debugger().setBreakpoint(scriptId, breakpoint, &lineNumber, &columnNumber, asBool(interstateLocationOpt)); |
| resume(errorString); |
| } |
| |
| void InspectorDebuggerAgent::getStepInPositions(ErrorString* errorString, const String& callFrameId, RefPtr<Array<TypeBuilder::Debugger::Location> >& positions) |
| { |
| if (dartAgent() && DartInjectedScript::isDartObjectId(callFrameId)) { |
| *errorString = "Not supported for Dart"; |
| return; |
| } |
| |
| if (!isPaused() || m_currentCallStack.isEmpty()) { |
| *errorString = "Attempt to access callframe when debugger is not on pause"; |
| return; |
| } |
| InjectedScript injectedScript = m_injectedScriptManager->injectedScriptForObjectId(callFrameId); |
| if (injectedScript.isEmpty()) { |
| *errorString = "Inspected frame has gone"; |
| return; |
| } |
| |
| injectedScript.getStepInPositions(errorString, m_currentCallStack, callFrameId, positions); |
| } |
| |
| void InspectorDebuggerAgent::getBacktrace(ErrorString* errorString, RefPtr<Array<CallFrame> >& callFrames, RefPtr<StackTrace>& asyncStackTrace) |
| { |
| if (dartAgent() && dartAgent()->isPaused()) { |
| dartAgent()->getBacktrace(errorString, callFrames, asyncStackTrace); |
| return; |
| } |
| |
| if (!assertPaused(errorString)) |
| return; |
| m_currentCallStack = debugger().currentCallFrames(); |
| callFrames = currentCallFrames(); |
| asyncStackTrace = currentAsyncStackTrace(); |
| } |
| |
| bool InspectorDebuggerAgent::isCallStackEmptyOrBlackboxed() |
| { |
| ASSERT(enabled()); |
| for (int index = 0; ; ++index) { |
| RefPtr<JavaScriptCallFrame> frame = debugger().callFrameNoScopes(index); |
| if (!frame) |
| break; |
| if (!isCallFrameWithUnknownScriptOrBlackboxed(frame.release())) |
| return false; |
| } |
| return true; |
| } |
| |
| bool InspectorDebuggerAgent::isTopCallFrameBlackboxed() |
| { |
| ASSERT(enabled()); |
| return isCallFrameWithUnknownScriptOrBlackboxed(debugger().callFrameNoScopes(0)); |
| } |
| |
| bool InspectorDebuggerAgent::isCallFrameWithUnknownScriptOrBlackboxed(PassRefPtr<JavaScriptCallFrame> pFrame) |
| { |
| RefPtr<JavaScriptCallFrame> frame = pFrame; |
| if (!frame) |
| return true; |
| ScriptsMap::iterator it = m_scripts.find(String::number(frame->sourceID())); |
| if (it == m_scripts.end()) { |
| // Unknown scripts are blackboxed. |
| return true; |
| } |
| if (m_skipContentScripts && it->value.isContentScript()) |
| return true; |
| bool isBlackboxed = false; |
| String scriptURL = it->value.sourceURL(); |
| if (m_cachedSkipStackRegExp && !scriptURL.isEmpty()) { |
| if (!it->value.getBlackboxedState(m_cachedSkipStackGeneration, &isBlackboxed)) { |
| isBlackboxed = m_cachedSkipStackRegExp->match(scriptURL) != -1; |
| it->value.setBlackboxedState(m_cachedSkipStackGeneration, isBlackboxed); |
| } |
| } |
| return isBlackboxed; |
| } |
| |
| ScriptDebugListener::SkipPauseRequest InspectorDebuggerAgent::shouldSkipExceptionPause() |
| { |
| if (m_steppingFromFramework) |
| return ScriptDebugListener::NoSkip; |
| if (isTopCallFrameBlackboxed()) |
| return ScriptDebugListener::Continue; |
| return ScriptDebugListener::NoSkip; |
| } |
| |
| ScriptDebugListener::SkipPauseRequest InspectorDebuggerAgent::shouldSkipStepPause() |
| { |
| if (m_steppingFromFramework) |
| return ScriptDebugListener::NoSkip; |
| |
| if (m_skipNextDebuggerStepOut) { |
| m_skipNextDebuggerStepOut = false; |
| if (m_scheduledDebuggerStep == StepOut) |
| return ScriptDebugListener::StepOut; |
| } |
| |
| if (!isTopCallFrameBlackboxed()) |
| return ScriptDebugListener::NoSkip; |
| |
| if (m_skippedStepFrameCount >= maxSkipStepFrameCount) |
| return ScriptDebugListener::StepOut; |
| |
| if (!m_skippedStepFrameCount) |
| m_recursionLevelForStepFrame = 1; |
| |
| ++m_skippedStepFrameCount; |
| return ScriptDebugListener::StepFrame; |
| } |
| |
| PassRefPtr<TypeBuilder::Debugger::Location> InspectorDebuggerAgent::resolveBreakpoint(const String& breakpointId, const String& scriptId, const ScriptBreakpoint& breakpoint, BreakpointSource source) |
| { |
| ASSERT(enabled()); |
| if (breakpointId.isEmpty()) { |
| ASSERT_NOT_REACHED(); |
| return nullptr; |
| } |
| ScriptsMap::iterator scriptIterator = m_scripts.find(scriptId); |
| if (scriptIterator == m_scripts.end()) |
| return nullptr; |
| Script& script = scriptIterator->value; |
| if (breakpoint.lineNumber < script.startLine() || script.endLine() < breakpoint.lineNumber) |
| return nullptr; |
| |
| int actualLineNumber; |
| int actualColumnNumber; |
| String debuggerBreakpointId = debugger().setBreakpoint(scriptId, breakpoint, &actualLineNumber, &actualColumnNumber, false); |
| if (debuggerBreakpointId.isEmpty()) |
| return nullptr; |
| |
| m_serverBreakpoints.set(debuggerBreakpointId, std::make_pair(breakpointId, source)); |
| |
| BreakpointIdToDebuggerBreakpointIdsMap::iterator debuggerBreakpointIdsIterator = m_breakpointIdToDebuggerBreakpointIds.find(breakpointId); |
| if (debuggerBreakpointIdsIterator == m_breakpointIdToDebuggerBreakpointIds.end()) |
| m_breakpointIdToDebuggerBreakpointIds.set(breakpointId, Vector<String>()).storedValue->value.append(debuggerBreakpointId); |
| else |
| debuggerBreakpointIdsIterator->value.append(debuggerBreakpointId); |
| |
| RefPtr<TypeBuilder::Debugger::Location> location = TypeBuilder::Debugger::Location::create() |
| .setScriptId(scriptId) |
| .setLineNumber(actualLineNumber); |
| location->setColumnNumber(actualColumnNumber); |
| return location; |
| } |
| |
| void InspectorDebuggerAgent::searchInContent(ErrorString* error, const String& scriptId, const String& query, const bool* const optionalCaseSensitive, const bool* const optionalIsRegex, RefPtr<Array<TypeBuilder::Debugger::SearchMatch>>& results) |
| { |
| if (dartAgent() && dartAgent()->isDartScriptId(scriptId)) { |
| dartAgent()->searchInContent(error, scriptId, query, optionalCaseSensitive, optionalIsRegex, results); |
| return; |
| } |
| |
| ScriptsMap::iterator it = m_scripts.find(scriptId); |
| if (it != m_scripts.end()) |
| results = ContentSearchUtils::searchInTextByLines(it->value.source(), query, asBool(optionalCaseSensitive), asBool(optionalIsRegex)); |
| else |
| *error = "No script for id: " + scriptId; |
| } |
| |
| void InspectorDebuggerAgent::setScriptSource(ErrorString* error, RefPtr<TypeBuilder::Debugger::SetScriptSourceError>& errorData, const String& scriptId, const String& newContent, const bool* const preview, RefPtr<Array<CallFrame> >& newCallFrames, RefPtr<JSONObject>& result, RefPtr<StackTrace>& asyncStackTrace) |
| { |
| // Dart doesn't support this yet. |
| if (dartAgent() && dartAgent()->isDartScriptId(scriptId)) { |
| *error = "Not supported for Dart"; |
| return; |
| } |
| |
| if (!checkEnabled(error)) |
| return; |
| if (!debugger().setScriptSource(scriptId, newContent, asBool(preview), error, errorData, &m_currentCallStack, &result)) |
| return; |
| |
| newCallFrames = currentCallFrames(); |
| asyncStackTrace = currentAsyncStackTrace(); |
| |
| ScriptsMap::iterator it = m_scripts.find(scriptId); |
| if (it == m_scripts.end()) |
| return; |
| String url = it->value.url(); |
| if (url.isEmpty()) |
| return; |
| m_editedScripts.set(url, newContent); |
| } |
| |
| void InspectorDebuggerAgent::restartFrame(ErrorString* errorString, const String& callFrameId, RefPtr<Array<CallFrame> >& newCallFrames, RefPtr<JSONObject>& result, RefPtr<StackTrace>& asyncStackTrace) |
| { |
| if (DartInjectedScript::isDartObjectId(callFrameId) && dartAgent()) { |
| *errorString = "Not supported for Dart"; |
| return; |
| } |
| |
| if (!isPaused() || m_currentCallStack.isEmpty()) { |
| *errorString = "Attempt to access callframe when debugger is not on pause"; |
| return; |
| } |
| InjectedScript injectedScript = m_injectedScriptManager->injectedScriptForObjectId(callFrameId); |
| if (injectedScript.isEmpty()) { |
| *errorString = "Inspected frame has gone"; |
| return; |
| } |
| |
| injectedScript.restartFrame(errorString, m_currentCallStack, callFrameId, &result); |
| m_currentCallStack = debugger().currentCallFrames(); |
| newCallFrames = currentCallFrames(); |
| asyncStackTrace = currentAsyncStackTrace(); |
| } |
| |
| void InspectorDebuggerAgent::getScriptSource(ErrorString* error, const String& scriptId, String* scriptSource) |
| { |
| if (dartAgent() && dartAgent()->isDartScriptId(scriptId)) { |
| dartAgent()->getScriptSource(error, scriptId, scriptSource); |
| return; |
| } |
| |
| if (!checkEnabled(error)) |
| return; |
| ScriptsMap::iterator it = m_scripts.find(scriptId); |
| if (it == m_scripts.end()) { |
| *error = "No script for id: " + scriptId; |
| return; |
| } |
| |
| String url = it->value.url(); |
| if (!url.isEmpty() && getEditedScript(url, scriptSource)) |
| return; |
| *scriptSource = it->value.source(); |
| } |
| |
| void InspectorDebuggerAgent::getFunctionDetails(ErrorString* errorString, const String& functionId, RefPtr<FunctionDetails>& details) |
| { |
| if (dartAgent() && DartInjectedScript::isDartObjectId(functionId)) { |
| dartAgent()->getFunctionDetails(errorString, functionId, details); |
| return; |
| } |
| |
| if (!checkEnabled(errorString)) |
| return; |
| InjectedScript injectedScript = m_injectedScriptManager->injectedScriptForObjectId(functionId); |
| if (injectedScript.isEmpty()) { |
| *errorString = "Function object id is obsolete"; |
| return; |
| } |
| injectedScript.getFunctionDetails(errorString, functionId, &details); |
| } |
| |
| void InspectorDebuggerAgent::getGeneratorObjectDetails(ErrorString* errorString, const String& objectId, RefPtr<GeneratorObjectDetails>& details) |
| { |
| if (dartAgent() && DartInjectedScript::isDartObjectId(objectId)) { |
| // Shouldn't be hit as Dart doesn't support generators. |
| *errorString = "Not supported for Dart"; |
| } |
| |
| if (!checkEnabled(errorString)) |
| return; |
| InjectedScript injectedScript = m_injectedScriptManager->injectedScriptForObjectId(objectId); |
| if (injectedScript.isEmpty()) { |
| *errorString = "Inspected frame has gone"; |
| return; |
| } |
| injectedScript.getGeneratorObjectDetails(errorString, objectId, &details); |
| } |
| |
| void InspectorDebuggerAgent::getCollectionEntries(ErrorString* errorString, const String& objectId, RefPtr<TypeBuilder::Array<CollectionEntry> >& entries) |
| { |
| // This wasn't supported in Dart in 39 so we can likely release without it. |
| // However it would be awesome to support it in the future and call this for all Iterable |
| // Dart object. TODO(jacobr): implement this. |
| if (dartAgent() && DartInjectedScript::isDartObjectId(objectId)) { |
| // Shouldn't be hit as Dart doesn't have generators. |
| *errorString = "Not supported for Dart. Contact jacobr@google if you want it to be supported"; |
| } |
| |
| if (!checkEnabled(errorString)) |
| return; |
| InjectedScript injectedScript = m_injectedScriptManager->injectedScriptForObjectId(objectId); |
| if (injectedScript.isEmpty()) { |
| *errorString = "Inspected frame has gone"; |
| return; |
| } |
| injectedScript.getCollectionEntries(errorString, objectId, &entries); |
| } |
| |
| void InspectorDebuggerAgent::schedulePauseOnNextStatement(InspectorFrontend::Debugger::Reason::Enum breakReason, PassRefPtr<JSONObject> data) |
| { |
| ASSERT(enabled()); |
| if (m_scheduledDebuggerStep == StepInto || m_javaScriptPauseScheduled || isPaused()) |
| return; |
| m_breakReason = breakReason; |
| m_breakAuxData = data; |
| m_pausingOnNativeEvent = true; |
| m_skipNextDebuggerStepOut = false; |
| debugger().setPauseOnNextStatement(true); |
| } |
| |
| void InspectorDebuggerAgent::schedulePauseOnNextStatementIfSteppingInto() |
| { |
| ASSERT(enabled()); |
| if (m_scheduledDebuggerStep != StepInto || m_javaScriptPauseScheduled || isPaused()) |
| return; |
| clearBreakDetails(); |
| m_pausingOnNativeEvent = false; |
| m_skippedStepFrameCount = 0; |
| m_recursionLevelForStepFrame = 0; |
| debugger().setPauseOnNextStatement(true); |
| } |
| |
| void InspectorDebuggerAgent::didFireTimer() |
| { |
| cancelPauseOnNextStatement(); |
| } |
| |
| void InspectorDebuggerAgent::didHandleEvent() |
| { |
| cancelPauseOnNextStatement(); |
| } |
| |
| void InspectorDebuggerAgent::cancelPauseOnNextStatement() |
| { |
| if (m_javaScriptPauseScheduled || isPaused()) |
| return; |
| clearBreakDetails(); |
| m_pausingOnNativeEvent = false; |
| debugger().setPauseOnNextStatement(false); |
| } |
| |
| bool InspectorDebuggerAgent::v8AsyncTaskEventsEnabled() const |
| { |
| return trackingAsyncCalls(); |
| } |
| |
| void InspectorDebuggerAgent::didReceiveV8AsyncTaskEvent(ScriptState* state, const String& eventType, const String& eventName, int id) |
| { |
| ASSERT(trackingAsyncCalls()); |
| m_v8AsyncCallTracker->didReceiveV8AsyncTaskEvent(state, eventType, eventName, id); |
| } |
| |
| bool InspectorDebuggerAgent::v8PromiseEventsEnabled() const |
| { |
| return promiseTracker().isEnabled() || (m_listener && m_listener->canPauseOnPromiseEvent()); |
| } |
| |
| void InspectorDebuggerAgent::didReceiveV8PromiseEvent(ScriptState* scriptState, v8::Local<v8::Object> promise, v8::Local<v8::Value> parentPromise, int status) |
| { |
| if (promiseTracker().isEnabled()) |
| promiseTracker().didReceiveV8PromiseEvent(scriptState, promise, parentPromise, status); |
| if (!m_listener) |
| return; |
| if (!parentPromise.IsEmpty() && parentPromise->IsObject()) |
| return; |
| if (status < 0) |
| m_listener->didRejectPromise(); |
| else if (status > 0) |
| m_listener->didResolvePromise(); |
| else |
| m_listener->didCreatePromise(); |
| } |
| |
| void InspectorDebuggerAgent::pause(ErrorString* errorString) |
| { |
| if (dartAgent()) { |
| dartAgent()->pause(0); |
| // TODO(jacobr): pausing is a hard edge cause. Review whether we need |
| // to do any more work to make sure we don't crash by pausing both v8 |
| // and dart. One plausible solution would be to say that for dartium |
| // pausing only pauses dart as that is probably what our users want |
| // and would greatly reduce the risk |
| } |
| |
| if (!checkEnabled(errorString)) |
| return; |
| if (m_javaScriptPauseScheduled || isPaused()) |
| return; |
| clearBreakDetails(); |
| clearStepIntoAsync(); |
| m_javaScriptPauseScheduled = true; |
| m_scheduledDebuggerStep = NoStep; |
| m_skippedStepFrameCount = 0; |
| m_steppingFromFramework = false; |
| debugger().setPauseOnNextStatement(true); |
| } |
| |
| void InspectorDebuggerAgent::resume(ErrorString* errorString) |
| { |
| if (dartAgent() && dartAgent()->isPaused()) { |
| dartAgent()->resume(errorString); |
| return; |
| } |
| |
| if (!assertPaused(errorString)) |
| return; |
| m_scheduledDebuggerStep = NoStep; |
| m_steppingFromFramework = false; |
| m_injectedScriptManager->releaseObjectGroup(InspectorDebuggerAgent::backtraceObjectGroup); |
| debugger().continueProgram(); |
| } |
| |
| void InspectorDebuggerAgent::stepOver(ErrorString* errorString) |
| { |
| if (dartAgent() && dartAgent()->isPaused()) { |
| dartAgent()->stepOver(errorString); |
| return; |
| } |
| |
| if (!assertPaused(errorString)) |
| return; |
| // StepOver at function return point should fallback to StepInto. |
| RefPtr<JavaScriptCallFrame> frame = debugger().callFrameNoScopes(0); |
| if (frame && frame->isAtReturn()) { |
| stepInto(errorString); |
| return; |
| } |
| m_scheduledDebuggerStep = StepOver; |
| m_steppingFromFramework = isTopCallFrameBlackboxed(); |
| m_injectedScriptManager->releaseObjectGroup(InspectorDebuggerAgent::backtraceObjectGroup); |
| debugger().stepOverStatement(); |
| } |
| |
| void InspectorDebuggerAgent::stepInto(ErrorString* errorString) |
| { |
| if (dartAgent()->isPaused()) { |
| dartAgent()->stepInto(errorString); |
| return; |
| } |
| |
| if (!assertPaused(errorString)) |
| return; |
| m_scheduledDebuggerStep = StepInto; |
| m_steppingFromFramework = isTopCallFrameBlackboxed(); |
| m_injectedScriptManager->releaseObjectGroup(InspectorDebuggerAgent::backtraceObjectGroup); |
| debugger().stepIntoStatement(); |
| } |
| |
| void InspectorDebuggerAgent::stepOut(ErrorString* errorString) |
| { |
| if (dartAgent() && dartAgent()->isPaused()) { |
| dartAgent()->stepOut(errorString); |
| return; |
| } |
| |
| if (!assertPaused(errorString)) |
| return; |
| m_scheduledDebuggerStep = StepOut; |
| m_skipNextDebuggerStepOut = false; |
| m_recursionLevelForStepOut = 1; |
| m_steppingFromFramework = isTopCallFrameBlackboxed(); |
| m_injectedScriptManager->releaseObjectGroup(InspectorDebuggerAgent::backtraceObjectGroup); |
| debugger().stepOutOfFunction(); |
| } |
| |
| void InspectorDebuggerAgent::stepIntoAsync(ErrorString* errorString) |
| { |
| if (dartAgent() && dartAgent()->isPaused()) { |
| *errorString = "Not supported for Dart"; |
| return; |
| } |
| |
| if (!assertPaused(errorString)) |
| return; |
| if (!trackingAsyncCalls()) { |
| *errorString = "Can only perform operation if async call stacks are enabled."; |
| return; |
| } |
| clearStepIntoAsync(); |
| m_startingStepIntoAsync = true; |
| stepInto(errorString); |
| } |
| |
| void InspectorDebuggerAgent::setPauseOnExceptions(ErrorString* errorString, const String& stringPauseState) |
| { |
| if (dartAgent()) |
| dartAgent()->setPauseOnExceptions(errorString, stringPauseState); |
| |
| if (!checkEnabled(errorString)) |
| return; |
| V8Debugger::PauseOnExceptionsState pauseState; |
| if (stringPauseState == "none") |
| pauseState = V8Debugger::DontPauseOnExceptions; |
| else if (stringPauseState == "all") |
| pauseState = V8Debugger::PauseOnAllExceptions; |
| else if (stringPauseState == "uncaught") |
| pauseState = V8Debugger::PauseOnUncaughtExceptions; |
| else { |
| *errorString = "Unknown pause on exceptions mode: " + stringPauseState; |
| return; |
| } |
| setPauseOnExceptionsImpl(errorString, pauseState); |
| } |
| |
| void InspectorDebuggerAgent::setPauseOnExceptionsImpl(ErrorString* errorString, int pauseState) |
| { |
| debugger().setPauseOnExceptionsState(static_cast<V8Debugger::PauseOnExceptionsState>(pauseState)); |
| if (debugger().pauseOnExceptionsState() != pauseState) |
| *errorString = "Internal error. Could not change pause on exceptions state"; |
| else |
| m_state->setLong(DebuggerAgentState::pauseOnExceptionsState, pauseState); |
| } |
| |
| void InspectorDebuggerAgent::evaluateOnCallFrame(ErrorString* errorString, const String& callFrameId, const String& expression, const String* const objectGroup, const bool* const includeCommandLineAPI, const bool* const doNotPauseOnExceptionsAndMuteConsole, const bool* const returnByValue, const bool* generatePreview, RefPtr<RemoteObject>& result, TypeBuilder::OptOutput<bool>* wasThrown, RefPtr<TypeBuilder::Debugger::ExceptionDetails>& exceptionDetails) |
| { |
| if (dartAgent() && DartInjectedScript::isDartObjectId(callFrameId)) { |
| dartAgent()->evaluateOnCallFrame(errorString, callFrameId, expression, objectGroup, includeCommandLineAPI, doNotPauseOnExceptionsAndMuteConsole, returnByValue, generatePreview, result, wasThrown, exceptionDetails); |
| return; |
| } |
| |
| if (!isPaused() || m_currentCallStack.isEmpty()) { |
| *errorString = "Attempt to access callframe when debugger is not on pause"; |
| return; |
| } |
| InjectedScript injectedScript = m_injectedScriptManager->injectedScriptForObjectId(callFrameId); |
| if (injectedScript.isEmpty()) { |
| *errorString = "Inspected frame has gone"; |
| return; |
| } |
| |
| V8Debugger::PauseOnExceptionsState previousPauseOnExceptionsState = debugger().pauseOnExceptionsState(); |
| if (asBool(doNotPauseOnExceptionsAndMuteConsole)) { |
| if (previousPauseOnExceptionsState != V8Debugger::DontPauseOnExceptions) |
| debugger().setPauseOnExceptionsState(V8Debugger::DontPauseOnExceptions); |
| muteConsole(); |
| } |
| |
| Vector<ScriptValue> asyncCallStacks; |
| if (m_currentAsyncCallChain) { |
| const AsyncCallStackVector& callStacks = m_currentAsyncCallChain->callStacks(); |
| asyncCallStacks.resize(callStacks.size()); |
| AsyncCallStackVector::const_iterator it = callStacks.begin(); |
| for (size_t i = 0; it != callStacks.end(); ++it, ++i) |
| asyncCallStacks[i] = (*it)->callFrames(); |
| } |
| |
| injectedScript.evaluateOnCallFrame(errorString, m_currentCallStack, asyncCallStacks, callFrameId, expression, objectGroup ? *objectGroup : "", asBool(includeCommandLineAPI), asBool(returnByValue), asBool(generatePreview), &result, wasThrown, &exceptionDetails); |
| if (asBool(doNotPauseOnExceptionsAndMuteConsole)) { |
| unmuteConsole(); |
| if (debugger().pauseOnExceptionsState() != previousPauseOnExceptionsState) |
| debugger().setPauseOnExceptionsState(previousPauseOnExceptionsState); |
| } |
| } |
| |
| // CUSTOM METHODS FOR DART |
| void InspectorDebuggerAgent::getCompletionsOnCallFrame(ErrorString* errorString, const String& callFrameId, const String& expression, RefPtr<TypeBuilder::Array<String>>& result) |
| { |
| if (dartAgent() && DartInjectedScript::isDartObjectId(callFrameId)) { |
| dartAgent()->getCompletionsOnCallFrame(errorString, callFrameId, expression, result); |
| return; |
| } |
| *errorString = "This method is not supported for JavaScript"; |
| } |
| // END CUSTOM METHODS FOR DART |
| |
| void InspectorDebuggerAgent::compileScript(ErrorString* errorString, const String& expression, const String& sourceURL, bool persistScript, const int* executionContextId, TypeBuilder::OptOutput<ScriptId>* scriptId, RefPtr<ExceptionDetails>& exceptionDetails) |
| { |
| // Intentionally not supported in Dart. |
| if (!checkEnabled(errorString)) |
| return; |
| InjectedScript injectedScript = injectedScriptForEval(errorString, executionContextId); |
| if (injectedScript.isEmpty() || !injectedScript.scriptState()->contextIsValid()) { |
| *errorString = "Inspected frame has gone"; |
| return; |
| } |
| |
| v8::Isolate* isolate = debugger().isolate(); |
| ScriptState::Scope scope(injectedScript.scriptState()); |
| v8::Local<v8::String> source = v8String(isolate, expression); |
| v8::TryCatch tryCatch; |
| v8::Local<v8::Script> script; |
| if (!v8Call(V8ScriptRunner::compileScript(source, sourceURL, String(), TextPosition(), isolate), script, tryCatch)) { |
| v8::Local<v8::Message> message = tryCatch.Message(); |
| if (!message.IsEmpty()) |
| exceptionDetails = createExceptionDetails(isolate, message); |
| else |
| *errorString = "Script compilation failed"; |
| return; |
| } |
| |
| if (!persistScript) |
| return; |
| |
| String scriptValueId = String::number(script->GetUnboundScript()->GetId()); |
| m_compiledScripts.Set(scriptValueId, script); |
| *scriptId = scriptValueId; |
| } |
| |
| void InspectorDebuggerAgent::runScript(ErrorString* errorString, const ScriptId& scriptId, const int* executionContextId, const String* const objectGroup, const bool* const doNotPauseOnExceptionsAndMuteConsole, RefPtr<RemoteObject>& result, RefPtr<ExceptionDetails>& exceptionDetails) |
| { |
| if (!checkEnabled(errorString)) |
| return; |
| InjectedScript injectedScript = injectedScriptForEval(errorString, executionContextId); |
| if (injectedScript.isEmpty()) { |
| *errorString = "Inspected frame has gone"; |
| return; |
| } |
| |
| V8Debugger::PauseOnExceptionsState previousPauseOnExceptionsState = debugger().pauseOnExceptionsState(); |
| if (asBool(doNotPauseOnExceptionsAndMuteConsole)) { |
| if (previousPauseOnExceptionsState != V8Debugger::DontPauseOnExceptions) |
| debugger().setPauseOnExceptionsState(V8Debugger::DontPauseOnExceptions); |
| muteConsole(); |
| } |
| |
| if (!m_compiledScripts.Contains(scriptId)) { |
| *errorString = "Script execution failed"; |
| return; |
| } |
| |
| v8::Isolate* isolate = debugger().isolate(); |
| ScriptState* scriptState = injectedScript.scriptState(); |
| ScriptState::Scope scope(scriptState); |
| v8::Local<v8::Script> script = v8::Local<v8::Script>::New(isolate, m_compiledScripts.Remove(scriptId)); |
| |
| if (script.IsEmpty() || !scriptState->contextIsValid()) { |
| *errorString = "Script execution failed"; |
| return; |
| } |
| v8::TryCatch tryCatch; |
| v8::Local<v8::Value> value; |
| ScriptValue scriptValue; |
| if (v8Call(V8ScriptRunner::runCompiledScript(isolate, script, scriptState->executionContext()), value, tryCatch)) { |
| scriptValue = ScriptValue(scriptState, value); |
| } else { |
| scriptValue = ScriptValue(scriptState, tryCatch.Exception()); |
| v8::Local<v8::Message> message = tryCatch.Message(); |
| if (!message.IsEmpty()) |
| exceptionDetails = createExceptionDetails(isolate, message); |
| } |
| |
| if (scriptValue.isEmpty()) { |
| *errorString = "Script execution failed"; |
| return; |
| } |
| |
| result = injectedScript.wrapObject(scriptValue, objectGroup ? *objectGroup : ""); |
| |
| if (asBool(doNotPauseOnExceptionsAndMuteConsole)) { |
| unmuteConsole(); |
| if (debugger().pauseOnExceptionsState() != previousPauseOnExceptionsState) |
| debugger().setPauseOnExceptionsState(previousPauseOnExceptionsState); |
| } |
| } |
| |
| void InspectorDebuggerAgent::setVariableValue(ErrorString* errorString, int scopeNumber, const String& variableName, const RefPtr<JSONObject>& newValue, const String* callFrameId, const String* functionObjectId) |
| { |
| if (dartAgent() && callFrameId && DartInjectedScript::isDartObjectId(*callFrameId)) { |
| dartAgent()->setVariableValue(errorString, scopeNumber, variableName, newValue, callFrameId, functionObjectId); |
| return; |
| } |
| |
| if (!checkEnabled(errorString)) |
| return; |
| InjectedScript injectedScript; |
| if (callFrameId) { |
| if (!isPaused() || m_currentCallStack.isEmpty()) { |
| *errorString = "Attempt to access callframe when debugger is not on pause"; |
| return; |
| } |
| injectedScript = m_injectedScriptManager->injectedScriptForObjectId(*callFrameId); |
| if (injectedScript.isEmpty()) { |
| *errorString = "Inspected frame has gone"; |
| return; |
| } |
| } else if (functionObjectId) { |
| injectedScript = m_injectedScriptManager->injectedScriptForObjectId(*functionObjectId); |
| if (injectedScript.isEmpty()) { |
| *errorString = "Function object id cannot be resolved"; |
| return; |
| } |
| } else { |
| *errorString = "Either call frame or function object must be specified"; |
| return; |
| } |
| String newValueString = newValue->toJSONString(); |
| |
| injectedScript.setVariableValue(errorString, m_currentCallStack, callFrameId, functionObjectId, scopeNumber, variableName, newValueString); |
| } |
| |
| void InspectorDebuggerAgent::skipStackFrames(ErrorString* errorString, const String* pattern, const bool* skipContentScripts) |
| { |
| // Content script don't matter for Dart. |
| if (dartAgent()) |
| dartAgent()->skipStackFrames(errorString, pattern); |
| |
| if (!checkEnabled(errorString)) |
| return; |
| OwnPtr<ScriptRegexp> compiled; |
| String patternValue = pattern ? *pattern : ""; |
| if (!patternValue.isEmpty()) { |
| compiled = compileSkipCallFramePattern(patternValue); |
| if (!compiled) { |
| *errorString = "Invalid regular expression"; |
| return; |
| } |
| } |
| m_state->setString(DebuggerAgentState::skipStackPattern, patternValue); |
| m_cachedSkipStackRegExp = compiled.release(); |
| increaseCachedSkipStackGeneration(); |
| m_skipContentScripts = asBool(skipContentScripts); |
| m_state->setBoolean(DebuggerAgentState::skipContentScripts, m_skipContentScripts); |
| } |
| |
| void InspectorDebuggerAgent::setAsyncCallStackDepth(ErrorString* errorString, int depth) |
| { |
| // Not supported in Dart. |
| |
| if (!checkEnabled(errorString)) |
| return; |
| m_state->setLong(DebuggerAgentState::asyncCallStackDepth, depth); |
| internalSetAsyncCallStackDepth(depth); |
| } |
| |
| void InspectorDebuggerAgent::enablePromiseTracker(ErrorString* errorString, const bool* captureStacks) |
| { |
| // Not supported in Dart. |
| |
| if (!checkEnabled(errorString)) |
| return; |
| m_state->setBoolean(DebuggerAgentState::promiseTrackerEnabled, true); |
| m_state->setBoolean(DebuggerAgentState::promiseTrackerCaptureStacks, asBool(captureStacks)); |
| promiseTracker().setEnabled(true, asBool(captureStacks)); |
| } |
| |
| void InspectorDebuggerAgent::disablePromiseTracker(ErrorString* errorString) |
| { |
| // Not supported in Dart. |
| |
| if (!checkEnabled(errorString)) |
| return; |
| m_state->setBoolean(DebuggerAgentState::promiseTrackerEnabled, false); |
| promiseTracker().setEnabled(false, false); |
| } |
| |
| void InspectorDebuggerAgent::getPromiseById(ErrorString* errorString, int promiseId, const String* objectGroup, RefPtr<RemoteObject>& promise) |
| { |
| // Not supported in Dart. |
| |
| if (!checkEnabled(errorString)) |
| return; |
| if (!promiseTracker().isEnabled()) { |
| *errorString = "Promise tracking is disabled"; |
| return; |
| } |
| ScriptValue value = promiseTracker().promiseById(promiseId); |
| if (value.isEmpty()) { |
| *errorString = "Promise with specified ID not found."; |
| return; |
| } |
| InjectedScript injectedScript = m_injectedScriptManager->injectedScriptFor(value.scriptState()); |
| promise = injectedScript.wrapObject(value, objectGroup ? *objectGroup : ""); |
| } |
| |
| void InspectorDebuggerAgent::didUpdatePromise(InspectorFrontend::Debugger::EventType::Enum eventType, PassRefPtr<TypeBuilder::Debugger::PromiseDetails> promise) |
| { |
| if (frontend()) |
| frontend()->promiseUpdated(eventType, promise); |
| } |
| |
| int InspectorDebuggerAgent::traceAsyncOperationStarting(const String& description) |
| { |
| ScriptValue callFrames = debugger().currentCallFramesForAsyncStack(); |
| RefPtrWillBeRawPtr<AsyncCallChain> chain = nullptr; |
| if (callFrames.isEmpty()) { |
| if (m_currentAsyncCallChain) |
| chain = AsyncCallChain::create(nullptr, m_currentAsyncCallChain.get(), m_maxAsyncCallStackDepth); |
| } else { |
| chain = AsyncCallChain::create(adoptRefWillBeNoop(new AsyncCallStack(description, callFrames)), m_currentAsyncCallChain.get(), m_maxAsyncCallStackDepth); |
| } |
| do { |
| ++m_lastAsyncOperationId; |
| if (m_lastAsyncOperationId <= 0) |
| m_lastAsyncOperationId = 1; |
| } while (m_asyncOperations.contains(m_lastAsyncOperationId)); |
| m_asyncOperations.set(m_lastAsyncOperationId, chain); |
| if (chain) |
| m_asyncOperationNotifications.add(m_lastAsyncOperationId); |
| |
| if (m_startingStepIntoAsync) { |
| // We have successfully started a StepIntoAsync, so revoke the debugger's StepInto |
| // and wait for the corresponding async operation breakpoint. |
| ASSERT(m_pausingAsyncOperations.isEmpty()); |
| m_pausingAsyncOperations.add(m_lastAsyncOperationId); |
| m_startingStepIntoAsync = false; |
| m_scheduledDebuggerStep = NoStep; |
| debugger().clearStepping(); |
| } else if (m_pausingOnAsyncOperation) { |
| m_pausingAsyncOperations.add(m_lastAsyncOperationId); |
| } |
| |
| if (m_pausedScriptState) |
| flushAsyncOperationEvents(nullptr); |
| return m_lastAsyncOperationId; |
| } |
| |
| void InspectorDebuggerAgent::traceAsyncCallbackStarting(int operationId) |
| { |
| ASSERT(operationId > 0 || operationId == unknownAsyncOperationId); |
| AsyncCallChain* chain = operationId > 0 ? m_asyncOperations.get(operationId) : nullptr; |
| // FIXME: extract recursion check into a delegate. |
| v8::Isolate* isolate = debugger().isolate(); |
| int recursionLevel = V8RecursionScope::recursionLevel(isolate); |
| if (chain && (!recursionLevel || (recursionLevel == 1 && Microtask::performingCheckpoint(isolate)))) { |
| // There can be still an old m_currentAsyncCallChain set if we start running Microtasks |
| // right after executing a JS callback but before the corresponding traceAsyncCallbackCompleted(). |
| // In this case just call traceAsyncCallbackCompleted() now, and the subsequent ones will be ignored. |
| // |
| // The nested levels count may be greater than 1, for example, when events are guarded via custom |
| // traceAsync* calls, like in window.postMessage(). In this case there will be a willHandleEvent |
| // instrumentation with unknownAsyncOperationId bumping up the nested levels count. |
| if (m_currentAsyncCallChain) { |
| ASSERT(m_nestedAsyncCallCount >= 1); |
| ASSERT(recursionLevel == 1 && Microtask::performingCheckpoint(isolate)); |
| m_nestedAsyncCallCount = 1; |
| traceAsyncCallbackCompleted(); |
| } |
| |
| // Current AsyncCallChain corresponds to the bottommost JS call frame. |
| ASSERT(!m_currentAsyncCallChain); |
| m_currentAsyncCallChain = chain; |
| m_currentAsyncOperationId = operationId; |
| m_pendingTraceAsyncOperationCompleted = false; |
| m_nestedAsyncCallCount = 1; |
| |
| if (m_pausingAsyncOperations.contains(operationId) || m_asyncOperationBreakpoints.contains(operationId)) { |
| m_pausingOnAsyncOperation = true; |
| m_scheduledDebuggerStep = StepInto; |
| m_skippedStepFrameCount = 0; |
| m_recursionLevelForStepFrame = 0; |
| debugger().setPauseOnNextStatement(true); |
| } |
| } else { |
| if (m_currentAsyncCallChain) |
| ++m_nestedAsyncCallCount; |
| } |
| } |
| |
| void InspectorDebuggerAgent::traceAsyncCallbackCompleted() |
| { |
| if (!m_nestedAsyncCallCount) |
| return; |
| ASSERT(m_currentAsyncCallChain); |
| --m_nestedAsyncCallCount; |
| if (!m_nestedAsyncCallCount) { |
| clearCurrentAsyncOperation(); |
| if (!m_pausingOnAsyncOperation) |
| return; |
| m_pausingOnAsyncOperation = false; |
| m_scheduledDebuggerStep = NoStep; |
| debugger().setPauseOnNextStatement(false); |
| if (m_startingStepIntoAsync && m_pausingAsyncOperations.isEmpty()) |
| clearStepIntoAsync(); |
| } |
| } |
| |
| void InspectorDebuggerAgent::traceAsyncOperationCompleted(int operationId) |
| { |
| ASSERT(operationId > 0 || operationId == unknownAsyncOperationId); |
| bool shouldNotify = false; |
| if (operationId > 0) { |
| if (m_currentAsyncOperationId == operationId) { |
| if (m_pendingTraceAsyncOperationCompleted) { |
| m_pendingTraceAsyncOperationCompleted = false; |
| } else { |
| // Delay traceAsyncOperationCompleted() until the last async callback (being currently executed) is done. |
| m_pendingTraceAsyncOperationCompleted = true; |
| return; |
| } |
| } |
| m_asyncOperations.remove(operationId); |
| m_asyncOperationBreakpoints.remove(operationId); |
| m_pausingAsyncOperations.remove(operationId); |
| shouldNotify = !m_asyncOperationNotifications.take(operationId); |
| } |
| if (m_startingStepIntoAsync) { |
| if (!m_pausingOnAsyncOperation && m_pausingAsyncOperations.isEmpty()) |
| clearStepIntoAsync(); |
| } |
| if (frontend() && shouldNotify) |
| frontend()->asyncOperationCompleted(operationId); |
| } |
| |
| void InspectorDebuggerAgent::flushAsyncOperationEvents(ErrorString*) |
| { |
| // Not supported in Dart. |
| |
| if (!frontend()) |
| return; |
| |
| for (int operationId : m_asyncOperationNotifications) { |
| RefPtrWillBeRawPtr<AsyncCallChain> chain = m_asyncOperations.get(operationId); |
| ASSERT(chain); |
| const AsyncCallStackVector& callStacks = chain->callStacks(); |
| ASSERT(!callStacks.isEmpty()); |
| |
| RefPtr<AsyncOperation> operation; |
| RefPtr<AsyncStackTrace> lastAsyncStackTrace; |
| for (const auto& callStack : callStacks) { |
| RefPtrWillBeRawPtr<ScriptCallStack> scriptCallStack = toScriptCallStack(callStack->callFrames()); |
| if (!scriptCallStack) |
| break; |
| if (!operation) { |
| operation = AsyncOperation::create() |
| .setId(operationId) |
| .setDescription(callStack->description()) |
| .release(); |
| operation->setStackTrace(scriptCallStack->buildInspectorArray()); |
| continue; |
| } |
| RefPtr<AsyncStackTrace> asyncStackTrace = AsyncStackTrace::create() |
| .setCallFrames(scriptCallStack->buildInspectorArray()); |
| asyncStackTrace->setDescription(callStack->description()); |
| if (lastAsyncStackTrace) |
| lastAsyncStackTrace->setAsyncStackTrace(asyncStackTrace); |
| else |
| operation->setAsyncStackTrace(asyncStackTrace); |
| lastAsyncStackTrace = asyncStackTrace.release(); |
| } |
| |
| if (operation) |
| frontend()->asyncOperationStarted(operation.release()); |
| } |
| |
| m_asyncOperationNotifications.clear(); |
| } |
| |
| void InspectorDebuggerAgent::clearCurrentAsyncOperation() |
| { |
| if (m_pendingTraceAsyncOperationCompleted && m_currentAsyncOperationId != unknownAsyncOperationId) |
| traceAsyncOperationCompleted(m_currentAsyncOperationId); |
| |
| m_currentAsyncOperationId = unknownAsyncOperationId; |
| m_pendingTraceAsyncOperationCompleted = false; |
| m_nestedAsyncCallCount = 0; |
| m_currentAsyncCallChain.clear(); |
| } |
| |
| void InspectorDebuggerAgent::resetAsyncCallTracker() |
| { |
| clearCurrentAsyncOperation(); |
| clearStepIntoAsync(); |
| for (auto& listener: m_asyncCallTrackingListeners) |
| listener->resetAsyncOperations(); |
| m_asyncOperations.clear(); |
| m_asyncOperationNotifications.clear(); |
| m_asyncOperationBreakpoints.clear(); |
| } |
| |
| void InspectorDebuggerAgent::setAsyncOperationBreakpoint(ErrorString* errorString, int operationId) |
| { |
| // Not supported in Dart. |
| |
| if (!trackingAsyncCalls()) { |
| *errorString = "Can only perform operation while tracking async call stacks."; |
| return; |
| } |
| if (operationId <= 0) { |
| *errorString = "Wrong async operation id."; |
| return; |
| } |
| if (!m_asyncOperations.contains(operationId)) { |
| *errorString = "Unknown async operation id."; |
| return; |
| } |
| m_asyncOperationBreakpoints.add(operationId); |
| } |
| |
| void InspectorDebuggerAgent::removeAsyncOperationBreakpoint(ErrorString* errorString, int operationId) |
| { |
| // Not supported in Dart. |
| |
| if (!trackingAsyncCalls()) { |
| *errorString = "Can only perform operation while tracking async call stacks."; |
| return; |
| } |
| if (operationId <= 0) { |
| *errorString = "Wrong async operation id."; |
| return; |
| } |
| m_asyncOperationBreakpoints.remove(operationId); |
| } |
| |
| void InspectorDebuggerAgent::scriptExecutionBlockedByCSP(const String& directiveText) |
| { |
| if (debugger().pauseOnExceptionsState() != V8Debugger::DontPauseOnExceptions) { |
| RefPtr<JSONObject> directive = JSONObject::create(); |
| directive->setString("directiveText", directiveText); |
| breakProgram(InspectorFrontend::Debugger::Reason::CSPViolation, directive.release()); |
| } |
| } |
| |
| void InspectorDebuggerAgent::addAsyncCallTrackingListener(AsyncCallTrackingListener* listener) |
| { |
| m_asyncCallTrackingListeners.add(listener); |
| } |
| |
| void InspectorDebuggerAgent::removeAsyncCallTrackingListener(AsyncCallTrackingListener* listener) |
| { |
| ASSERT(m_asyncCallTrackingListeners.contains(listener)); |
| m_asyncCallTrackingListeners.remove(listener); |
| } |
| |
| void InspectorDebuggerAgent::willCallFunction(ExecutionContext*, const DevToolsFunctionInfo& info) |
| { |
| changeJavaScriptRecursionLevel(+1); |
| // Fast return. |
| if (m_scheduledDebuggerStep != StepInto) |
| return; |
| // Skip unknown scripts (e.g. InjectedScript). |
| if (!m_scripts.contains(String::number(info.scriptId()))) |
| return; |
| schedulePauseOnNextStatementIfSteppingInto(); |
| } |
| |
| void InspectorDebuggerAgent::didCallFunction() |
| { |
| changeJavaScriptRecursionLevel(-1); |
| } |
| |
| void InspectorDebuggerAgent::willEvaluateScript(LocalFrame*, const String&, int) |
| { |
| changeJavaScriptRecursionLevel(+1); |
| schedulePauseOnNextStatementIfSteppingInto(); |
| } |
| |
| void InspectorDebuggerAgent::didEvaluateScript() |
| { |
| changeJavaScriptRecursionLevel(-1); |
| } |
| |
| void InspectorDebuggerAgent::changeJavaScriptRecursionLevel(int step) |
| { |
| if (m_javaScriptPauseScheduled && !m_skipAllPauses && !isPaused()) { |
| // Do not ever loose user's pause request until we have actually paused. |
| debugger().setPauseOnNextStatement(true); |
| } |
| if (m_scheduledDebuggerStep == StepOut) { |
| m_recursionLevelForStepOut += step; |
| if (!m_recursionLevelForStepOut) { |
| // When StepOut crosses a task boundary (i.e. js -> blink_c++) from where it was requested, |
| // switch stepping to step into a next JS task, as if we exited to a blackboxed framework. |
| m_scheduledDebuggerStep = StepInto; |
| m_skipNextDebuggerStepOut = false; |
| } |
| } |
| if (m_recursionLevelForStepFrame) { |
| m_recursionLevelForStepFrame += step; |
| if (!m_recursionLevelForStepFrame) { |
| // We have walked through a blackboxed framework and got back to where we started. |
| // If there was no stepping scheduled, we should cancel the stepping explicitly, |
| // since there may be a scheduled StepFrame left. |
| // Otherwise, if we were stepping in/over, the StepFrame will stop at the right location, |
| // whereas if we were stepping out, we should continue doing so after debugger pauses |
| // from the old StepFrame. |
| m_skippedStepFrameCount = 0; |
| if (m_scheduledDebuggerStep == NoStep) |
| debugger().clearStepping(); |
| else if (m_scheduledDebuggerStep == StepOut) |
| m_skipNextDebuggerStepOut = true; |
| } |
| } |
| } |
| |
| bool InspectorDebuggerAgent::getEditedScript(const String& url, String* content) |
| { |
| if (!m_editedScripts.contains(url)) |
| return false; |
| *content = m_editedScripts.get(url); |
| return true; |
| } |
| |
| PassRefPtr<Array<CallFrame> > InspectorDebuggerAgent::currentCallFrames() |
| { |
| if (!m_pausedScriptState || m_currentCallStack.isEmpty()) |
| return Array<CallFrame>::create(); |
| InjectedScript injectedScript = m_injectedScriptManager->injectedScriptFor(m_pausedScriptState.get()); |
| if (injectedScript.isEmpty()) { |
| ASSERT_NOT_REACHED(); |
| return Array<CallFrame>::create(); |
| } |
| return injectedScript.wrapCallFrames(m_currentCallStack, 0); |
| } |
| |
| PassRefPtr<StackTrace> InspectorDebuggerAgent::currentAsyncStackTrace() |
| { |
| if (!m_pausedScriptState || !trackingAsyncCalls()) |
| return nullptr; |
| const AsyncCallChain* chain = m_currentAsyncCallChain.get(); |
| if (!chain) |
| return nullptr; |
| const AsyncCallStackVector& callStacks = chain->callStacks(); |
| if (callStacks.isEmpty()) |
| return nullptr; |
| RefPtr<StackTrace> result; |
| int asyncOrdinal = callStacks.size(); |
| for (AsyncCallStackVector::const_reverse_iterator it = callStacks.rbegin(); it != callStacks.rend(); ++it, --asyncOrdinal) { |
| ScriptValue callFrames = (*it)->callFrames(); |
| ScriptState* scriptState = callFrames.scriptState(); |
| InjectedScript injectedScript = scriptState ? m_injectedScriptManager->injectedScriptFor(scriptState) : InjectedScript(); |
| if (injectedScript.isEmpty()) { |
| result.clear(); |
| continue; |
| } |
| RefPtr<StackTrace> next = StackTrace::create() |
| .setCallFrames(injectedScript.wrapCallFrames(callFrames, asyncOrdinal)) |
| .release(); |
| next->setDescription((*it)->description()); |
| if (result) |
| next->setAsyncStackTrace(result.release()); |
| result.swap(next); |
| } |
| return result.release(); |
| } |
| |
| PassRefPtrWillBeRawPtr<ScriptAsyncCallStack> InspectorDebuggerAgent::currentAsyncStackTraceForConsole() |
| { |
| if (!trackingAsyncCalls()) |
| return nullptr; |
| const AsyncCallChain* chain = m_currentAsyncCallChain.get(); |
| if (!chain) |
| return nullptr; |
| const AsyncCallStackVector& callStacks = chain->callStacks(); |
| if (callStacks.isEmpty()) |
| return nullptr; |
| RefPtrWillBeRawPtr<ScriptAsyncCallStack> result = nullptr; |
| for (AsyncCallStackVector::const_reverse_iterator it = callStacks.rbegin(); it != callStacks.rend(); ++it) { |
| RefPtr<JavaScriptCallFrame> callFrame = V8Debugger::toJavaScriptCallFrameUnsafe((*it)->callFrames()); |
| if (!callFrame) |
| break; |
| result = ScriptAsyncCallStack::create((*it)->description(), toScriptCallStack(callFrame.get()), result.release()); |
| } |
| return result.release(); |
| } |
| |
| String InspectorDebuggerAgent::sourceMapURLForScript(const Script& script, CompileResult compileResult) |
| { |
| bool hasSyntaxError = compileResult != CompileSuccess; |
| if (!hasSyntaxError) |
| return script.sourceMappingURL(); |
| return ContentSearchUtils::findSourceMapURL(script.source(), ContentSearchUtils::JavaScriptMagicComment); |
| } |
| |
| // ScriptDebugListener functions |
| |
| void InspectorDebuggerAgent::didParseSource(const ParsedScript& parsedScript) |
| { |
| Script script = parsedScript.script; |
| |
| bool hasSyntaxError = parsedScript.compileResult != CompileSuccess; |
| if (hasSyntaxError) |
| script.setSourceURL(ContentSearchUtils::findSourceURL(script.source(), ContentSearchUtils::JavaScriptMagicComment)); |
| |
| bool isContentScript = script.isContentScript(); |
| bool isInternalScript = script.isInternalScript(); |
| bool hasSourceURL = script.hasSourceURL(); |
| String scriptURL = script.sourceURL(); |
| String sourceMapURL = sourceMapURLForScript(script, parsedScript.compileResult); |
| |
| const String* sourceMapURLParam = sourceMapURL.isNull() ? nullptr : &sourceMapURL; |
| const bool* isContentScriptParam = isContentScript ? &isContentScript : nullptr; |
| const bool* isInternalScriptParam = isInternalScript ? &isInternalScript : nullptr; |
| const bool* hasSourceURLParam = hasSourceURL ? &hasSourceURL : nullptr; |
| const String jsLanguageParam = String("JavaScript"); |
| const int* libraryId = 0; |
| |
| // FIXMEDART: Had to add the optional values for language and libraryId. |
| if (!hasSyntaxError) |
| frontend()->scriptParsed(parsedScript.scriptId, scriptURL, script.startLine(), script.startColumn(), script.endLine(), script.endColumn(), isContentScriptParam, isInternalScriptParam, sourceMapURLParam, hasSourceURLParam, &jsLanguageParam, libraryId); |
| else |
| frontend()->scriptFailedToParse(parsedScript.scriptId, scriptURL, script.startLine(), script.startColumn(), script.endLine(), script.endColumn(), isContentScriptParam, isInternalScriptParam, sourceMapURLParam, hasSourceURLParam, &jsLanguageParam, libraryId); |
| |
| m_scripts.set(parsedScript.scriptId, script); |
| |
| if (scriptURL.isEmpty() || hasSyntaxError) |
| return; |
| |
| RefPtr<JSONObject> breakpointsCookie = m_state->getObject(DebuggerAgentState::javaScriptBreakpoints); |
| for (auto& cookie : *breakpointsCookie) { |
| RefPtr<JSONObject> breakpointObject = cookie.value->asObject(); |
| bool isRegex; |
| breakpointObject->getBoolean(DebuggerAgentState::isRegex, &isRegex); |
| String url; |
| breakpointObject->getString(DebuggerAgentState::url, &url); |
| if (!matches(scriptURL, url, isRegex)) |
| continue; |
| ScriptBreakpoint breakpoint; |
| breakpointObject->getNumber(DebuggerAgentState::lineNumber, &breakpoint.lineNumber); |
| breakpointObject->getNumber(DebuggerAgentState::columnNumber, &breakpoint.columnNumber); |
| breakpointObject->getString(DebuggerAgentState::condition, &breakpoint.condition); |
| RefPtr<TypeBuilder::Debugger::Location> location = resolveBreakpoint(cookie.key, parsedScript.scriptId, breakpoint, UserBreakpointSource); |
| if (location) |
| frontend()->breakpointResolved(cookie.key, location); |
| } |
| } |
| |
| ScriptDebugListener::SkipPauseRequest InspectorDebuggerAgent::didPause(ScriptState* scriptState, const ScriptValue& callFrames, const ScriptValue& exception, const Vector<String>& hitBreakpoints, bool isPromiseRejection) |
| { |
| ScriptDebugListener::SkipPauseRequest result; |
| if (m_skipAllPauses) |
| result = ScriptDebugListener::Continue; |
| else if (!hitBreakpoints.isEmpty()) |
| result = ScriptDebugListener::NoSkip; // Don't skip explicit breakpoints even if set in frameworks. |
| else if (!exception.isEmpty()) |
| result = shouldSkipExceptionPause(); |
| else if (m_scheduledDebuggerStep != NoStep || m_javaScriptPauseScheduled || m_pausingOnNativeEvent) |
| result = shouldSkipStepPause(); |
| else |
| result = ScriptDebugListener::NoSkip; |
| |
| m_skipNextDebuggerStepOut = false; |
| if (result != ScriptDebugListener::NoSkip) |
| return result; |
| |
| // Skip pauses inside V8 internal scripts and on syntax errors. |
| if (callFrames.isEmpty()) |
| return ScriptDebugListener::Continue; |
| |
| ASSERT(scriptState); |
| ASSERT(!m_pausedScriptState); |
| m_pausedScriptState = scriptState; |
| m_currentCallStack = callFrames; |
| |
| if (!exception.isEmpty()) { |
| InjectedScript injectedScript = m_injectedScriptManager->injectedScriptFor(scriptState); |
| if (!injectedScript.isEmpty()) { |
| m_breakReason = isPromiseRejection ? InspectorFrontend::Debugger::Reason::PromiseRejection : InspectorFrontend::Debugger::Reason::Exception; |
| m_breakAuxData = injectedScript.wrapObject(exception, InspectorDebuggerAgent::backtraceObjectGroup)->openAccessors(); |
| // m_breakAuxData might be null after this. |
| } |
| } else if (m_pausingOnAsyncOperation) { |
| m_breakReason = InspectorFrontend::Debugger::Reason::AsyncOperation; |
| m_breakAuxData = JSONObject::create(); |
| m_breakAuxData->setNumber("operationId", m_currentAsyncOperationId); |
| } |
| |
| RefPtr<Array<String> > hitBreakpointIds = Array<String>::create(); |
| |
| for (const auto& point : hitBreakpoints) { |
| DebugServerBreakpointToBreakpointIdAndSourceMap::iterator breakpointIterator = m_serverBreakpoints.find(point); |
| if (breakpointIterator != m_serverBreakpoints.end()) { |
| const String& localId = breakpointIterator->value.first; |
| hitBreakpointIds->addItem(localId); |
| |
| BreakpointSource source = breakpointIterator->value.second; |
| if (m_breakReason == InspectorFrontend::Debugger::Reason::Other && source == DebugCommandBreakpointSource) |
| m_breakReason = InspectorFrontend::Debugger::Reason::DebugCommand; |
| } |
| } |
| |
| if (!m_asyncOperationNotifications.isEmpty()) |
| flushAsyncOperationEvents(nullptr); |
| |
| frontend()->paused(currentCallFrames(), m_breakReason, m_breakAuxData, hitBreakpointIds, currentAsyncStackTrace()); |
| m_scheduledDebuggerStep = NoStep; |
| m_javaScriptPauseScheduled = false; |
| m_steppingFromFramework = false; |
| m_pausingOnNativeEvent = false; |
| m_skippedStepFrameCount = 0; |
| m_recursionLevelForStepFrame = 0; |
| clearStepIntoAsync(); |
| |
| if (!m_continueToLocationBreakpointId.isEmpty()) { |
| debugger().removeBreakpoint(m_continueToLocationBreakpointId); |
| m_continueToLocationBreakpointId = ""; |
| } |
| return result; |
| } |
| |
| void InspectorDebuggerAgent::didContinue() |
| { |
| m_pausedScriptState = nullptr; |
| m_currentCallStack = ScriptValue(); |
| clearBreakDetails(); |
| frontend()->resumed(); |
| } |
| |
| bool InspectorDebuggerAgent::canBreakProgram() |
| { |
| return debugger().canBreakProgram(); |
| } |
| |
| void InspectorDebuggerAgent::breakProgram(InspectorFrontend::Debugger::Reason::Enum breakReason, PassRefPtr<JSONObject> data) |
| { |
| ASSERT(enabled()); |
| if (m_skipAllPauses || m_pausedScriptState || isCallStackEmptyOrBlackboxed()) |
| return; |
| m_breakReason = breakReason; |
| m_breakAuxData = data; |
| m_scheduledDebuggerStep = NoStep; |
| m_steppingFromFramework = false; |
| m_pausingOnNativeEvent = false; |
| clearStepIntoAsync(); |
| debugger().breakProgram(); |
| } |
| |
| void InspectorDebuggerAgent::clear() |
| { |
| m_pausedScriptState = nullptr; |
| m_currentCallStack = ScriptValue(); |
| m_scripts.clear(); |
| m_breakpointIdToDebuggerBreakpointIds.clear(); |
| internalSetAsyncCallStackDepth(0); |
| promiseTracker().clear(); |
| m_continueToLocationBreakpointId = String(); |
| clearBreakDetails(); |
| m_scheduledDebuggerStep = NoStep; |
| m_skipNextDebuggerStepOut = false; |
| m_javaScriptPauseScheduled = false; |
| m_steppingFromFramework = false; |
| m_pausingOnNativeEvent = false; |
| m_skippedStepFrameCount = 0; |
| m_recursionLevelForStepFrame = 0; |
| m_asyncOperationNotifications.clear(); |
| m_compiledScripts.Clear(); |
| clearStepIntoAsync(); |
| } |
| |
| void InspectorDebuggerAgent::clearStepIntoAsync() |
| { |
| m_startingStepIntoAsync = false; |
| m_pausingOnAsyncOperation = false; |
| m_pausingAsyncOperations.clear(); |
| } |
| |
| bool InspectorDebuggerAgent::assertPaused(ErrorString* errorString) |
| { |
| if (!m_pausedScriptState) { |
| *errorString = "Can only perform operation while paused."; |
| return false; |
| } |
| return true; |
| } |
| |
| void InspectorDebuggerAgent::clearBreakDetails() |
| { |
| m_breakReason = InspectorFrontend::Debugger::Reason::Other; |
| m_breakAuxData = nullptr; |
| } |
| |
| void InspectorDebuggerAgent::setBreakpoint(const String& scriptId, int lineNumber, int columnNumber, BreakpointSource source, const String& condition) |
| { |
| String breakpointId = generateBreakpointId(scriptId, lineNumber, columnNumber, source); |
| ScriptBreakpoint breakpoint(lineNumber, columnNumber, condition); |
| resolveBreakpoint(breakpointId, scriptId, breakpoint, source); |
| } |
| |
| void InspectorDebuggerAgent::removeBreakpoint(const String& scriptId, int lineNumber, int columnNumber, BreakpointSource source) |
| { |
| removeBreakpoint(generateBreakpointId(scriptId, lineNumber, columnNumber, source)); |
| } |
| |
| void InspectorDebuggerAgent::reset() |
| { |
| m_scheduledDebuggerStep = NoStep; |
| m_scripts.clear(); |
| m_breakpointIdToDebuggerBreakpointIds.clear(); |
| resetAsyncCallTracker(); |
| promiseTracker().clear(); |
| if (frontend()) |
| frontend()->globalObjectCleared(); |
| } |
| |
| void InspectorDebuggerAgent::resetModifiedSources() |
| { |
| m_editedScripts.clear(); |
| } |
| |
| DEFINE_TRACE(InspectorDebuggerAgent) |
| { |
| #if ENABLE(OILPAN) |
| visitor->trace(m_injectedScriptManager); |
| visitor->trace(m_listener); |
| visitor->trace(m_v8AsyncCallTracker); |
| visitor->trace(m_promiseTracker); |
| visitor->trace(m_asyncOperations); |
| visitor->trace(m_currentAsyncCallChain); |
| visitor->trace(m_asyncCallTrackingListeners); |
| #endif |
| InspectorBaseAgent::trace(visitor); |
| } |
| |
| PassRefPtr<TypeBuilder::Debugger::ExceptionDetails> InspectorDebuggerAgent::createExceptionDetails(v8::Isolate* isolate, v8::Local<v8::Message> message) |
| { |
| RefPtr<ExceptionDetails> exceptionDetails = ExceptionDetails::create().setText(toCoreStringWithUndefinedOrNullCheck(message->Get())); |
| exceptionDetails->setLine(message->GetLineNumber()); |
| exceptionDetails->setColumn(message->GetStartColumn()); |
| v8::Local<v8::StackTrace> messageStackTrace = message->GetStackTrace(); |
| if (!messageStackTrace.IsEmpty() && messageStackTrace->GetFrameCount() > 0) |
| exceptionDetails->setStackTrace(createScriptCallStack(isolate, messageStackTrace, messageStackTrace->GetFrameCount())->buildInspectorArray()); |
| return exceptionDetails.release(); |
| } |
| |
| } // namespace blink |