Web Inspector: Remove recompileAllJSFunctions timer in ScriptDebugServer
[WebKit-https.git] / Source / WebCore / bindings / js / PageScriptDebugServer.cpp
1 /*
2  * Copyright (c) 2011 Google Inc. All rights reserved.
3  * Copyright (C) 2013 Apple Inc. All rights reserved.
4  *
5  * Redistribution and use in source and binary forms, with or without
6  * modification, are permitted provided that the following conditions are
7  * met:
8  *
9  *     * Redistributions of source code must retain the above copyright
10  * notice, this list of conditions and the following disclaimer.
11  *     * Redistributions in binary form must reproduce the above
12  * copyright notice, this list of conditions and the following disclaimer
13  * in the documentation and/or other materials provided with the
14  * distribution.
15  *     * Neither the name of Google Inc. nor the names of its
16  * contributors may be used to endorse or promote products derived from
17  * this software without specific prior written permission.
18  *
19  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
20  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
21  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
22  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
23  * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
24  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
25  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
26  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
27  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
28  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
29  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
30  */
31
32 #include "config.h"
33
34 #if ENABLE(JAVASCRIPT_DEBUGGER)
35
36 #include "PageScriptDebugServer.h"
37
38 #include "Document.h"
39 #include "EventLoop.h"
40 #include "FrameView.h"
41 #include "JSDOMWindowCustom.h"
42 #include "MainFrame.h"
43 #include "Page.h"
44 #include "PageGroup.h"
45 #include "PluginView.h"
46 #include "ScriptController.h"
47 #include "ScriptDebugListener.h"
48 #include "Widget.h"
49 #include <runtime/JSLock.h>
50 #include <wtf/MainThread.h>
51 #include <wtf/OwnPtr.h>
52 #include <wtf/PassOwnPtr.h>
53 #include <wtf/StdLibExtras.h>
54
55 #if PLATFORM(IOS)
56 #include "JSDOMWindowBase.h"
57 #include "WebCoreThreadInternal.h"
58 #endif
59
60 using namespace JSC;
61
62 namespace WebCore {
63
64 static Page* toPage(JSGlobalObject* globalObject)
65 {
66     ASSERT_ARG(globalObject, globalObject);
67
68     JSDOMWindow* window = asJSDOMWindow(globalObject);
69     Frame* frame = window->impl().frame();
70     return frame ? frame->page() : 0;
71 }
72
73 PageScriptDebugServer& PageScriptDebugServer::shared()
74 {
75     DEFINE_STATIC_LOCAL(PageScriptDebugServer, server, ());
76     return server;
77 }
78
79 PageScriptDebugServer::PageScriptDebugServer()
80     : ScriptDebugServer()
81     , m_pausedPage(0)
82 {
83 }
84
85 PageScriptDebugServer::~PageScriptDebugServer()
86 {
87 }
88
89 void PageScriptDebugServer::addListener(ScriptDebugListener* listener, Page* page)
90 {
91     ASSERT_ARG(listener, listener);
92     ASSERT_ARG(page, page);
93
94     OwnPtr<ListenerSet>& listeners = m_pageListenersMap.add(page, nullptr).iterator->value;
95     if (!listeners)
96         listeners = adoptPtr(new ListenerSet);
97
98     bool wasEmpty = listeners->isEmpty();
99     listeners->add(listener);
100
101     if (wasEmpty)
102         didAddFirstListener(page);
103 }
104
105 void PageScriptDebugServer::removeListener(ScriptDebugListener* listener, Page* page, bool skipRecompile)
106 {
107     ASSERT_ARG(listener, listener);
108     ASSERT_ARG(page, page);
109
110     PageListenersMap::iterator it = m_pageListenersMap.find(page);
111     if (it == m_pageListenersMap.end())
112         return;
113
114     ListenerSet* listeners = it->value.get();
115     listeners->remove(listener);
116
117     if (listeners->isEmpty()) {
118         m_pageListenersMap.remove(it);
119         didRemoveLastListener(page, skipRecompile);
120     }
121 }
122
123 void PageScriptDebugServer::recompileAllJSFunctions()
124 {
125     JSLockHolder lock(JSDOMWindow::commonVM());
126     Debugger::recompileAllJSFunctions(JSDOMWindow::commonVM());
127 }
128
129 ScriptDebugServer::ListenerSet* PageScriptDebugServer::getListenersForGlobalObject(JSGlobalObject* globalObject)
130 {
131     Page* page = toPage(globalObject);
132     if (!page)
133         return 0;
134     return m_pageListenersMap.get(page);
135 }
136
137 void PageScriptDebugServer::didPause(JSC::JSGlobalObject* globalObject)
138 {
139     ASSERT(!m_pausedPage);
140
141     Page* page = toPage(globalObject);
142     ASSERT(page);
143     if (!page)
144         return;
145
146     m_pausedPage = page;
147
148     setJavaScriptPaused(page->group(), true);
149 }
150
151 void PageScriptDebugServer::didContinue(JSC::JSGlobalObject* globalObject)
152 {
153     // Page can be null if we are continuing because the Page closed.
154     Page* page = toPage(globalObject);
155     ASSERT(!page || page == m_pausedPage);
156
157     m_pausedPage = 0;
158
159     if (page)
160         setJavaScriptPaused(page->group(), false);
161 }
162
163 void PageScriptDebugServer::didAddFirstListener(Page* page)
164 {
165     // Set debugger before recompiling to get sourceParsed callbacks.
166     page->setDebugger(this);
167     recompileAllJSFunctions();
168 }
169
170 void PageScriptDebugServer::didRemoveLastListener(Page* page, bool skipRecompile)
171 {
172     ASSERT(page);
173
174     if (m_pausedPage == page)
175         m_doneProcessingDebuggerEvents = true;
176
177     // Clear debugger before recompiling because we do not need sourceParsed callbacks.
178     page->setDebugger(nullptr);
179
180     if (!skipRecompile)
181         recompileAllJSFunctions();
182 }
183
184 void PageScriptDebugServer::runEventLoopWhilePaused()
185 {
186 #if PLATFORM(IOS)
187     // On iOS, running an EventLoop causes us to run a nested WebRunLoop.
188     // Since the WebThread is autoreleased at the end of run loop iterations
189     // we need to gracefully handle releasing and reacquiring the lock.
190     ASSERT(WebThreadIsLockedOrDisabled());
191     {
192         if (WebThreadIsEnabled())
193             JSC::JSLock::DropAllLocks dropAllLocks(WebCore::JSDOMWindowBase::commonVM(), JSC::JSLock::DropAllLocks::AlwaysDropLocks);
194         WebRunLoopEnableNested();
195 #endif
196
197     EventLoop loop;
198     while (!m_doneProcessingDebuggerEvents && !loop.ended())
199         loop.cycle();
200
201 #if PLATFORM(IOS)
202         WebRunLoopDisableNested();
203     }
204     ASSERT(WebThreadIsLockedOrDisabled());
205 #endif
206 }
207
208 void PageScriptDebugServer::setJavaScriptPaused(const PageGroup& pageGroup, bool paused)
209 {
210     setMainThreadCallbacksPaused(paused);
211
212     const HashSet<Page*>& pages = pageGroup.pages();
213
214     HashSet<Page*>::const_iterator end = pages.end();
215     for (HashSet<Page*>::const_iterator it = pages.begin(); it != end; ++it)
216         setJavaScriptPaused(*it, paused);
217 }
218
219 void PageScriptDebugServer::setJavaScriptPaused(Page* page, bool paused)
220 {
221     ASSERT_ARG(page, page);
222
223     page->setDefersLoading(paused);
224
225     for (Frame* frame = &page->mainFrame(); frame; frame = frame->tree().traverseNext())
226         setJavaScriptPaused(frame, paused);
227 }
228
229 void PageScriptDebugServer::setJavaScriptPaused(Frame* frame, bool paused)
230 {
231     ASSERT_ARG(frame, frame);
232
233     if (!frame->script().canExecuteScripts(NotAboutToExecuteScript))
234         return;
235
236     frame->script().setPaused(paused);
237
238     Document* document = frame->document();
239     if (paused) {
240         document->suspendScriptedAnimationControllerCallbacks();
241         document->suspendActiveDOMObjects(ActiveDOMObject::JavaScriptDebuggerPaused);
242     } else {
243         document->resumeActiveDOMObjects(ActiveDOMObject::JavaScriptDebuggerPaused);
244         document->resumeScriptedAnimationControllerCallbacks();
245     }
246
247     setJavaScriptPaused(frame->view(), paused);
248 }
249
250 void PageScriptDebugServer::setJavaScriptPaused(FrameView* view, bool paused)
251 {
252     if (!view)
253         return;
254
255     for (auto it = view->children().begin(), end = view->children().end(); it != end; ++it) {
256         Widget* widget = (*it).get();
257         if (!widget->isPluginView())
258             continue;
259         toPluginView(widget)->setJavaScriptPaused(paused);
260     }
261 }
262
263 } // namespace WebCore
264
265 #endif // ENABLE(JAVASCRIPT_DEBUGGER)