• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2011 Google Inc. All rights reserved.
3  *
4  * Redistribution and use in source and binary forms, with or without
5  * modification, are permitted provided that the following conditions are
6  * met:
7  *
8  *     * Redistributions of source code must retain the above copyright
9  * notice, this list of conditions and the following disclaimer.
10  *     * Redistributions in binary form must reproduce the above
11  * copyright notice, this list of conditions and the following disclaimer
12  * in the documentation and/or other materials provided with the
13  * distribution.
14  *     * Neither the name of Google Inc. nor the names of its
15  * contributors may be used to endorse or promote products derived from
16  * this software without specific prior written permission.
17  *
18  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
19  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
20  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
21  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
22  * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
23  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
24  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
25  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
26  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
27  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
28  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
29  */
30 
31 #include "config.h"
32 
33 #include "InspectorBrowserDebuggerAgent.h"
34 
35 #if ENABLE(INSPECTOR) && ENABLE(JAVASCRIPT_DEBUGGER)
36 
37 #include "HTMLElement.h"
38 #include "InspectorAgent.h"
39 #include "InspectorDOMAgent.h"
40 #include "InspectorDebuggerAgent.h"
41 #include "InspectorState.h"
42 #include "InspectorValues.h"
43 #include "InstrumentingAgents.h"
44 #include <wtf/text/StringConcatenate.h>
45 
46 namespace {
47 
48 enum DOMBreakpointType {
49     SubtreeModified = 0,
50     AttributeModified,
51     NodeRemoved,
52     DOMBreakpointTypesCount
53 };
54 
55 static const char* const domNativeBreakpointType = "DOM";
56 static const char* const eventListenerNativeBreakpointType = "EventListener";
57 static const char* const xhrNativeBreakpointType = "XHR";
58 
59 const uint32_t inheritableDOMBreakpointTypesMask = (1 << SubtreeModified);
60 const int domBreakpointDerivedTypeShift = 16;
61 
62 }
63 
64 namespace WebCore {
65 
66 namespace BrowserDebuggerAgentState {
67 static const char eventListenerBreakpoints[] = "eventListenerBreakpoints";
68 static const char pauseOnAllXHRs[] = "pauseOnAllXHRs";
69 static const char xhrBreakpoints[] = "xhrBreakpoints";
70 }
71 
create(InstrumentingAgents * instrumentingAgents,InspectorState * inspectorState,InspectorDOMAgent * domAgent,InspectorDebuggerAgent * debuggerAgent,InspectorAgent * inspectorAgent)72 PassOwnPtr<InspectorBrowserDebuggerAgent> InspectorBrowserDebuggerAgent::create(InstrumentingAgents* instrumentingAgents, InspectorState* inspectorState, InspectorDOMAgent* domAgent, InspectorDebuggerAgent* debuggerAgent, InspectorAgent* inspectorAgent)
73 {
74     return adoptPtr(new InspectorBrowserDebuggerAgent(instrumentingAgents, inspectorState, domAgent, debuggerAgent, inspectorAgent));
75 }
76 
InspectorBrowserDebuggerAgent(InstrumentingAgents * instrumentingAgents,InspectorState * inspectorState,InspectorDOMAgent * domAgent,InspectorDebuggerAgent * debuggerAgent,InspectorAgent * inspectorAgent)77 InspectorBrowserDebuggerAgent::InspectorBrowserDebuggerAgent(InstrumentingAgents* instrumentingAgents, InspectorState* inspectorState, InspectorDOMAgent* domAgent, InspectorDebuggerAgent* debuggerAgent, InspectorAgent* inspectorAgent)
78     : m_instrumentingAgents(instrumentingAgents)
79     , m_inspectorState(inspectorState)
80     , m_domAgent(domAgent)
81     , m_debuggerAgent(debuggerAgent)
82     , m_inspectorAgent(inspectorAgent)
83 {
84     m_debuggerAgent->setListener(this);
85 }
86 
~InspectorBrowserDebuggerAgent()87 InspectorBrowserDebuggerAgent::~InspectorBrowserDebuggerAgent()
88 {
89     m_debuggerAgent->setListener(0);
90     ASSERT(!m_instrumentingAgents->inspectorBrowserDebuggerAgent());
91 }
92 
93 // Browser debugger agent enabled only when JS debugger is enabled.
debuggerWasEnabled()94 void InspectorBrowserDebuggerAgent::debuggerWasEnabled()
95 {
96     m_instrumentingAgents->setInspectorBrowserDebuggerAgent(this);
97 }
98 
debuggerWasDisabled()99 void InspectorBrowserDebuggerAgent::debuggerWasDisabled()
100 {
101     disable();
102 }
103 
disable()104 void InspectorBrowserDebuggerAgent::disable()
105 {
106     m_instrumentingAgents->setInspectorBrowserDebuggerAgent(0);
107     clear();
108 }
109 
clearFrontend()110 void InspectorBrowserDebuggerAgent::clearFrontend()
111 {
112     disable();
113 }
114 
discardBindings()115 void InspectorBrowserDebuggerAgent::discardBindings()
116 {
117     m_domBreakpoints.clear();
118 }
119 
setEventListenerBreakpoint(ErrorString * error,const String & eventName)120 void InspectorBrowserDebuggerAgent::setEventListenerBreakpoint(ErrorString* error, const String& eventName)
121 {
122     if (eventName.isEmpty()) {
123         *error = "Event name is empty";
124         return;
125     }
126 
127     RefPtr<InspectorObject> eventListenerBreakpoints = m_inspectorState->getObject(BrowserDebuggerAgentState::eventListenerBreakpoints);
128     eventListenerBreakpoints->setBoolean(eventName, true);
129     m_inspectorState->setObject(BrowserDebuggerAgentState::eventListenerBreakpoints, eventListenerBreakpoints);
130 }
131 
removeEventListenerBreakpoint(ErrorString * error,const String & eventName)132 void InspectorBrowserDebuggerAgent::removeEventListenerBreakpoint(ErrorString* error, const String& eventName)
133 {
134     if (eventName.isEmpty()) {
135         *error = "Event name is empty";
136         return;
137     }
138 
139     RefPtr<InspectorObject> eventListenerBreakpoints = m_inspectorState->getObject(BrowserDebuggerAgentState::eventListenerBreakpoints);
140     eventListenerBreakpoints->remove(eventName);
141     m_inspectorState->setObject(BrowserDebuggerAgentState::eventListenerBreakpoints, eventListenerBreakpoints);
142 }
143 
didInsertDOMNode(Node * node)144 void InspectorBrowserDebuggerAgent::didInsertDOMNode(Node* node)
145 {
146     if (m_domBreakpoints.size()) {
147         uint32_t mask = m_domBreakpoints.get(InspectorDOMAgent::innerParentNode(node));
148         uint32_t inheritableTypesMask = (mask | (mask >> domBreakpointDerivedTypeShift)) & inheritableDOMBreakpointTypesMask;
149         if (inheritableTypesMask)
150             updateSubtreeBreakpoints(node, inheritableTypesMask, true);
151     }
152 }
153 
didRemoveDOMNode(Node * node)154 void InspectorBrowserDebuggerAgent::didRemoveDOMNode(Node* node)
155 {
156     if (m_domBreakpoints.size()) {
157         // Remove subtree breakpoints.
158         m_domBreakpoints.remove(node);
159         Vector<Node*> stack(1, InspectorDOMAgent::innerFirstChild(node));
160         do {
161             Node* node = stack.last();
162             stack.removeLast();
163             if (!node)
164                 continue;
165             m_domBreakpoints.remove(node);
166             stack.append(InspectorDOMAgent::innerFirstChild(node));
167             stack.append(InspectorDOMAgent::innerNextSibling(node));
168         } while (!stack.isEmpty());
169     }
170 }
171 
setDOMBreakpoint(ErrorString *,int nodeId,int type)172 void InspectorBrowserDebuggerAgent::setDOMBreakpoint(ErrorString*, int nodeId, int type)
173 {
174     Node* node = m_domAgent->nodeForId(nodeId);
175     if (!node)
176         return;
177 
178     uint32_t rootBit = 1 << type;
179     m_domBreakpoints.set(node, m_domBreakpoints.get(node) | rootBit);
180     if (rootBit & inheritableDOMBreakpointTypesMask) {
181         for (Node* child = InspectorDOMAgent::innerFirstChild(node); child; child = InspectorDOMAgent::innerNextSibling(child))
182             updateSubtreeBreakpoints(child, rootBit, true);
183     }
184 }
185 
removeDOMBreakpoint(ErrorString *,int nodeId,int type)186 void InspectorBrowserDebuggerAgent::removeDOMBreakpoint(ErrorString*, int nodeId, int type)
187 {
188     Node* node = m_domAgent->nodeForId(nodeId);
189     if (!node)
190         return;
191 
192     uint32_t rootBit = 1 << type;
193     uint32_t mask = m_domBreakpoints.get(node) & ~rootBit;
194     if (mask)
195         m_domBreakpoints.set(node, mask);
196     else
197         m_domBreakpoints.remove(node);
198 
199     if ((rootBit & inheritableDOMBreakpointTypesMask) && !(mask & (rootBit << domBreakpointDerivedTypeShift))) {
200         for (Node* child = InspectorDOMAgent::innerFirstChild(node); child; child = InspectorDOMAgent::innerNextSibling(child))
201             updateSubtreeBreakpoints(child, rootBit, false);
202     }
203 }
204 
willInsertDOMNode(Node *,Node * parent)205 void InspectorBrowserDebuggerAgent::willInsertDOMNode(Node*, Node* parent)
206 {
207     InspectorDebuggerAgent* debuggerAgent = m_debuggerAgent;
208     if (!debuggerAgent)
209         return;
210 
211     if (hasBreakpoint(parent, SubtreeModified)) {
212         RefPtr<InspectorObject> eventData = InspectorObject::create();
213         descriptionForDOMEvent(parent, SubtreeModified, true, eventData.get());
214         eventData->setString("breakpointType", domNativeBreakpointType);
215         debuggerAgent->breakProgram(NativeBreakpointDebuggerEventType, eventData.release());
216     }
217 }
218 
willRemoveDOMNode(Node * node)219 void InspectorBrowserDebuggerAgent::willRemoveDOMNode(Node* node)
220 {
221     InspectorDebuggerAgent* debuggerAgent = m_debuggerAgent;
222     if (!debuggerAgent)
223         return;
224 
225     Node* parentNode = InspectorDOMAgent::innerParentNode(node);
226     if (hasBreakpoint(node, NodeRemoved)) {
227         RefPtr<InspectorObject> eventData = InspectorObject::create();
228         descriptionForDOMEvent(node, NodeRemoved, false, eventData.get());
229         eventData->setString("breakpointType", domNativeBreakpointType);
230         debuggerAgent->breakProgram(NativeBreakpointDebuggerEventType, eventData.release());
231     } else if (parentNode && hasBreakpoint(parentNode, SubtreeModified)) {
232         RefPtr<InspectorObject> eventData = InspectorObject::create();
233         descriptionForDOMEvent(node, SubtreeModified, false, eventData.get());
234         eventData->setString("breakpointType", domNativeBreakpointType);
235         debuggerAgent->breakProgram(NativeBreakpointDebuggerEventType, eventData.release());
236     }
237 }
238 
willModifyDOMAttr(Element * element)239 void InspectorBrowserDebuggerAgent::willModifyDOMAttr(Element* element)
240 {
241     InspectorDebuggerAgent* debuggerAgent = m_debuggerAgent;
242     if (!debuggerAgent)
243         return;
244 
245     if (hasBreakpoint(element, AttributeModified)) {
246         RefPtr<InspectorObject> eventData = InspectorObject::create();
247         descriptionForDOMEvent(element, AttributeModified, false, eventData.get());
248         eventData->setString("breakpointType", domNativeBreakpointType);
249         debuggerAgent->breakProgram(NativeBreakpointDebuggerEventType, eventData.release());
250     }
251 }
252 
descriptionForDOMEvent(Node * target,int breakpointType,bool insertion,InspectorObject * description)253 void InspectorBrowserDebuggerAgent::descriptionForDOMEvent(Node* target, int breakpointType, bool insertion, InspectorObject* description)
254 {
255     ASSERT(hasBreakpoint(target, breakpointType));
256 
257     Node* breakpointOwner = target;
258     if ((1 << breakpointType) & inheritableDOMBreakpointTypesMask) {
259         // For inheritable breakpoint types, target node isn't always the same as the node that owns a breakpoint.
260         // Target node may be unknown to frontend, so we need to push it first.
261         RefPtr<InspectorObject> targetNodeObject = m_domAgent->resolveNode(target);
262         description->setObject("targetNode", targetNodeObject);
263 
264         // Find breakpoint owner node.
265         if (!insertion)
266             breakpointOwner = InspectorDOMAgent::innerParentNode(target);
267         ASSERT(breakpointOwner);
268         while (!(m_domBreakpoints.get(breakpointOwner) & (1 << breakpointType))) {
269             breakpointOwner = InspectorDOMAgent::innerParentNode(breakpointOwner);
270             ASSERT(breakpointOwner);
271         }
272 
273         if (breakpointType == SubtreeModified)
274             description->setBoolean("insertion", insertion);
275     }
276 
277     int breakpointOwnerNodeId = m_domAgent->boundNodeId(breakpointOwner);
278     ASSERT(breakpointOwnerNodeId);
279     description->setNumber("nodeId", breakpointOwnerNodeId);
280     description->setNumber("type", breakpointType);
281 }
282 
hasBreakpoint(Node * node,int type)283 bool InspectorBrowserDebuggerAgent::hasBreakpoint(Node* node, int type)
284 {
285     uint32_t rootBit = 1 << type;
286     uint32_t derivedBit = rootBit << domBreakpointDerivedTypeShift;
287     return m_domBreakpoints.get(node) & (rootBit | derivedBit);
288 }
289 
updateSubtreeBreakpoints(Node * node,uint32_t rootMask,bool set)290 void InspectorBrowserDebuggerAgent::updateSubtreeBreakpoints(Node* node, uint32_t rootMask, bool set)
291 {
292     uint32_t oldMask = m_domBreakpoints.get(node);
293     uint32_t derivedMask = rootMask << domBreakpointDerivedTypeShift;
294     uint32_t newMask = set ? oldMask | derivedMask : oldMask & ~derivedMask;
295     if (newMask)
296         m_domBreakpoints.set(node, newMask);
297     else
298         m_domBreakpoints.remove(node);
299 
300     uint32_t newRootMask = rootMask & ~newMask;
301     if (!newRootMask)
302         return;
303 
304     for (Node* child = InspectorDOMAgent::innerFirstChild(node); child; child = InspectorDOMAgent::innerNextSibling(child))
305         updateSubtreeBreakpoints(child, newRootMask, set);
306 }
307 
pauseOnNativeEventIfNeeded(const String & categoryType,const String & eventName,bool synchronous)308 void InspectorBrowserDebuggerAgent::pauseOnNativeEventIfNeeded(const String& categoryType, const String& eventName, bool synchronous)
309 {
310     InspectorDebuggerAgent* debuggerAgent = m_debuggerAgent;
311     if (!debuggerAgent)
312         return;
313 
314     String fullEventName = makeString(categoryType, ":", eventName);
315     RefPtr<InspectorObject> eventListenerBreakpoints = m_inspectorState->getObject(BrowserDebuggerAgentState::eventListenerBreakpoints);
316     if (eventListenerBreakpoints->find(fullEventName) == eventListenerBreakpoints->end())
317         return;
318 
319     RefPtr<InspectorObject> eventData = InspectorObject::create();
320     eventData->setString("breakpointType", eventListenerNativeBreakpointType);
321     eventData->setString("eventName", fullEventName);
322     if (synchronous)
323         debuggerAgent->breakProgram(NativeBreakpointDebuggerEventType, eventData.release());
324     else
325         debuggerAgent->schedulePauseOnNextStatement(NativeBreakpointDebuggerEventType, eventData.release());
326 }
327 
setXHRBreakpoint(ErrorString *,const String & url)328 void InspectorBrowserDebuggerAgent::setXHRBreakpoint(ErrorString*, const String& url)
329 {
330     if (url.isEmpty()) {
331         m_inspectorState->setBoolean(BrowserDebuggerAgentState::pauseOnAllXHRs, true);
332         return;
333     }
334 
335     RefPtr<InspectorObject> xhrBreakpoints = m_inspectorState->getObject(BrowserDebuggerAgentState::xhrBreakpoints);
336     xhrBreakpoints->setBoolean(url, true);
337     m_inspectorState->setObject(BrowserDebuggerAgentState::xhrBreakpoints, xhrBreakpoints);
338 }
339 
removeXHRBreakpoint(ErrorString *,const String & url)340 void InspectorBrowserDebuggerAgent::removeXHRBreakpoint(ErrorString*, const String& url)
341 {
342     if (url.isEmpty()) {
343         m_inspectorState->setBoolean(BrowserDebuggerAgentState::pauseOnAllXHRs, false);
344         return;
345     }
346 
347     RefPtr<InspectorObject> xhrBreakpoints = m_inspectorState->getObject(BrowserDebuggerAgentState::xhrBreakpoints);
348     xhrBreakpoints->remove(url);
349     m_inspectorState->setObject(BrowserDebuggerAgentState::xhrBreakpoints, xhrBreakpoints);
350 }
351 
willSendXMLHttpRequest(const String & url)352 void InspectorBrowserDebuggerAgent::willSendXMLHttpRequest(const String& url)
353 {
354     InspectorDebuggerAgent* debuggerAgent = m_debuggerAgent;
355     if (!debuggerAgent)
356         return;
357 
358     String breakpointURL;
359     if (m_inspectorState->getBoolean(BrowserDebuggerAgentState::pauseOnAllXHRs))
360         breakpointURL = "";
361     else {
362         RefPtr<InspectorObject> xhrBreakpoints = m_inspectorState->getObject(BrowserDebuggerAgentState::xhrBreakpoints);
363         for (InspectorObject::iterator it = xhrBreakpoints->begin(); it != xhrBreakpoints->end(); ++it) {
364             if (url.contains(it->first)) {
365                 breakpointURL = it->first;
366                 break;
367             }
368         }
369     }
370 
371     if (breakpointURL.isNull())
372         return;
373 
374     RefPtr<InspectorObject> eventData = InspectorObject::create();
375     eventData->setString("breakpointType", xhrNativeBreakpointType);
376     eventData->setString("breakpointURL", breakpointURL);
377     eventData->setString("url", url);
378     debuggerAgent->breakProgram(NativeBreakpointDebuggerEventType, eventData.release());
379 }
380 
clear()381 void InspectorBrowserDebuggerAgent::clear()
382 {
383     m_domBreakpoints.clear();
384 }
385 
386 } // namespace WebCore
387 
388 #endif // ENABLE(INSPECTOR) && ENABLE(JAVASCRIPT_DEBUGGER)
389