blob: 4c8307697e6e122075230ca3ce30a7eed1f67033 [file] [log] [blame]
/*
* 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