blob: ef72e4dad6a88373c0e00c19733db655520eaed0 [file] [log] [blame]
/*
* Copyright (C) 2014-2021 Apple 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.
*
* THIS SOFTWARE IS PROVIDED BY APPLE INC. ``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 INC. 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 "config.h"
#include "JSGlobalObjectInspectorController.h"
#include "CatchScope.h"
#include "Completion.h"
#include "ConsoleMessage.h"
#include "ErrorHandlingScope.h"
#include "Exception.h"
#include "InjectedScriptHost.h"
#include "InjectedScriptManager.h"
#include "InspectorAgent.h"
#include "InspectorBackendDispatcher.h"
#include "InspectorConsoleAgent.h"
#include "InspectorFrontendRouter.h"
#include "InspectorHeapAgent.h"
#include "InspectorScriptProfilerAgent.h"
#include "JSGlobalObject.h"
#include "JSGlobalObjectAuditAgent.h"
#include "JSGlobalObjectConsoleClient.h"
#include "JSGlobalObjectDebuggerAgent.h"
#include "JSGlobalObjectRuntimeAgent.h"
#include "ScriptCallStack.h"
#include "ScriptCallStackFactory.h"
#include <wtf/StackTrace.h>
#include <wtf/Stopwatch.h>
#if ENABLE(REMOTE_INSPECTOR)
#include "JSGlobalObjectDebuggable.h"
#include "RemoteInspector.h"
#endif
namespace Inspector {
using namespace JSC;
JSGlobalObjectInspectorController::JSGlobalObjectInspectorController(JSGlobalObject& globalObject)
: m_globalObject(globalObject)
, m_injectedScriptManager(makeUnique<InjectedScriptManager>(*this, InjectedScriptHost::create()))
, m_executionStopwatch(Stopwatch::create())
, m_debugger(globalObject)
, m_frontendRouter(FrontendRouter::create())
, m_backendDispatcher(BackendDispatcher::create(m_frontendRouter.copyRef()))
{
auto context = jsAgentContext();
auto consoleAgent = makeUnique<InspectorConsoleAgent>(context);
m_consoleAgent = consoleAgent.get();
m_agents.append(WTFMove(consoleAgent));
m_consoleClient = makeUnique<JSGlobalObjectConsoleClient>(m_consoleAgent);
m_executionStopwatch->start();
}
JSGlobalObjectInspectorController::~JSGlobalObjectInspectorController()
{
#if ENABLE(INSPECTOR_ALTERNATE_DISPATCHERS)
if (m_augmentingClient)
m_augmentingClient->inspectorControllerDestroyed();
#endif
}
void JSGlobalObjectInspectorController::globalObjectDestroyed()
{
ASSERT(!m_frontendRouter->hasFrontends());
m_injectedScriptManager->disconnect();
m_agents.discardValues();
}
void JSGlobalObjectInspectorController::connectFrontend(FrontendChannel& frontendChannel, bool isAutomaticInspection, bool immediatelyPause)
{
m_isAutomaticInspection = isAutomaticInspection;
m_pauseAfterInitialization = immediatelyPause;
createLazyAgents();
bool connectedFirstFrontend = !m_frontendRouter->hasFrontends();
m_frontendRouter->connectFrontend(frontendChannel);
if (!connectedFirstFrontend)
return;
// Keep the JSGlobalObject and VM alive while we are debugging it.
m_strongVM = &m_globalObject.vm();
m_strongGlobalObject.set(m_globalObject.vm(), &m_globalObject);
// FIXME: change this to notify agents which frontend has connected (by id).
m_agents.didCreateFrontendAndBackend(nullptr, nullptr);
#if ENABLE(INSPECTOR_ALTERNATE_DISPATCHERS)
if (m_augmentingClient)
m_augmentingClient->inspectorConnected();
#endif
}
void JSGlobalObjectInspectorController::disconnectFrontend(FrontendChannel& frontendChannel)
{
// FIXME: change this to notify agents which frontend has disconnected (by id).
m_agents.willDestroyFrontendAndBackend(DisconnectReason::InspectorDestroyed);
m_frontendRouter->disconnectFrontend(frontendChannel);
m_isAutomaticInspection = false;
m_pauseAfterInitialization = false;
bool disconnectedLastFrontend = !m_frontendRouter->hasFrontends();
if (!disconnectedLastFrontend)
return;
#if ENABLE(INSPECTOR_ALTERNATE_DISPATCHERS)
if (m_augmentingClient)
m_augmentingClient->inspectorDisconnected();
#endif
// Remove our JSGlobalObject and VM references, we are done debugging it.
m_strongGlobalObject.clear();
m_strongVM = nullptr;
}
void JSGlobalObjectInspectorController::dispatchMessageFromFrontend(const String& message)
{
m_backendDispatcher->dispatch(message);
}
void JSGlobalObjectInspectorController::appendAPIBacktrace(ScriptCallStack& callStack)
{
static constexpr int framesToShow = 31;
static constexpr int framesToSkip = 3; // WTFGetBacktrace, appendAPIBacktrace, reportAPIException.
void* samples[framesToShow + framesToSkip];
int frames = framesToShow + framesToSkip;
WTFGetBacktrace(samples, &frames);
void** stack = samples + framesToSkip;
int size = frames - framesToSkip;
for (int i = 0; i < size; ++i) {
auto demangled = StackTrace::demangle(stack[i]);
if (demangled)
callStack.append(ScriptCallFrame(String::fromLatin1(demangled->demangledName() ? demangled->demangledName() : demangled->mangledName()), "[native code]"_s, noSourceID, 0, 0));
else
callStack.append(ScriptCallFrame("?"_s, "[native code]"_s, noSourceID, 0, 0));
}
}
void JSGlobalObjectInspectorController::reportAPIException(JSGlobalObject* globalObject, Exception* exception)
{
VM& vm = globalObject->vm();
if (vm.isTerminationException(exception))
return;
auto scope = DECLARE_CATCH_SCOPE(vm);
ErrorHandlingScope errorScope(vm);
Ref<ScriptCallStack> callStack = createScriptCallStackFromException(globalObject, exception);
if (includesNativeCallStackWhenReportingExceptions())
appendAPIBacktrace(callStack.get());
// FIXME: <http://webkit.org/b/115087> Web Inspector: Should not evaluate JavaScript handling exceptions
// If this is a custom exception object, call toString on it to try and get a nice string representation for the exception.
String errorMessage = exception->value().toWTFString(globalObject);
scope.clearException();
if (JSGlobalObjectConsoleClient::logToSystemConsole()) {
if (callStack->size()) {
const ScriptCallFrame& callFrame = callStack->at(0);
ConsoleClient::printConsoleMessage(MessageSource::JS, MessageType::Log, MessageLevel::Error, errorMessage, callFrame.sourceURL(), callFrame.lineNumber(), callFrame.columnNumber());
} else
ConsoleClient::printConsoleMessage(MessageSource::JS, MessageType::Log, MessageLevel::Error, errorMessage, String(), 0, 0);
}
m_consoleAgent->addMessageToConsole(makeUnique<ConsoleMessage>(MessageSource::JS, MessageType::Log, MessageLevel::Error, errorMessage, WTFMove(callStack)));
}
WeakPtr<ConsoleClient> JSGlobalObjectInspectorController::consoleClient() const
{
return m_consoleClient.get();
}
bool JSGlobalObjectInspectorController::developerExtrasEnabled() const
{
#if ENABLE(REMOTE_INSPECTOR)
if (!RemoteInspector::singleton().enabled())
return false;
if (!m_globalObject.inspectorDebuggable().remoteDebuggingAllowed())
return false;
#endif
return true;
}
InspectorFunctionCallHandler JSGlobalObjectInspectorController::functionCallHandler() const
{
return JSC::call;
}
InspectorEvaluateHandler JSGlobalObjectInspectorController::evaluateHandler() const
{
return JSC::evaluate;
}
void JSGlobalObjectInspectorController::frontendInitialized()
{
if (m_pauseAfterInitialization) {
m_pauseAfterInitialization = false;
ensureDebuggerAgent().enable();
ensureDebuggerAgent().pause();
}
#if ENABLE(REMOTE_INSPECTOR)
if (m_isAutomaticInspection)
m_globalObject.inspectorDebuggable().unpauseForInitializedInspector();
#endif
}
Stopwatch& JSGlobalObjectInspectorController::executionStopwatch() const
{
return m_executionStopwatch;
}
JSGlobalObjectDebugger& JSGlobalObjectInspectorController::debugger()
{
return m_debugger;
}
VM& JSGlobalObjectInspectorController::vm()
{
return m_globalObject.vm();
}
#if ENABLE(INSPECTOR_ALTERNATE_DISPATCHERS)
void JSGlobalObjectInspectorController::registerAlternateAgent(std::unique_ptr<InspectorAgentBase> agent)
{
// FIXME: change this to notify agents which frontend has connected (by id).
agent->didCreateFrontendAndBackend(nullptr, nullptr);
m_agents.append(WTFMove(agent));
}
#endif
InspectorAgent& JSGlobalObjectInspectorController::ensureInspectorAgent()
{
if (!m_inspectorAgent) {
auto context = jsAgentContext();
auto inspectorAgent = makeUnique<InspectorAgent>(context);
m_inspectorAgent = inspectorAgent.get();
m_agents.append(WTFMove(inspectorAgent));
}
return *m_inspectorAgent;
}
InspectorDebuggerAgent& JSGlobalObjectInspectorController::ensureDebuggerAgent()
{
if (!m_debuggerAgent) {
auto context = jsAgentContext();
auto debuggerAgent = makeUnique<JSGlobalObjectDebuggerAgent>(context, m_consoleAgent);
m_debuggerAgent = debuggerAgent.get();
m_consoleClient->setDebuggerAgent(m_debuggerAgent);
m_agents.append(WTFMove(debuggerAgent));
}
return *m_debuggerAgent;
}
JSAgentContext JSGlobalObjectInspectorController::jsAgentContext()
{
AgentContext baseContext = {
*this,
*m_injectedScriptManager,
m_frontendRouter.get(),
m_backendDispatcher.get()
};
JSAgentContext context = {
baseContext,
m_globalObject
};
return context;
}
void JSGlobalObjectInspectorController::createLazyAgents()
{
if (m_didCreateLazyAgents)
return;
m_didCreateLazyAgents = true;
auto context = jsAgentContext();
ensureInspectorAgent();
m_agents.append(makeUnique<JSGlobalObjectRuntimeAgent>(context));
ensureDebuggerAgent();
auto scriptProfilerAgentPtr = makeUnique<InspectorScriptProfilerAgent>(context);
m_consoleClient->setPersistentScriptProfilerAgent(scriptProfilerAgentPtr.get());
m_agents.append(WTFMove(scriptProfilerAgentPtr));
auto heapAgent = makeUnique<InspectorHeapAgent>(context);
if (m_consoleAgent)
m_consoleAgent->setHeapAgent(heapAgent.get());
m_agents.append(WTFMove(heapAgent));
m_agents.append(makeUnique<JSGlobalObjectAuditAgent>(context));
}
} // namespace Inspector