3655dc52b2
- introduced session id, which fixes an issue of reconnect while evaluating; - forEachContext provides a safe way to iterate over contexts in a group. BUG=chromium:590878 Review-Url: https://codereview.chromium.org/2905543004 Cr-Commit-Position: refs/heads/master@{#45613}
350 lines
14 KiB
C++
350 lines
14 KiB
C++
/*
|
|
* Copyright (c) 2010-2011 Google Inc. All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions are
|
|
* met:
|
|
*
|
|
* * Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
* * Redistributions in binary form must reproduce the above
|
|
* copyright notice, this list of conditions and the following disclaimer
|
|
* in the documentation and/or other materials provided with the
|
|
* distribution.
|
|
* * Neither the name of Google Inc. nor the names of its
|
|
* contributors may be used to endorse or promote products derived from
|
|
* this software without specific prior written permission.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
|
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
|
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
|
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
|
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
|
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
*/
|
|
|
|
#include "src/inspector/v8-inspector-impl.h"
|
|
|
|
#include <vector>
|
|
|
|
#include "src/inspector/inspected-context.h"
|
|
#include "src/inspector/string-util.h"
|
|
#include "src/inspector/v8-console-agent-impl.h"
|
|
#include "src/inspector/v8-console-message.h"
|
|
#include "src/inspector/v8-console.h"
|
|
#include "src/inspector/v8-debugger-agent-impl.h"
|
|
#include "src/inspector/v8-debugger.h"
|
|
#include "src/inspector/v8-inspector-session-impl.h"
|
|
#include "src/inspector/v8-profiler-agent-impl.h"
|
|
#include "src/inspector/v8-runtime-agent-impl.h"
|
|
#include "src/inspector/v8-stack-trace-impl.h"
|
|
|
|
namespace v8_inspector {
|
|
|
|
std::unique_ptr<V8Inspector> V8Inspector::create(v8::Isolate* isolate,
|
|
V8InspectorClient* client) {
|
|
return std::unique_ptr<V8Inspector>(new V8InspectorImpl(isolate, client));
|
|
}
|
|
|
|
V8InspectorImpl::V8InspectorImpl(v8::Isolate* isolate,
|
|
V8InspectorClient* client)
|
|
: m_isolate(isolate),
|
|
m_client(client),
|
|
m_debugger(new V8Debugger(isolate, this)),
|
|
m_capturingStackTracesCount(0),
|
|
m_lastExceptionId(0),
|
|
m_lastContextId(0) {
|
|
v8::debug::SetConsoleDelegate(m_isolate, console());
|
|
}
|
|
|
|
V8InspectorImpl::~V8InspectorImpl() {
|
|
v8::debug::SetConsoleDelegate(m_isolate, nullptr);
|
|
}
|
|
|
|
int V8InspectorImpl::contextGroupId(v8::Local<v8::Context> context) {
|
|
return contextGroupId(InspectedContext::contextId(context));
|
|
}
|
|
|
|
int V8InspectorImpl::contextGroupId(int contextId) {
|
|
protocol::HashMap<int, int>::iterator it =
|
|
m_contextIdToGroupIdMap.find(contextId);
|
|
return it != m_contextIdToGroupIdMap.end() ? it->second : 0;
|
|
}
|
|
|
|
v8::MaybeLocal<v8::Value> V8InspectorImpl::compileAndRunInternalScript(
|
|
v8::Local<v8::Context> context, v8::Local<v8::String> source) {
|
|
v8::Local<v8::UnboundScript> unboundScript;
|
|
if (!v8::debug::CompileInspectorScript(m_isolate, source)
|
|
.ToLocal(&unboundScript))
|
|
return v8::MaybeLocal<v8::Value>();
|
|
v8::MicrotasksScope microtasksScope(m_isolate,
|
|
v8::MicrotasksScope::kDoNotRunMicrotasks);
|
|
v8::Context::Scope contextScope(context);
|
|
return unboundScript->BindToCurrentContext()->Run(context);
|
|
}
|
|
|
|
v8::MaybeLocal<v8::Script> V8InspectorImpl::compileScript(
|
|
v8::Local<v8::Context> context, const String16& code,
|
|
const String16& fileName) {
|
|
v8::ScriptOrigin origin(
|
|
toV8String(m_isolate, fileName), v8::Integer::New(m_isolate, 0),
|
|
v8::Integer::New(m_isolate, 0),
|
|
v8::False(m_isolate), // sharable
|
|
v8::Local<v8::Integer>(), toV8String(m_isolate, String16()), // sourceMap
|
|
v8::True(m_isolate)); // opaqueresource
|
|
v8::ScriptCompiler::Source source(toV8String(m_isolate, code), origin);
|
|
return v8::ScriptCompiler::Compile(context, &source,
|
|
v8::ScriptCompiler::kNoCompileOptions);
|
|
}
|
|
|
|
void V8InspectorImpl::enableStackCapturingIfNeeded() {
|
|
if (!m_capturingStackTracesCount)
|
|
V8StackTraceImpl::setCaptureStackTraceForUncaughtExceptions(m_isolate,
|
|
true);
|
|
++m_capturingStackTracesCount;
|
|
}
|
|
|
|
void V8InspectorImpl::disableStackCapturingIfNeeded() {
|
|
if (!(--m_capturingStackTracesCount))
|
|
V8StackTraceImpl::setCaptureStackTraceForUncaughtExceptions(m_isolate,
|
|
false);
|
|
}
|
|
|
|
void V8InspectorImpl::muteExceptions(int contextGroupId) {
|
|
m_muteExceptionsMap[contextGroupId]++;
|
|
}
|
|
|
|
void V8InspectorImpl::unmuteExceptions(int contextGroupId) {
|
|
m_muteExceptionsMap[contextGroupId]--;
|
|
}
|
|
|
|
V8ConsoleMessageStorage* V8InspectorImpl::ensureConsoleMessageStorage(
|
|
int contextGroupId) {
|
|
ConsoleStorageMap::iterator storageIt =
|
|
m_consoleStorageMap.find(contextGroupId);
|
|
if (storageIt == m_consoleStorageMap.end())
|
|
storageIt = m_consoleStorageMap
|
|
.insert(std::make_pair(
|
|
contextGroupId,
|
|
std::unique_ptr<V8ConsoleMessageStorage>(
|
|
new V8ConsoleMessageStorage(this, contextGroupId))))
|
|
.first;
|
|
return storageIt->second.get();
|
|
}
|
|
|
|
bool V8InspectorImpl::hasConsoleMessageStorage(int contextGroupId) {
|
|
ConsoleStorageMap::iterator storageIt =
|
|
m_consoleStorageMap.find(contextGroupId);
|
|
return storageIt != m_consoleStorageMap.end();
|
|
}
|
|
|
|
std::unique_ptr<V8StackTrace> V8InspectorImpl::createStackTrace(
|
|
v8::Local<v8::StackTrace> stackTrace) {
|
|
return m_debugger->createStackTrace(stackTrace);
|
|
}
|
|
|
|
std::unique_ptr<V8InspectorSession> V8InspectorImpl::connect(
|
|
int contextGroupId, V8Inspector::Channel* channel,
|
|
const StringView& state) {
|
|
DCHECK(m_sessions.find(contextGroupId) == m_sessions.cend());
|
|
int sessionId = ++m_lastSessionId;
|
|
std::unique_ptr<V8InspectorSessionImpl> session =
|
|
V8InspectorSessionImpl::create(this, contextGroupId, sessionId, channel,
|
|
state);
|
|
m_sessions[contextGroupId] = session.get();
|
|
m_sessionById[sessionId] = session.get();
|
|
return std::move(session);
|
|
}
|
|
|
|
void V8InspectorImpl::disconnect(V8InspectorSessionImpl* session) {
|
|
DCHECK(m_sessions.find(session->contextGroupId()) != m_sessions.end());
|
|
m_sessions.erase(session->contextGroupId());
|
|
m_sessionById.erase(session->sessionId());
|
|
}
|
|
|
|
InspectedContext* V8InspectorImpl::getContext(int groupId,
|
|
int contextId) const {
|
|
if (!groupId || !contextId) return nullptr;
|
|
|
|
ContextsByGroupMap::const_iterator contextGroupIt = m_contexts.find(groupId);
|
|
if (contextGroupIt == m_contexts.end()) return nullptr;
|
|
|
|
ContextByIdMap::iterator contextIt = contextGroupIt->second->find(contextId);
|
|
if (contextIt == contextGroupIt->second->end()) return nullptr;
|
|
|
|
return contextIt->second.get();
|
|
}
|
|
|
|
void V8InspectorImpl::contextCreated(const V8ContextInfo& info) {
|
|
int contextId = ++m_lastContextId;
|
|
InspectedContext* context = new InspectedContext(this, info, contextId);
|
|
m_contextIdToGroupIdMap[contextId] = info.contextGroupId;
|
|
|
|
ContextsByGroupMap::iterator contextIt = m_contexts.find(info.contextGroupId);
|
|
if (contextIt == m_contexts.end())
|
|
contextIt = m_contexts
|
|
.insert(std::make_pair(
|
|
info.contextGroupId,
|
|
std::unique_ptr<ContextByIdMap>(new ContextByIdMap())))
|
|
.first;
|
|
const auto& contextById = contextIt->second;
|
|
|
|
DCHECK(contextById->find(contextId) == contextById->cend());
|
|
(*contextById)[contextId].reset(context);
|
|
SessionMap::iterator sessionIt = m_sessions.find(info.contextGroupId);
|
|
if (sessionIt != m_sessions.end())
|
|
sessionIt->second->runtimeAgent()->reportExecutionContextCreated(context);
|
|
}
|
|
|
|
void V8InspectorImpl::contextDestroyed(v8::Local<v8::Context> context) {
|
|
int contextId = InspectedContext::contextId(context);
|
|
int groupId = contextGroupId(context);
|
|
m_contextIdToGroupIdMap.erase(contextId);
|
|
|
|
ConsoleStorageMap::iterator storageIt = m_consoleStorageMap.find(groupId);
|
|
if (storageIt != m_consoleStorageMap.end())
|
|
storageIt->second->contextDestroyed(contextId);
|
|
|
|
InspectedContext* inspectedContext = getContext(groupId, contextId);
|
|
if (!inspectedContext) return;
|
|
|
|
SessionMap::iterator iter = m_sessions.find(groupId);
|
|
if (iter != m_sessions.end())
|
|
iter->second->runtimeAgent()->reportExecutionContextDestroyed(
|
|
inspectedContext);
|
|
discardInspectedContext(groupId, contextId);
|
|
}
|
|
|
|
void V8InspectorImpl::resetContextGroup(int contextGroupId) {
|
|
m_consoleStorageMap.erase(contextGroupId);
|
|
m_muteExceptionsMap.erase(contextGroupId);
|
|
SessionMap::iterator session = m_sessions.find(contextGroupId);
|
|
if (session != m_sessions.end()) session->second->reset();
|
|
m_contexts.erase(contextGroupId);
|
|
m_debugger->wasmTranslation()->Clear();
|
|
}
|
|
|
|
void V8InspectorImpl::idleStarted() {
|
|
for (auto it = m_sessions.begin(); it != m_sessions.end(); ++it) {
|
|
if (it->second->profilerAgent()->idleStarted()) return;
|
|
}
|
|
}
|
|
|
|
void V8InspectorImpl::idleFinished() {
|
|
for (auto it = m_sessions.begin(); it != m_sessions.end(); ++it) {
|
|
if (it->second->profilerAgent()->idleFinished()) return;
|
|
}
|
|
}
|
|
|
|
unsigned V8InspectorImpl::exceptionThrown(
|
|
v8::Local<v8::Context> context, const StringView& message,
|
|
v8::Local<v8::Value> exception, const StringView& detailedMessage,
|
|
const StringView& url, unsigned lineNumber, unsigned columnNumber,
|
|
std::unique_ptr<V8StackTrace> stackTrace, int scriptId) {
|
|
int groupId = contextGroupId(context);
|
|
if (!groupId || m_muteExceptionsMap[groupId]) return 0;
|
|
std::unique_ptr<V8StackTraceImpl> stackTraceImpl(
|
|
static_cast<V8StackTraceImpl*>(stackTrace.release()));
|
|
unsigned exceptionId = nextExceptionId();
|
|
std::unique_ptr<V8ConsoleMessage> consoleMessage =
|
|
V8ConsoleMessage::createForException(
|
|
m_client->currentTimeMS(), toString16(detailedMessage),
|
|
toString16(url), lineNumber, columnNumber, std::move(stackTraceImpl),
|
|
scriptId, m_isolate, toString16(message),
|
|
InspectedContext::contextId(context), exception, exceptionId);
|
|
ensureConsoleMessageStorage(groupId)->addMessage(std::move(consoleMessage));
|
|
return exceptionId;
|
|
}
|
|
|
|
void V8InspectorImpl::exceptionRevoked(v8::Local<v8::Context> context,
|
|
unsigned exceptionId,
|
|
const StringView& message) {
|
|
int groupId = contextGroupId(context);
|
|
if (!groupId) return;
|
|
|
|
std::unique_ptr<V8ConsoleMessage> consoleMessage =
|
|
V8ConsoleMessage::createForRevokedException(
|
|
m_client->currentTimeMS(), toString16(message), exceptionId);
|
|
ensureConsoleMessageStorage(groupId)->addMessage(std::move(consoleMessage));
|
|
}
|
|
|
|
std::unique_ptr<V8StackTrace> V8InspectorImpl::captureStackTrace(
|
|
bool fullStack) {
|
|
return m_debugger->captureStackTrace(fullStack);
|
|
}
|
|
|
|
void V8InspectorImpl::asyncTaskScheduled(const StringView& taskName, void* task,
|
|
bool recurring) {
|
|
m_debugger->asyncTaskScheduled(taskName, task, recurring);
|
|
}
|
|
|
|
void V8InspectorImpl::asyncTaskCanceled(void* task) {
|
|
m_debugger->asyncTaskCanceled(task);
|
|
}
|
|
|
|
void V8InspectorImpl::asyncTaskStarted(void* task) {
|
|
m_debugger->asyncTaskStarted(task);
|
|
}
|
|
|
|
void V8InspectorImpl::asyncTaskFinished(void* task) {
|
|
m_debugger->asyncTaskFinished(task);
|
|
}
|
|
|
|
void V8InspectorImpl::allAsyncTasksCanceled() {
|
|
m_debugger->allAsyncTasksCanceled();
|
|
}
|
|
|
|
v8::Local<v8::Context> V8InspectorImpl::regexContext() {
|
|
if (m_regexContext.IsEmpty())
|
|
m_regexContext.Reset(m_isolate, v8::Context::New(m_isolate));
|
|
return m_regexContext.Get(m_isolate);
|
|
}
|
|
|
|
void V8InspectorImpl::discardInspectedContext(int contextGroupId,
|
|
int contextId) {
|
|
if (!getContext(contextGroupId, contextId)) return;
|
|
m_contexts[contextGroupId]->erase(contextId);
|
|
if (m_contexts[contextGroupId]->empty()) m_contexts.erase(contextGroupId);
|
|
}
|
|
|
|
V8InspectorSessionImpl* V8InspectorImpl::sessionForContextGroup(
|
|
int contextGroupId) {
|
|
if (!contextGroupId) return nullptr;
|
|
SessionMap::iterator iter = m_sessions.find(contextGroupId);
|
|
return iter == m_sessions.end() ? nullptr : iter->second;
|
|
}
|
|
|
|
V8InspectorSessionImpl* V8InspectorImpl::sessionById(int sessionId) {
|
|
auto it = m_sessionById.find(sessionId);
|
|
return it == m_sessionById.end() ? nullptr : it->second;
|
|
}
|
|
|
|
V8Console* V8InspectorImpl::console() {
|
|
if (!m_console) m_console.reset(new V8Console(this));
|
|
return m_console.get();
|
|
}
|
|
|
|
void V8InspectorImpl::forEachContext(
|
|
int contextGroupId, std::function<void(InspectedContext*)> callback) {
|
|
auto it = m_contexts.find(contextGroupId);
|
|
if (it == m_contexts.end()) return;
|
|
std::vector<int> ids;
|
|
ids.reserve(it->second->size());
|
|
for (auto& contextIt : *(it->second)) ids.push_back(contextIt.first);
|
|
|
|
// Retrieve by ids each time since |callback| may destroy some contexts.
|
|
for (auto& contextId : ids) {
|
|
it = m_contexts.find(contextGroupId);
|
|
if (it == m_contexts.end()) continue;
|
|
auto contextIt = it->second->find(contextId);
|
|
if (contextIt != it->second->end()) callback(contextIt->second.get());
|
|
}
|
|
}
|
|
|
|
} // namespace v8_inspector
|