/* * Copyright (C) 2008 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. * 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 "JavaScriptDebugServer.h" #if ENABLE(JAVASCRIPT_DEBUGGER) #include "DOMWindow.h" #include "EventLoop.h" #include "Frame.h" #include "FrameTree.h" #include "FrameView.h" #include "JSDOMWindowCustom.h" #include "JavaScriptCallFrame.h" #include "JavaScriptDebugListener.h" #include "Page.h" #include "PageGroup.h" #include "PluginView.h" #include "ScrollView.h" #include "Widget.h" #include "ScriptController.h" #include #include #include #include #include #include #include using namespace JSC; namespace WebCore { typedef JavaScriptDebugServer::ListenerSet ListenerSet; JavaScriptDebugServer& JavaScriptDebugServer::shared() { DEFINE_STATIC_LOCAL(JavaScriptDebugServer, server, ()); return server; } JavaScriptDebugServer::JavaScriptDebugServer() : m_callingListeners(false) , m_pauseOnExceptions(false) , m_pauseOnNextStatement(false) , m_paused(false) , m_doneProcessingDebuggerEvents(true) , m_pauseOnCallFrame(0) , m_recompileTimer(this, &JavaScriptDebugServer::recompileAllJSFunctions) { } JavaScriptDebugServer::~JavaScriptDebugServer() { deleteAllValues(m_pageListenersMap); deleteAllValues(m_breakpoints); } void JavaScriptDebugServer::addListener(JavaScriptDebugListener* listener) { ASSERT_ARG(listener, listener); m_listeners.add(listener); didAddListener(0); } void JavaScriptDebugServer::removeListener(JavaScriptDebugListener* listener) { ASSERT_ARG(listener, listener); m_listeners.remove(listener); didRemoveListener(0); if (!hasListeners()) didRemoveLastListener(); } void JavaScriptDebugServer::addListener(JavaScriptDebugListener* listener, Page* page) { ASSERT_ARG(listener, listener); ASSERT_ARG(page, page); pair result = m_pageListenersMap.add(page, 0); if (result.second) result.first->second = new ListenerSet; ListenerSet* listeners = result.first->second; listeners->add(listener); didAddListener(page); } void JavaScriptDebugServer::removeListener(JavaScriptDebugListener* listener, Page* page) { ASSERT_ARG(listener, listener); ASSERT_ARG(page, page); PageListenersMap::iterator it = m_pageListenersMap.find(page); if (it == m_pageListenersMap.end()) return; ListenerSet* listeners = it->second; listeners->remove(listener); if (listeners->isEmpty()) { m_pageListenersMap.remove(it); delete listeners; } didRemoveListener(page); if (!hasListeners()) didRemoveLastListener(); } void JavaScriptDebugServer::pageCreated(Page* page) { ASSERT_ARG(page, page); if (!hasListenersInterestedInPage(page)) return; page->setDebugger(this); } bool JavaScriptDebugServer::hasListenersInterestedInPage(Page* page) { ASSERT_ARG(page, page); if (hasGlobalListeners()) return true; return m_pageListenersMap.contains(page); } void JavaScriptDebugServer::addBreakpoint(intptr_t sourceID, unsigned lineNumber) { HashSet* lines = m_breakpoints.get(sourceID); if (!lines) { lines = new HashSet; m_breakpoints.set(sourceID, lines); } lines->add(lineNumber); } void JavaScriptDebugServer::removeBreakpoint(intptr_t sourceID, unsigned lineNumber) { HashSet* lines = m_breakpoints.get(sourceID); if (!lines) return; lines->remove(lineNumber); if (!lines->isEmpty()) return; m_breakpoints.remove(sourceID); delete lines; } bool JavaScriptDebugServer::hasBreakpoint(intptr_t sourceID, unsigned lineNumber) const { HashSet* lines = m_breakpoints.get(sourceID); if (!lines) return false; return lines->contains(lineNumber); } void JavaScriptDebugServer::clearBreakpoints() { deleteAllValues(m_breakpoints); m_breakpoints.clear(); } void JavaScriptDebugServer::setPauseOnExceptions(bool pause) { m_pauseOnExceptions = pause; } void JavaScriptDebugServer::pauseProgram() { m_pauseOnNextStatement = true; } void JavaScriptDebugServer::continueProgram() { if (!m_paused) return; m_pauseOnNextStatement = false; m_doneProcessingDebuggerEvents = true; } void JavaScriptDebugServer::stepIntoStatement() { if (!m_paused) return; m_pauseOnNextStatement = true; m_doneProcessingDebuggerEvents = true; } void JavaScriptDebugServer::stepOverStatement() { if (!m_paused) return; m_pauseOnCallFrame = m_currentCallFrame.get(); m_doneProcessingDebuggerEvents = true; } void JavaScriptDebugServer::stepOutOfFunction() { if (!m_paused) return; m_pauseOnCallFrame = m_currentCallFrame ? m_currentCallFrame->caller() : 0; m_doneProcessingDebuggerEvents = true; } JavaScriptCallFrame* JavaScriptDebugServer::currentCallFrame() { if (!m_paused) return 0; return m_currentCallFrame.get(); } static void dispatchDidParseSource(const ListenerSet& listeners, ExecState* exec, const JSC::SourceCode& source) { Vector copy; copyToVector(listeners, copy); for (size_t i = 0; i < copy.size(); ++i) copy[i]->didParseSource(exec, source); } static void dispatchFailedToParseSource(const ListenerSet& listeners, ExecState* exec, const SourceCode& source, int errorLine, const String& errorMessage) { Vector copy; copyToVector(listeners, copy); for (size_t i = 0; i < copy.size(); ++i) copy[i]->failedToParseSource(exec, source, errorLine, errorMessage); } static Page* toPage(JSGlobalObject* globalObject) { ASSERT_ARG(globalObject, globalObject); JSDOMWindow* window = asJSDOMWindow(globalObject); Frame* frame = window->impl()->frame(); return frame ? frame->page() : 0; } void JavaScriptDebugServer::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 JavaScriptDebugServer::sourceParsed(ExecState* exec, const SourceCode& source, int errorLine, const UString& errorMessage) { if (m_callingListeners) return; Page* page = toPage(exec->dynamicGlobalObject()); if (!page) return; m_callingListeners = true; ASSERT(hasListeners()); bool isError = errorLine != -1; if (hasGlobalListeners()) { if (isError) dispatchFailedToParseSource(m_listeners, exec, source, errorLine, errorMessage); else dispatchDidParseSource(m_listeners, exec, source); } if (ListenerSet* pageListeners = m_pageListenersMap.get(page)) { ASSERT(!pageListeners->isEmpty()); if (isError) dispatchFailedToParseSource(*pageListeners, exec, source, errorLine, errorMessage); else dispatchDidParseSource(*pageListeners, exec, source); } m_callingListeners = false; } static void dispatchFunctionToListeners(const ListenerSet& listeners, JavaScriptDebugServer::JavaScriptExecutionCallback callback) { Vector copy; copyToVector(listeners, copy); for (size_t i = 0; i < copy.size(); ++i) (copy[i]->*callback)(); } void JavaScriptDebugServer::dispatchFunctionToListeners(JavaScriptExecutionCallback callback, Page* page) { if (m_callingListeners) return; m_callingListeners = true; ASSERT(hasListeners()); WebCore::dispatchFunctionToListeners(m_listeners, callback); if (ListenerSet* pageListeners = m_pageListenersMap.get(page)) { ASSERT(!pageListeners->isEmpty()); WebCore::dispatchFunctionToListeners(*pageListeners, callback); } m_callingListeners = false; } void JavaScriptDebugServer::setJavaScriptPaused(const PageGroup& pageGroup, bool paused) { setMainThreadCallbacksPaused(paused); const HashSet& pages = pageGroup.pages(); HashSet::const_iterator end = pages.end(); for (HashSet::const_iterator it = pages.begin(); it != end; ++it) setJavaScriptPaused(*it, paused); } void JavaScriptDebugServer::setJavaScriptPaused(Page* page, bool paused) { ASSERT_ARG(page, page); page->setDefersLoading(paused); for (Frame* frame = page->mainFrame(); frame; frame = frame->tree()->traverseNext()) setJavaScriptPaused(frame, paused); } void JavaScriptDebugServer::setJavaScriptPaused(Frame* frame, bool paused) { ASSERT_ARG(frame, frame); if (!frame->script()->isEnabled()) return; frame->script()->setPaused(paused); Document* document = frame->document(); if (paused) document->suspendActiveDOMObjects(); else document->resumeActiveDOMObjects(); setJavaScriptPaused(frame->view(), paused); } #if PLATFORM(MAC) void JavaScriptDebugServer::setJavaScriptPaused(FrameView*, bool) { } #else void JavaScriptDebugServer::setJavaScriptPaused(FrameView* view, bool paused) { if (!view) return; const HashSet >* children = view->children(); ASSERT(children); HashSet >::const_iterator end = children->end(); for (HashSet >::const_iterator it = children->begin(); it != end; ++it) { Widget* widget = (*it).get(); if (!widget->isPluginView()) continue; static_cast(widget)->setJavaScriptPaused(paused); } } #endif void JavaScriptDebugServer::pauseIfNeeded(Page* page) { if (m_paused) return; if (!page || !hasListenersInterestedInPage(page)) return; bool pauseNow = m_pauseOnNextStatement; pauseNow |= (m_pauseOnCallFrame == m_currentCallFrame); pauseNow |= (m_currentCallFrame->line() > 0 && hasBreakpoint(m_currentCallFrame->sourceID(), m_currentCallFrame->line())); if (!pauseNow) return; m_pauseOnCallFrame = 0; m_pauseOnNextStatement = false; m_paused = true; dispatchFunctionToListeners(&JavaScriptDebugListener::didPause, page); setJavaScriptPaused(page->group(), true); TimerBase::fireTimersInNestedEventLoop(); EventLoop loop; m_doneProcessingDebuggerEvents = false; while (!m_doneProcessingDebuggerEvents && !loop.ended()) loop.cycle(); setJavaScriptPaused(page->group(), false); m_paused = false; dispatchFunctionToListeners(&JavaScriptDebugListener::didContinue, page); } void JavaScriptDebugServer::callEvent(const DebuggerCallFrame& debuggerCallFrame, intptr_t sourceID, int lineNumber) { if (m_paused) return; m_currentCallFrame = JavaScriptCallFrame::create(debuggerCallFrame, m_currentCallFrame, sourceID, lineNumber); pauseIfNeeded(toPage(debuggerCallFrame.dynamicGlobalObject())); } void JavaScriptDebugServer::atStatement(const DebuggerCallFrame& debuggerCallFrame, intptr_t sourceID, int lineNumber) { if (m_paused) return; ASSERT(m_currentCallFrame); if (!m_currentCallFrame) return; m_currentCallFrame->update(debuggerCallFrame, sourceID, lineNumber); pauseIfNeeded(toPage(debuggerCallFrame.dynamicGlobalObject())); } void JavaScriptDebugServer::returnEvent(const DebuggerCallFrame& debuggerCallFrame, intptr_t sourceID, int lineNumber) { if (m_paused) return; ASSERT(m_currentCallFrame); if (!m_currentCallFrame) return; m_currentCallFrame->update(debuggerCallFrame, sourceID, lineNumber); pauseIfNeeded(toPage(debuggerCallFrame.dynamicGlobalObject())); // 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 JavaScriptDebugServer::exception(const DebuggerCallFrame& debuggerCallFrame, intptr_t sourceID, int lineNumber) { if (m_paused) return; ASSERT(m_currentCallFrame); if (!m_currentCallFrame) return; if (m_pauseOnExceptions) m_pauseOnNextStatement = true; m_currentCallFrame->update(debuggerCallFrame, sourceID, lineNumber); pauseIfNeeded(toPage(debuggerCallFrame.dynamicGlobalObject())); } void JavaScriptDebugServer::willExecuteProgram(const DebuggerCallFrame& debuggerCallFrame, intptr_t sourceID, int lineNumber) { if (m_paused) return; m_currentCallFrame = JavaScriptCallFrame::create(debuggerCallFrame, m_currentCallFrame, sourceID, lineNumber); pauseIfNeeded(toPage(debuggerCallFrame.dynamicGlobalObject())); } void JavaScriptDebugServer::didExecuteProgram(const DebuggerCallFrame& debuggerCallFrame, intptr_t sourceID, int lineNumber) { if (m_paused) return; ASSERT(m_currentCallFrame); if (!m_currentCallFrame) return; m_currentCallFrame->update(debuggerCallFrame, sourceID, lineNumber); pauseIfNeeded(toPage(debuggerCallFrame.dynamicGlobalObject())); // 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 JavaScriptDebugServer::didReachBreakpoint(const DebuggerCallFrame& debuggerCallFrame, intptr_t sourceID, int lineNumber) { if (m_paused) return; ASSERT(m_currentCallFrame); if (!m_currentCallFrame) return; m_pauseOnNextStatement = true; m_currentCallFrame->update(debuggerCallFrame, sourceID, lineNumber); pauseIfNeeded(toPage(debuggerCallFrame.dynamicGlobalObject())); } void JavaScriptDebugServer::recompileAllJSFunctionsSoon() { m_recompileTimer.startOneShot(0); } void JavaScriptDebugServer::recompileAllJSFunctions(Timer*) { JSLock lock(false); JSGlobalData* globalData = JSDOMWindow::commonJSGlobalData(); // If JavaScript is running, it's not safe to recompile, since we'll end // up throwing away code that is live on the stack. ASSERT(!globalData->dynamicGlobalObject); if (globalData->dynamicGlobalObject) return; Vector > functions; Heap::iterator heapEnd = globalData->heap.primaryHeapEnd(); for (Heap::iterator it = globalData->heap.primaryHeapBegin(); it != heapEnd; ++it) { if ((*it)->isObject(&JSFunction::info)) { JSFunction* function = static_cast(*it); if (!function->isHostFunction()) functions.append(function); } } typedef HashMap, RefPtr > FunctionBodyMap; typedef HashMap SourceProviderMap; FunctionBodyMap functionBodies; SourceProviderMap sourceProviders; size_t size = functions.size(); for (size_t i = 0; i < size; ++i) { JSFunction* function = functions[i]; FunctionBodyNode* oldBody = function->body(); pair result = functionBodies.add(oldBody, 0); if (!result.second) { function->setBody(result.first->second.get()); continue; } ExecState* exec = function->scope().globalObject()->JSGlobalObject::globalExec(); const SourceCode& sourceCode = oldBody->source(); RefPtr newBody = globalData->parser->parse(exec, 0, sourceCode); ASSERT(newBody); newBody->finishParsing(oldBody->copyParameters(), oldBody->parameterCount()); result.first->second = newBody; function->setBody(newBody.release()); if (hasListeners() && function->scope().globalObject()->debugger() == this) sourceProviders.add(sourceCode.provider(), exec); } // Call sourceParsed() after reparsing all functions because it will execute // JavaScript in the inspector. SourceProviderMap::const_iterator end = sourceProviders.end(); for (SourceProviderMap::const_iterator iter = sourceProviders.begin(); iter != end; ++iter) sourceParsed((*iter).second, SourceCode((*iter).first), -1, 0); } void JavaScriptDebugServer::didAddListener(Page* page) { recompileAllJSFunctionsSoon(); if (page) page->setDebugger(this); else Page::setDebuggerForAllPages(this); } void JavaScriptDebugServer::didRemoveListener(Page* page) { if (hasGlobalListeners() || (page && hasListenersInterestedInPage(page))) return; recompileAllJSFunctionsSoon(); if (page) page->setDebugger(0); else Page::setDebuggerForAllPages(0); } void JavaScriptDebugServer::didRemoveLastListener() { m_doneProcessingDebuggerEvents = true; } } // namespace WebCore #endif // ENABLE(JAVASCRIPT_DEBUGGER)