/* * Copyright (C) 2008, 2009 Apple Inc. All rights reserved. * 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: * * 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 "ScriptDebugServer.h" #if ENABLE(JAVASCRIPT_DEBUGGER) #include "EventLoop.h" #include "Frame.h" #include "JavaScriptCallFrame.h" #include "ScriptBreakpoint.h" #include "ScriptController.h" #include "ScriptDebugListener.h" #include <debugger/DebuggerCallFrame.h> #include <parser/SourceProvider.h> #include <runtime/JSLock.h> #include <wtf/text/StringConcatenate.h> #include <wtf/MainThread.h> using namespace JSC; namespace WebCore { ScriptDebugServer::ScriptDebugServer() : m_callingListeners(false) , m_pauseOnExceptionsState(DontPauseOnExceptions) , m_pauseOnNextStatement(false) , m_paused(false) , m_doneProcessingDebuggerEvents(true) , m_breakpointsActivated(true) , m_pauseOnCallFrame(0) , m_recompileTimer(this, &ScriptDebugServer::recompileAllJSFunctions) { } ScriptDebugServer::~ScriptDebugServer() { deleteAllValues(m_pageListenersMap); } String ScriptDebugServer::setBreakpoint(const String& sourceID, const ScriptBreakpoint& scriptBreakpoint, int* actualLineNumber, int* actualColumnNumber) { intptr_t sourceIDValue = sourceID.toIntPtr(); if (!sourceIDValue) return ""; SourceIdToBreakpointsMap::iterator it = m_sourceIdToBreakpoints.find(sourceIDValue); if (it == m_sourceIdToBreakpoints.end()) it = m_sourceIdToBreakpoints.set(sourceIDValue, LineToBreakpointMap()).first; if (it->second.contains(scriptBreakpoint.lineNumber + 1)) return ""; it->second.set(scriptBreakpoint.lineNumber + 1, scriptBreakpoint); *actualLineNumber = scriptBreakpoint.lineNumber; // FIXME(WK53003): implement setting breakpoints by line:column. *actualColumnNumber = 0; return makeString(sourceID, ":", String::number(scriptBreakpoint.lineNumber)); } void ScriptDebugServer::removeBreakpoint(const String& breakpointId) { Vector<String> tokens; breakpointId.split(":", tokens); if (tokens.size() != 2) return; bool success; intptr_t sourceIDValue = tokens[0].toIntPtr(&success); if (!success) return; unsigned lineNumber = tokens[1].toUInt(&success); if (!success) return; SourceIdToBreakpointsMap::iterator it = m_sourceIdToBreakpoints.find(sourceIDValue); if (it != m_sourceIdToBreakpoints.end()) it->second.remove(lineNumber + 1); } bool ScriptDebugServer::hasBreakpoint(intptr_t sourceID, const TextPosition0& position) const { if (!m_breakpointsActivated) return false; SourceIdToBreakpointsMap::const_iterator it = m_sourceIdToBreakpoints.find(sourceID); if (it == m_sourceIdToBreakpoints.end()) return false; int lineNumber = position.m_line.convertAsOneBasedInt(); if (lineNumber <= 0) return false; LineToBreakpointMap::const_iterator breakIt = it->second.find(lineNumber); if (breakIt == it->second.end()) return false; // An empty condition counts as no condition which is equivalent to "true". if (breakIt->second.condition.isEmpty()) return true; JSValue exception; JSValue result = m_currentCallFrame->evaluate(stringToUString(breakIt->second.condition), exception); if (exception) { // An erroneous condition counts as "false". return false; } return result.toBoolean(m_currentCallFrame->scopeChain()->globalObject->globalExec()); } void ScriptDebugServer::clearBreakpoints() { m_sourceIdToBreakpoints.clear(); } void ScriptDebugServer::setBreakpointsActivated(bool activated) { m_breakpointsActivated = activated; } void ScriptDebugServer::setPauseOnExceptionsState(PauseOnExceptionsState pause) { m_pauseOnExceptionsState = pause; } void ScriptDebugServer::setPauseOnNextStatement(bool pause) { m_pauseOnNextStatement = pause; } void ScriptDebugServer::breakProgram() { // FIXME(WK43332): implement this. } void ScriptDebugServer::continueProgram() { if (!m_paused) return; m_pauseOnNextStatement = false; m_doneProcessingDebuggerEvents = true; } void ScriptDebugServer::stepIntoStatement() { if (!m_paused) return; m_pauseOnNextStatement = true; m_doneProcessingDebuggerEvents = true; } void ScriptDebugServer::stepOverStatement() { if (!m_paused) return; m_pauseOnCallFrame = m_currentCallFrame.get(); m_doneProcessingDebuggerEvents = true; } void ScriptDebugServer::stepOutOfFunction() { if (!m_paused) return; m_pauseOnCallFrame = m_currentCallFrame ? m_currentCallFrame->caller() : 0; m_doneProcessingDebuggerEvents = true; } bool ScriptDebugServer::editScriptSource(const String&, const String&, String*) { // FIXME(40300): implement this. return false; } JavaScriptCallFrame* ScriptDebugServer::currentCallFrame() { if (!m_paused) return 0; return m_currentCallFrame.get(); } void ScriptDebugServer::dispatchDidPause(ScriptDebugListener* listener) { ASSERT(m_paused); ScriptState* state = m_currentCallFrame->scopeChain()->globalObject->globalExec(); listener->didPause(state); } void ScriptDebugServer::dispatchDidContinue(ScriptDebugListener* listener) { listener->didContinue(); } void ScriptDebugServer::dispatchDidParseSource(const ListenerSet& listeners, SourceProvider* sourceProvider, bool isContentScript) { String sourceID = ustringToString(JSC::UString::number(sourceProvider->asID())); String url = ustringToString(sourceProvider->url()); String data = ustringToString(JSC::UString(sourceProvider->data(), sourceProvider->length())); int lineOffset = sourceProvider->startPosition().m_line.convertAsZeroBasedInt(); int columnOffset = sourceProvider->startPosition().m_column.convertAsZeroBasedInt(); Vector<ScriptDebugListener*> copy; copyToVector(listeners, copy); for (size_t i = 0; i < copy.size(); ++i) copy[i]->didParseSource(sourceID, url, data, lineOffset, columnOffset, isContentScript); } void ScriptDebugServer::dispatchFailedToParseSource(const ListenerSet& listeners, SourceProvider* sourceProvider, int errorLine, const String& errorMessage) { String url = ustringToString(sourceProvider->url()); String data = ustringToString(JSC::UString(sourceProvider->data(), sourceProvider->length())); int firstLine = sourceProvider->startPosition().m_line.oneBasedInt(); Vector<ScriptDebugListener*> copy; copyToVector(listeners, copy); for (size_t i = 0; i < copy.size(); ++i) copy[i]->failedToParseSource(url, data, firstLine, errorLine, errorMessage); } static bool isContentScript(ExecState* exec) { return currentWorld(exec) != mainThreadNormalWorld(); } void ScriptDebugServer::detach(JSGlobalObject* globalObject) { // If we're detaching from the currently executing global object, manually tear down our // stack, since we won't get further debugger callbacks to do so. Also, resume execution, // since there's no point in staying paused once a window closes. if (m_currentCallFrame && m_currentCallFrame->dynamicGlobalObject() == globalObject) { m_currentCallFrame = 0; m_pauseOnCallFrame = 0; continueProgram(); } Debugger::detach(globalObject); } void ScriptDebugServer::sourceParsed(ExecState* exec, SourceProvider* sourceProvider, int errorLine, const UString& errorMessage) { if (m_callingListeners) return; ListenerSet* listeners = getListenersForGlobalObject(exec->lexicalGlobalObject()); if (!listeners) return; ASSERT(!listeners->isEmpty()); m_callingListeners = true; bool isError = errorLine != -1; if (isError) dispatchFailedToParseSource(*listeners, sourceProvider, errorLine, ustringToString(errorMessage)); else dispatchDidParseSource(*listeners, sourceProvider, isContentScript(exec)); m_callingListeners = false; } void ScriptDebugServer::dispatchFunctionToListeners(const ListenerSet& listeners, JavaScriptExecutionCallback callback) { Vector<ScriptDebugListener*> copy; copyToVector(listeners, copy); for (size_t i = 0; i < copy.size(); ++i) (this->*callback)(copy[i]); } void ScriptDebugServer::dispatchFunctionToListeners(JavaScriptExecutionCallback callback, JSGlobalObject* globalObject) { if (m_callingListeners) return; m_callingListeners = true; if (ListenerSet* listeners = getListenersForGlobalObject(globalObject)) { ASSERT(!listeners->isEmpty()); dispatchFunctionToListeners(*listeners, callback); } m_callingListeners = false; } void ScriptDebugServer::createCallFrameAndPauseIfNeeded(const DebuggerCallFrame& debuggerCallFrame, intptr_t sourceID, int lineNumber) { TextPosition0 textPosition(WTF::OneBasedNumber::fromOneBasedInt(lineNumber).convertToZeroBased(), WTF::ZeroBasedNumber::base()); m_currentCallFrame = JavaScriptCallFrame::create(debuggerCallFrame, m_currentCallFrame, sourceID, textPosition); pauseIfNeeded(debuggerCallFrame.dynamicGlobalObject()); } void ScriptDebugServer::updateCallFrameAndPauseIfNeeded(const DebuggerCallFrame& debuggerCallFrame, intptr_t sourceID, int lineNumber) { ASSERT(m_currentCallFrame); if (!m_currentCallFrame) return; TextPosition0 textPosition(WTF::OneBasedNumber::fromOneBasedInt(lineNumber).convertToZeroBased(), WTF::ZeroBasedNumber::base()); m_currentCallFrame->update(debuggerCallFrame, sourceID, textPosition); pauseIfNeeded(debuggerCallFrame.dynamicGlobalObject()); } void ScriptDebugServer::pauseIfNeeded(JSGlobalObject* dynamicGlobalObject) { if (m_paused) return; if (!getListenersForGlobalObject(dynamicGlobalObject)) return; bool pauseNow = m_pauseOnNextStatement; pauseNow |= (m_pauseOnCallFrame == m_currentCallFrame); pauseNow |= hasBreakpoint(m_currentCallFrame->sourceID(), m_currentCallFrame->position()); if (!pauseNow) return; m_pauseOnCallFrame = 0; m_pauseOnNextStatement = false; m_paused = true; dispatchFunctionToListeners(&ScriptDebugServer::dispatchDidPause, dynamicGlobalObject); didPause(dynamicGlobalObject); TimerBase::fireTimersInNestedEventLoop(); EventLoop loop; m_doneProcessingDebuggerEvents = false; while (!m_doneProcessingDebuggerEvents && !loop.ended()) loop.cycle(); didContinue(dynamicGlobalObject); dispatchFunctionToListeners(&ScriptDebugServer::dispatchDidContinue, dynamicGlobalObject); m_paused = false; } void ScriptDebugServer::callEvent(const DebuggerCallFrame& debuggerCallFrame, intptr_t sourceID, int lineNumber) { if (!m_paused) createCallFrameAndPauseIfNeeded(debuggerCallFrame, sourceID, lineNumber); } void ScriptDebugServer::atStatement(const DebuggerCallFrame& debuggerCallFrame, intptr_t sourceID, int lineNumber) { if (!m_paused) updateCallFrameAndPauseIfNeeded(debuggerCallFrame, sourceID, lineNumber); } void ScriptDebugServer::returnEvent(const DebuggerCallFrame& debuggerCallFrame, intptr_t sourceID, int lineNumber) { if (m_paused) return; updateCallFrameAndPauseIfNeeded(debuggerCallFrame, sourceID, lineNumber); // detach may have been called during pauseIfNeeded if (!m_currentCallFrame) return; // Treat stepping over a return statement like stepping out. if (m_currentCallFrame == m_pauseOnCallFrame) m_pauseOnCallFrame = m_currentCallFrame->caller(); m_currentCallFrame = m_currentCallFrame->caller(); } void ScriptDebugServer::exception(const DebuggerCallFrame& debuggerCallFrame, intptr_t sourceID, int lineNumber, bool hasHandler) { if (m_paused) return; if (m_pauseOnExceptionsState == PauseOnAllExceptions || (m_pauseOnExceptionsState == PauseOnUncaughtExceptions && !hasHandler)) m_pauseOnNextStatement = true; updateCallFrameAndPauseIfNeeded(debuggerCallFrame, sourceID, lineNumber); } void ScriptDebugServer::willExecuteProgram(const DebuggerCallFrame& debuggerCallFrame, intptr_t sourceID, int lineNumber) { if (!m_paused) createCallFrameAndPauseIfNeeded(debuggerCallFrame, sourceID, lineNumber); } void ScriptDebugServer::didExecuteProgram(const DebuggerCallFrame& debuggerCallFrame, intptr_t sourceID, int lineNumber) { if (m_paused) return; updateCallFrameAndPauseIfNeeded(debuggerCallFrame, sourceID, lineNumber); // Treat stepping over the end of a program like stepping out. if (m_currentCallFrame == m_pauseOnCallFrame) m_pauseOnCallFrame = m_currentCallFrame->caller(); m_currentCallFrame = m_currentCallFrame->caller(); } void ScriptDebugServer::didReachBreakpoint(const DebuggerCallFrame& debuggerCallFrame, intptr_t sourceID, int lineNumber) { if (m_paused) return; m_pauseOnNextStatement = true; updateCallFrameAndPauseIfNeeded(debuggerCallFrame, sourceID, lineNumber); } void ScriptDebugServer::recompileAllJSFunctionsSoon() { m_recompileTimer.startOneShot(0); } } // namespace WebCore #endif // ENABLE(JAVASCRIPT_DEBUGGER)