• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2014 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 #include "core/css/invalidation/DescendantInvalidationSet.h"
33 
34 #include "core/css/resolver/StyleResolver.h"
35 #include "core/dom/Element.h"
36 
37 namespace blink {
38 
DescendantInvalidationSet()39 DescendantInvalidationSet::DescendantInvalidationSet()
40     : m_allDescendantsMightBeInvalid(false)
41     , m_customPseudoInvalid(false)
42     , m_treeBoundaryCrossing(false)
43 {
44 }
45 
invalidatesElement(Element & element) const46 bool DescendantInvalidationSet::invalidatesElement(Element& element) const
47 {
48     if (m_allDescendantsMightBeInvalid)
49         return true;
50 
51     if (m_tagNames && m_tagNames->contains(element.tagQName().localName()))
52         return true;
53 
54     if (element.hasID() && m_ids && m_ids->contains(element.idForStyleResolution()))
55         return true;
56 
57     if (element.hasClass() && m_classes) {
58         const SpaceSplitString& classNames = element.classNames();
59         for (WillBeHeapHashSet<AtomicString>::const_iterator it = m_classes->begin(); it != m_classes->end(); ++it) {
60             if (classNames.contains(*it))
61                 return true;
62         }
63     }
64 
65     if (element.hasAttributes() && m_attributes) {
66         for (WillBeHeapHashSet<AtomicString>::const_iterator it = m_attributes->begin(); it != m_attributes->end(); ++it) {
67             if (element.hasAttribute(*it))
68                 return true;
69         }
70     }
71 
72     return false;
73 }
74 
combine(const DescendantInvalidationSet & other)75 void DescendantInvalidationSet::combine(const DescendantInvalidationSet& other)
76 {
77     // No longer bother combining data structures, since the whole subtree is deemed invalid.
78     if (wholeSubtreeInvalid())
79         return;
80 
81     if (other.wholeSubtreeInvalid()) {
82         setWholeSubtreeInvalid();
83         return;
84     }
85 
86     if (other.customPseudoInvalid())
87         setCustomPseudoInvalid();
88 
89     if (other.treeBoundaryCrossing())
90         setTreeBoundaryCrossing();
91 
92     if (other.m_classes) {
93         WillBeHeapHashSet<AtomicString>::const_iterator end = other.m_classes->end();
94         for (WillBeHeapHashSet<AtomicString>::const_iterator it = other.m_classes->begin(); it != end; ++it)
95             addClass(*it);
96     }
97 
98     if (other.m_ids) {
99         WillBeHeapHashSet<AtomicString>::const_iterator end = other.m_ids->end();
100         for (WillBeHeapHashSet<AtomicString>::const_iterator it = other.m_ids->begin(); it != end; ++it)
101             addId(*it);
102     }
103 
104     if (other.m_tagNames) {
105         WillBeHeapHashSet<AtomicString>::const_iterator end = other.m_tagNames->end();
106         for (WillBeHeapHashSet<AtomicString>::const_iterator it = other.m_tagNames->begin(); it != end; ++it)
107             addTagName(*it);
108     }
109 
110     if (other.m_attributes) {
111         WillBeHeapHashSet<AtomicString>::const_iterator end = other.m_attributes->end();
112         for (WillBeHeapHashSet<AtomicString>::const_iterator it = other.m_attributes->begin(); it != end; ++it)
113             addAttribute(*it);
114     }
115 }
116 
ensureClassSet()117 WillBeHeapHashSet<AtomicString>& DescendantInvalidationSet::ensureClassSet()
118 {
119     if (!m_classes)
120         m_classes = adoptPtrWillBeNoop(new WillBeHeapHashSet<AtomicString>);
121     return *m_classes;
122 }
123 
ensureIdSet()124 WillBeHeapHashSet<AtomicString>& DescendantInvalidationSet::ensureIdSet()
125 {
126     if (!m_ids)
127         m_ids = adoptPtrWillBeNoop(new WillBeHeapHashSet<AtomicString>);
128     return *m_ids;
129 }
130 
ensureTagNameSet()131 WillBeHeapHashSet<AtomicString>& DescendantInvalidationSet::ensureTagNameSet()
132 {
133     if (!m_tagNames)
134         m_tagNames = adoptPtrWillBeNoop(new WillBeHeapHashSet<AtomicString>);
135     return *m_tagNames;
136 }
137 
ensureAttributeSet()138 WillBeHeapHashSet<AtomicString>& DescendantInvalidationSet::ensureAttributeSet()
139 {
140     if (!m_attributes)
141         m_attributes = adoptPtrWillBeNoop(new WillBeHeapHashSet<AtomicString>);
142     return *m_attributes;
143 }
144 
addClass(const AtomicString & className)145 void DescendantInvalidationSet::addClass(const AtomicString& className)
146 {
147     if (wholeSubtreeInvalid())
148         return;
149     ensureClassSet().add(className);
150 }
151 
addId(const AtomicString & id)152 void DescendantInvalidationSet::addId(const AtomicString& id)
153 {
154     if (wholeSubtreeInvalid())
155         return;
156     ensureIdSet().add(id);
157 }
158 
addTagName(const AtomicString & tagName)159 void DescendantInvalidationSet::addTagName(const AtomicString& tagName)
160 {
161     if (wholeSubtreeInvalid())
162         return;
163     ensureTagNameSet().add(tagName);
164 }
165 
addAttribute(const AtomicString & attribute)166 void DescendantInvalidationSet::addAttribute(const AtomicString& attribute)
167 {
168     if (wholeSubtreeInvalid())
169         return;
170     ensureAttributeSet().add(attribute);
171 }
172 
setWholeSubtreeInvalid()173 void DescendantInvalidationSet::setWholeSubtreeInvalid()
174 {
175     if (m_allDescendantsMightBeInvalid)
176         return;
177 
178     m_allDescendantsMightBeInvalid = true;
179     m_treeBoundaryCrossing = false;
180     m_classes = nullptr;
181     m_ids = nullptr;
182     m_tagNames = nullptr;
183     m_attributes = nullptr;
184 }
185 
trace(Visitor * visitor)186 void DescendantInvalidationSet::trace(Visitor* visitor)
187 {
188 #if ENABLE(OILPAN)
189     visitor->trace(m_classes);
190     visitor->trace(m_ids);
191     visitor->trace(m_tagNames);
192     visitor->trace(m_attributes);
193 #endif
194 }
195 
196 #ifndef NDEBUG
show() const197 void DescendantInvalidationSet::show() const
198 {
199     fprintf(stderr, "DescendantInvalidationSet { ");
200     if (m_allDescendantsMightBeInvalid)
201         fprintf(stderr, "* ");
202     if (m_customPseudoInvalid)
203         fprintf(stderr, "::custom ");
204     if (m_treeBoundaryCrossing)
205         fprintf(stderr, "::shadow/deep/ ");
206     if (m_ids) {
207         for (WillBeHeapHashSet<AtomicString>::const_iterator it = m_ids->begin(); it != m_ids->end(); ++it)
208             fprintf(stderr, "#%s ", (*it).ascii().data());
209     }
210     if (m_classes) {
211         for (WillBeHeapHashSet<AtomicString>::const_iterator it = m_classes->begin(); it != m_classes->end(); ++it)
212             fprintf(stderr, ".%s ", (*it).ascii().data());
213     }
214     if (m_tagNames) {
215         for (WillBeHeapHashSet<AtomicString>::const_iterator it = m_tagNames->begin(); it != m_tagNames->end(); ++it)
216             fprintf(stderr, "<%s> ", (*it).ascii().data());
217     }
218     if (m_attributes) {
219         for (WillBeHeapHashSet<AtomicString>::const_iterator it = m_attributes->begin(); it != m_attributes->end(); ++it)
220             fprintf(stderr, "[%s] ", (*it).ascii().data());
221     }
222     fprintf(stderr, "}\n");
223 }
224 #endif // NDEBUG
225 
226 } // namespace blink
227