blob: 6e7f586f1ecd6f37c2ac34808f292065c71d10e0 [file] [log] [blame]
/*
* Copyright (C) 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 "third_party/blink/renderer/core/inspector/worker_inspector_controller.h"
#include "third_party/blink/renderer/core/core_probe_sink.h"
#include "third_party/blink/renderer/core/inspector/inspector_emulation_agent.h"
#include "third_party/blink/renderer/core/inspector/inspector_log_agent.h"
#include "third_party/blink/renderer/core/inspector/inspector_network_agent.h"
#include "third_party/blink/renderer/core/inspector/inspector_trace_events.h"
#include "third_party/blink/renderer/core/inspector/inspector_worker_agent.h"
#include "third_party/blink/renderer/core/inspector/protocol/Protocol.h"
#include "third_party/blink/renderer/core/inspector/worker_thread_debugger.h"
#include "third_party/blink/renderer/core/loader/worker_fetch_context.h"
#include "third_party/blink/renderer/core/probe/core_probes.h"
#include "third_party/blink/renderer/core/workers/worker_backing_thread.h"
#include "third_party/blink/renderer/core/workers/worker_global_scope.h"
#include "third_party/blink/renderer/core/workers/worker_reporting_proxy.h"
#include "third_party/blink/renderer/core/workers/worker_thread.h"
#include "third_party/blink/renderer/platform/layout_test_support.h"
#include "third_party/blink/renderer/platform/runtime_enabled_features.h"
#include "third_party/blink/renderer/platform/web_thread_supporting_gc.h"
namespace blink {
WorkerInspectorController* WorkerInspectorController::Create(
WorkerThread* thread) {
WorkerThreadDebugger* debugger =
WorkerThreadDebugger::From(thread->GetIsolate());
return debugger ? new WorkerInspectorController(thread, debugger) : nullptr;
}
WorkerInspectorController::WorkerInspectorController(
WorkerThread* thread,
WorkerThreadDebugger* debugger)
: debugger_(debugger), thread_(thread), probe_sink_(new CoreProbeSink()) {
probe_sink_->addInspectorTraceEvents(new InspectorTraceEvents());
if (auto* scope = DynamicTo<WorkerGlobalScope>(thread->GlobalScope())) {
worker_devtools_token_ = thread->GetDevToolsWorkerToken();
parent_devtools_token_ = scope->GetParentDevToolsToken();
url_ = scope->Url();
worker_thread_id_ = thread->GetPlatformThreadId();
}
TraceEvent::AddEnabledStateObserver(this);
EmitTraceEvent();
}
WorkerInspectorController::~WorkerInspectorController() {
DCHECK(!thread_);
TraceEvent::RemoveEnabledStateObserver(this);
}
void WorkerInspectorController::ConnectFrontend(int session_id) {
WorkerThread::ScopedDebuggerTask debugger_task(thread_);
if (sessions_.find(session_id) != sessions_.end())
return;
InspectedFrames* inspected_frames = new InspectedFrames(nullptr);
InspectorSession* session = new InspectorSession(
this, probe_sink_.Get(), inspected_frames, session_id,
debugger_->GetV8Inspector(), debugger_->ContextGroupId(thread_), nullptr);
session->Append(new InspectorLogAgent(thread_->GetConsoleMessageStorage(),
nullptr, session->V8Session()));
if (auto* scope = DynamicTo<WorkerGlobalScope>(thread_->GlobalScope())) {
DCHECK(scope->EnsureFetcher());
session->Append(new InspectorNetworkAgent(inspected_frames, scope,
session->V8Session()));
session->Append(new InspectorEmulationAgent(nullptr));
session->Append(new InspectorWorkerAgent(inspected_frames, scope));
}
if (sessions_.IsEmpty())
thread_->GetWorkerBackingThread().BackingThread().AddTaskObserver(this);
sessions_.insert(session_id, session);
}
void WorkerInspectorController::DisconnectFrontend(int session_id) {
WorkerThread::ScopedDebuggerTask debugger_task(thread_);
auto it = sessions_.find(session_id);
if (it == sessions_.end())
return;
it->value->Dispose();
sessions_.erase(it);
if (sessions_.IsEmpty())
thread_->GetWorkerBackingThread().BackingThread().RemoveTaskObserver(this);
}
void WorkerInspectorController::DispatchMessageFromFrontend(
int session_id,
const String& message) {
WorkerThread::ScopedDebuggerTask debugger_task(thread_);
auto it = sessions_.find(session_id);
if (it == sessions_.end())
return;
it->value->DispatchProtocolMessage(message);
}
void WorkerInspectorController::Dispose() {
Vector<int> ids;
CopyKeysToVector(sessions_, ids);
for (int session_id : ids)
DisconnectFrontend(session_id);
thread_ = nullptr;
}
void WorkerInspectorController::FlushProtocolNotifications() {
for (auto& it : sessions_)
it.value->flushProtocolNotifications();
}
void WorkerInspectorController::SendProtocolResponse(
int session_id,
int call_id,
const String& response,
mojom::blink::DevToolsSessionStatePtr updates) {
// Make tests more predictable by flushing all sessions before sending
// protocol response in any of them.
if (LayoutTestSupport::IsRunningLayoutTest())
FlushProtocolNotifications();
// Worker messages are wrapped, no need to handle callId or state.
thread_->GetWorkerReportingProxy().PostMessageToPageInspector(session_id,
response);
}
void WorkerInspectorController::SendProtocolNotification(
int session_id,
const String& message,
mojom::blink::DevToolsSessionStatePtr updates) {
thread_->GetWorkerReportingProxy().PostMessageToPageInspector(session_id,
message);
}
void WorkerInspectorController::WillProcessTask(
const base::PendingTask& pending_task) {}
void WorkerInspectorController::DidProcessTask(
const base::PendingTask& pending_task) {
for (auto& it : sessions_)
it.value->flushProtocolNotifications();
}
void WorkerInspectorController::OnTraceLogEnabled() {
EmitTraceEvent();
}
void WorkerInspectorController::OnTraceLogDisabled() {}
void WorkerInspectorController::EmitTraceEvent() {
if (worker_devtools_token_.is_empty())
return;
TRACE_EVENT_INSTANT1(TRACE_DISABLED_BY_DEFAULT("devtools.timeline"),
"TracingSessionIdForWorker", TRACE_EVENT_SCOPE_THREAD,
"data",
InspectorTracingSessionIdForWorkerEvent::Data(
worker_devtools_token_, parent_devtools_token_, url_,
worker_thread_id_));
}
void WorkerInspectorController::Trace(blink::Visitor* visitor) {
visitor->Trace(probe_sink_);
visitor->Trace(sessions_);
}
} // namespace blink