2 * Copyright (C) 2008, 2009 Apple Inc. All rights reserved.
4 * Redistribution and use in source and binary forms, with or without
5 * modification, are permitted provided that the following conditions
7 * 1. Redistributions of source code must retain the above copyright
8 * notice, this list of conditions and the following disclaimer.
9 * 2. Redistributions in binary form must reproduce the above copyright
10 * notice, this list of conditions and the following disclaimer in the
11 * documentation and/or other materials provided with the distribution.
13 * THIS SOFTWARE IS PROVIDED BY APPLE COMPUTER, INC. ``AS IS'' AND ANY
14 * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
15 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
16 * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE COMPUTER, INC. OR
17 * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
18 * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
19 * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
20 * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
21 * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
22 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
23 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27 #include "Structure.h"
29 #include "Identifier.h"
31 #include "JSPropertyNameIterator.h"
33 #include "PropertyNameArray.h"
34 #include "StructureChain.h"
35 #include <wtf/RefCountedLeakCounter.h>
36 #include <wtf/RefPtr.h>
38 #if ENABLE(JSC_MULTIPLE_THREADS)
39 #include <wtf/Threading.h>
42 #define DUMP_STRUCTURE_ID_STATISTICS 0
45 #define DO_PROPERTYMAP_CONSTENCY_CHECK 0
47 #define DO_PROPERTYMAP_CONSTENCY_CHECK 0
53 #if DUMP_PROPERTYMAP_STATS
64 #if DUMP_STRUCTURE_ID_STATISTICS
65 static HashSet
<Structure
*>& liveStructureSet
= *(new HashSet
<Structure
*>);
68 bool StructureTransitionTable::contains(StringImpl
* rep
, unsigned attributes
) const
70 if (isUsingSingleSlot()) {
71 Structure
* transition
= singleTransition();
72 return transition
&& transition
->m_nameInPrevious
== rep
&& transition
->m_attributesInPrevious
== attributes
;
74 return map()->contains(make_pair(rep
, attributes
));
77 inline Structure
* StructureTransitionTable::get(StringImpl
* rep
, unsigned attributes
) const
79 if (isUsingSingleSlot()) {
80 Structure
* transition
= singleTransition();
81 return (transition
&& transition
->m_nameInPrevious
== rep
&& transition
->m_attributesInPrevious
== attributes
) ? transition
: 0;
83 return map()->get(make_pair(rep
, attributes
));
86 inline void StructureTransitionTable::remove(Structure
* structure
)
88 if (isUsingSingleSlot()) {
89 // If more than one transition had been added, then we wouldn't be in
90 // single slot mode (even despecifying a from a specific value triggers
92 // As such, the passed structure *must* be the existing transition.
93 ASSERT(singleTransition() == structure
);
94 clearSingleTransition();
96 // Check whether a mapping exists for structure's key, and whether the
97 // entry is structure (the latter check may fail if we initially had a
98 // transition with a specific value, and this has been despecified).
100 // Newer versions of the STL have an std::make_pair function that takes rvalue references.
101 // When either of the parameters are bitfields, the C++ compiler will try to bind them as lvalues, which is invalid. To work around this, use unary "+" to make the parameter an rvalue.
102 // See https://bugs.webkit.org/show_bug.cgi?id=59261 for more details
103 TransitionMap::iterator entry
= map()->find(make_pair(structure
->m_nameInPrevious
, +structure
->m_attributesInPrevious
));
104 if (entry
!= map()->end() && structure
== entry
.get().second
)
105 map()->remove(entry
);
109 inline void StructureTransitionTable::add(JSGlobalData
& globalData
, Structure
* structure
)
111 if (isUsingSingleSlot()) {
112 Structure
* existingTransition
= singleTransition();
114 // This handles the first transition being added.
115 if (!existingTransition
) {
116 setSingleTransition(globalData
, structure
);
120 // This handles the second transition being added
121 // (or the first transition being despecified!)
122 setMap(new TransitionMap());
123 add(globalData
, existingTransition
);
126 // Add the structure to the map.
128 // Newer versions of the STL have an std::make_pair function that takes rvalue references.
129 // When either of the parameters are bitfields, the C++ compiler will try to bind them as lvalues, which is invalid. To work around this, use unary "+" to make the parameter an rvalue.
130 // See https://bugs.webkit.org/show_bug.cgi?id=59261 for more details
131 std::pair
<TransitionMap::iterator
, bool> result
= map()->add(globalData
, make_pair(structure
->m_nameInPrevious
, +structure
->m_attributesInPrevious
), structure
);
132 if (!result
.second
) {
133 // There already is an entry! - we should only hit this when despecifying.
134 ASSERT(result
.first
.get().second
->m_specificValueInPrevious
);
135 ASSERT(!structure
->m_specificValueInPrevious
);
136 map()->set(result
.first
, structure
);
140 void Structure::dumpStatistics()
142 #if DUMP_STRUCTURE_ID_STATISTICS
143 unsigned numberLeaf
= 0;
144 unsigned numberUsingSingleSlot
= 0;
145 unsigned numberSingletons
= 0;
146 unsigned numberWithPropertyMaps
= 0;
147 unsigned totalPropertyMapsSize
= 0;
149 HashSet
<Structure
*>::const_iterator end
= liveStructureSet
.end();
150 for (HashSet
<Structure
*>::const_iterator it
= liveStructureSet
.begin(); it
!= end
; ++it
) {
151 Structure
* structure
= *it
;
153 switch (structure
->m_transitionTable
.size()) {
156 if (!structure
->m_previous
)
161 ++numberUsingSingleSlot
;
165 if (structure
->m_propertyTable
) {
166 ++numberWithPropertyMaps
;
167 totalPropertyMapsSize
+= structure
->m_propertyTable
->sizeInMemory();
171 printf("Number of live Structures: %d\n", liveStructureSet
.size());
172 printf("Number of Structures using the single item optimization for transition map: %d\n", numberUsingSingleSlot
);
173 printf("Number of Structures that are leaf nodes: %d\n", numberLeaf
);
174 printf("Number of Structures that singletons: %d\n", numberSingletons
);
175 printf("Number of Structures with PropertyMaps: %d\n", numberWithPropertyMaps
);
177 printf("Size of a single Structures: %d\n", static_cast<unsigned>(sizeof(Structure
)));
178 printf("Size of sum of all property maps: %d\n", totalPropertyMapsSize
);
179 printf("Size of average of all property maps: %f\n", static_cast<double>(totalPropertyMapsSize
) / static_cast<double>(liveStructureSet
.size()));
181 printf("Dumping Structure statistics is not enabled.\n");
185 Structure::Structure(JSGlobalData
& globalData
, JSValue prototype
, const TypeInfo
& typeInfo
, unsigned anonymousSlotCount
, const ClassInfo
* classInfo
)
186 : JSCell(globalData
, globalData
.structureStructure
.get())
187 , m_typeInfo(typeInfo
)
188 , m_prototype(globalData
, this, prototype
)
189 , m_classInfo(classInfo
)
190 , m_propertyStorageCapacity(typeInfo
.isFinal() ? JSFinalObject_inlineStorageCapacity
: JSNonFinalObject_inlineStorageCapacity
)
192 , m_dictionaryKind(NoneDictionaryKind
)
193 , m_isPinnedPropertyTable(false)
194 , m_hasGetterSetterProperties(false)
195 , m_hasNonEnumerableProperties(false)
196 , m_attributesInPrevious(0)
197 , m_specificFunctionThrashCount(0)
198 , m_anonymousSlotCount(anonymousSlotCount
)
199 , m_preventExtensions(false)
200 , m_didTransition(false)
203 ASSERT(m_prototype
.isObject() || m_prototype
.isNull());
206 const ClassInfo
Structure::s_info
= { "Structure", 0, 0, 0 };
208 Structure::Structure(JSGlobalData
& globalData
)
209 : JSCell(globalData
, this, CreatingEarlyCell
)
210 , m_typeInfo(CompoundType
, OverridesVisitChildren
)
211 , m_prototype(globalData
, this, jsNull())
212 , m_classInfo(&s_info
)
213 , m_propertyStorageCapacity(0)
215 , m_dictionaryKind(NoneDictionaryKind
)
216 , m_isPinnedPropertyTable(false)
217 , m_hasGetterSetterProperties(false)
218 , m_hasNonEnumerableProperties(false)
219 , m_attributesInPrevious(0)
220 , m_specificFunctionThrashCount(0)
221 , m_anonymousSlotCount(0)
222 , m_preventExtensions(false)
223 , m_didTransition(false)
226 ASSERT(m_prototype
.isNull());
227 ASSERT(!globalData
.structureStructure
);
230 Structure::Structure(JSGlobalData
& globalData
, const Structure
* previous
)
231 : JSCell(globalData
, globalData
.structureStructure
.get())
232 , m_typeInfo(previous
->typeInfo())
233 , m_prototype(globalData
, this, previous
->storedPrototype())
234 , m_classInfo(previous
->m_classInfo
)
235 , m_propertyStorageCapacity(previous
->m_propertyStorageCapacity
)
237 , m_dictionaryKind(NoneDictionaryKind
)
238 , m_isPinnedPropertyTable(false)
239 , m_hasGetterSetterProperties(previous
->m_hasGetterSetterProperties
)
240 , m_hasNonEnumerableProperties(previous
->m_hasNonEnumerableProperties
)
241 , m_attributesInPrevious(0)
242 , m_specificFunctionThrashCount(previous
->m_specificFunctionThrashCount
)
243 , m_anonymousSlotCount(previous
->anonymousSlotCount())
244 , m_preventExtensions(previous
->m_preventExtensions
)
245 , m_didTransition(true)
248 ASSERT(m_prototype
.isObject() || m_prototype
.isNull());
251 Structure::~Structure()
255 void Structure::materializePropertyMap(JSGlobalData
& globalData
)
257 ASSERT(structure()->classInfo() == &s_info
);
258 ASSERT(!m_propertyTable
);
260 Vector
<Structure
*, 8> structures
;
261 structures
.append(this);
263 Structure
* structure
= this;
265 // Search for the last Structure with a property table.
266 while ((structure
= structure
->previousID())) {
267 if (structure
->m_isPinnedPropertyTable
) {
268 ASSERT(structure
->m_propertyTable
);
269 ASSERT(!structure
->m_previous
);
271 m_propertyTable
= structure
->m_propertyTable
->copy(globalData
, 0, m_offset
+ 1);
275 structures
.append(structure
);
278 if (!m_propertyTable
)
279 createPropertyMap(m_offset
+ 1);
281 for (ptrdiff_t i
= structures
.size() - 2; i
>= 0; --i
) {
282 structure
= structures
[i
];
283 PropertyMapEntry
entry(globalData
, this, structure
->m_nameInPrevious
.get(), m_anonymousSlotCount
+ structure
->m_offset
, structure
->m_attributesInPrevious
, structure
->m_specificValueInPrevious
.get());
284 m_propertyTable
->add(entry
);
288 void Structure::growPropertyStorageCapacity()
290 if (isUsingInlineStorage())
291 m_propertyStorageCapacity
= JSObject::baseExternalStorageCapacity
;
293 m_propertyStorageCapacity
*= 2;
296 void Structure::despecifyDictionaryFunction(JSGlobalData
& globalData
, const Identifier
& propertyName
)
298 StringImpl
* rep
= propertyName
.impl();
300 materializePropertyMapIfNecessary(globalData
);
302 ASSERT(isDictionary());
303 ASSERT(m_propertyTable
);
305 PropertyMapEntry
* entry
= m_propertyTable
->find(rep
).first
;
307 entry
->specificValue
.clear();
310 Structure
* Structure::addPropertyTransitionToExistingStructure(Structure
* structure
, const Identifier
& propertyName
, unsigned attributes
, JSCell
* specificValue
, size_t& offset
)
312 ASSERT(!structure
->isDictionary());
313 ASSERT(structure
->typeInfo().type() == ObjectType
);
315 if (Structure
* existingTransition
= structure
->m_transitionTable
.get(propertyName
.impl(), attributes
)) {
316 JSCell
* specificValueInPrevious
= existingTransition
->m_specificValueInPrevious
.get();
317 if (specificValueInPrevious
&& specificValueInPrevious
!= specificValue
)
319 ASSERT(existingTransition
->m_offset
!= noOffset
);
320 offset
= existingTransition
->m_offset
+ existingTransition
->m_anonymousSlotCount
;
321 ASSERT(offset
>= structure
->m_anonymousSlotCount
);
322 ASSERT(structure
->m_anonymousSlotCount
== existingTransition
->m_anonymousSlotCount
);
323 return existingTransition
;
329 Structure
* Structure::addPropertyTransition(JSGlobalData
& globalData
, Structure
* structure
, const Identifier
& propertyName
, unsigned attributes
, JSCell
* specificValue
, size_t& offset
)
331 // If we have a specific function, we may have got to this point if there is
332 // already a transition with the correct property name and attributes, but
333 // specialized to a different function. In this case we just want to give up
334 // and despecialize the transition.
335 // In this case we clear the value of specificFunction which will result
336 // in us adding a non-specific transition, and any subsequent lookup in
337 // Structure::addPropertyTransitionToExistingStructure will just use that.
338 if (specificValue
&& structure
->m_transitionTable
.contains(propertyName
.impl(), attributes
))
341 ASSERT(!structure
->isDictionary());
342 ASSERT(structure
->typeInfo().type() == ObjectType
);
343 ASSERT(!Structure::addPropertyTransitionToExistingStructure(structure
, propertyName
, attributes
, specificValue
, offset
));
345 if (structure
->m_specificFunctionThrashCount
== maxSpecificFunctionThrashCount
)
348 if (structure
->transitionCount() > s_maxTransitionLength
) {
349 Structure
* transition
= toCacheableDictionaryTransition(globalData
, structure
);
350 ASSERT(structure
!= transition
);
351 offset
= transition
->putSpecificValue(globalData
, propertyName
, attributes
, specificValue
);
352 ASSERT(offset
>= structure
->m_anonymousSlotCount
);
353 ASSERT(structure
->m_anonymousSlotCount
== transition
->m_anonymousSlotCount
);
354 if (transition
->propertyStorageSize() > transition
->propertyStorageCapacity())
355 transition
->growPropertyStorageCapacity();
359 Structure
* transition
= create(globalData
, structure
);
361 transition
->m_cachedPrototypeChain
.setMayBeNull(globalData
, transition
, structure
->m_cachedPrototypeChain
.get());
362 transition
->m_previous
.set(globalData
, transition
, structure
);
363 transition
->m_nameInPrevious
= propertyName
.impl();
364 transition
->m_attributesInPrevious
= attributes
;
365 transition
->m_specificValueInPrevious
.setMayBeNull(globalData
, transition
, specificValue
);
367 if (structure
->m_propertyTable
) {
368 if (structure
->m_isPinnedPropertyTable
)
369 transition
->m_propertyTable
= structure
->m_propertyTable
->copy(globalData
, 0, structure
->m_propertyTable
->size() + 1);
371 transition
->m_propertyTable
= structure
->m_propertyTable
.release();
373 if (structure
->m_previous
)
374 transition
->materializePropertyMap(globalData
);
376 transition
->createPropertyMap();
379 offset
= transition
->putSpecificValue(globalData
, propertyName
, attributes
, specificValue
);
380 ASSERT(offset
>= structure
->m_anonymousSlotCount
);
381 ASSERT(structure
->m_anonymousSlotCount
== transition
->m_anonymousSlotCount
);
382 if (transition
->propertyStorageSize() > transition
->propertyStorageCapacity())
383 transition
->growPropertyStorageCapacity();
385 transition
->m_offset
= offset
- structure
->m_anonymousSlotCount
;
386 ASSERT(structure
->anonymousSlotCount() == transition
->anonymousSlotCount());
387 structure
->m_transitionTable
.add(globalData
, transition
);
391 Structure
* Structure::removePropertyTransition(JSGlobalData
& globalData
, Structure
* structure
, const Identifier
& propertyName
, size_t& offset
)
393 ASSERT(!structure
->isUncacheableDictionary());
395 Structure
* transition
= toUncacheableDictionaryTransition(globalData
, structure
);
397 offset
= transition
->remove(propertyName
);
398 ASSERT(offset
>= structure
->m_anonymousSlotCount
);
399 ASSERT(structure
->m_anonymousSlotCount
== transition
->m_anonymousSlotCount
);
404 Structure
* Structure::changePrototypeTransition(JSGlobalData
& globalData
, Structure
* structure
, JSValue prototype
)
406 Structure
* transition
= create(globalData
, structure
);
408 transition
->m_prototype
.set(globalData
, transition
, prototype
);
410 // Don't set m_offset, as one can not transition to this.
412 structure
->materializePropertyMapIfNecessary(globalData
);
413 transition
->m_propertyTable
= structure
->copyPropertyTable(globalData
, transition
);
414 transition
->m_isPinnedPropertyTable
= true;
416 ASSERT(structure
->anonymousSlotCount() == transition
->anonymousSlotCount());
420 Structure
* Structure::despecifyFunctionTransition(JSGlobalData
& globalData
, Structure
* structure
, const Identifier
& replaceFunction
)
422 ASSERT(structure
->m_specificFunctionThrashCount
< maxSpecificFunctionThrashCount
);
423 Structure
* transition
= create(globalData
, structure
);
425 ++transition
->m_specificFunctionThrashCount
;
427 // Don't set m_offset, as one can not transition to this.
429 structure
->materializePropertyMapIfNecessary(globalData
);
430 transition
->m_propertyTable
= structure
->copyPropertyTable(globalData
, transition
);
431 transition
->m_isPinnedPropertyTable
= true;
433 if (transition
->m_specificFunctionThrashCount
== maxSpecificFunctionThrashCount
)
434 transition
->despecifyAllFunctions(globalData
);
436 bool removed
= transition
->despecifyFunction(globalData
, replaceFunction
);
437 ASSERT_UNUSED(removed
, removed
);
440 ASSERT(structure
->anonymousSlotCount() == transition
->anonymousSlotCount());
444 Structure
* Structure::getterSetterTransition(JSGlobalData
& globalData
, Structure
* structure
)
446 Structure
* transition
= create(globalData
, structure
);
448 // Don't set m_offset, as one can not transition to this.
450 structure
->materializePropertyMapIfNecessary(globalData
);
451 transition
->m_propertyTable
= structure
->copyPropertyTable(globalData
, transition
);
452 transition
->m_isPinnedPropertyTable
= true;
454 ASSERT(structure
->anonymousSlotCount() == transition
->anonymousSlotCount());
458 Structure
* Structure::toDictionaryTransition(JSGlobalData
& globalData
, Structure
* structure
, DictionaryKind kind
)
460 ASSERT(!structure
->isUncacheableDictionary());
462 Structure
* transition
= create(globalData
, structure
);
464 structure
->materializePropertyMapIfNecessary(globalData
);
465 transition
->m_propertyTable
= structure
->copyPropertyTable(globalData
, transition
);
466 transition
->m_isPinnedPropertyTable
= true;
467 transition
->m_dictionaryKind
= kind
;
469 ASSERT(structure
->anonymousSlotCount() == transition
->anonymousSlotCount());
473 Structure
* Structure::toCacheableDictionaryTransition(JSGlobalData
& globalData
, Structure
* structure
)
475 return toDictionaryTransition(globalData
, structure
, CachedDictionaryKind
);
478 Structure
* Structure::toUncacheableDictionaryTransition(JSGlobalData
& globalData
, Structure
* structure
)
480 return toDictionaryTransition(globalData
, structure
, UncachedDictionaryKind
);
483 // In future we may want to cache this transition.
484 Structure
* Structure::sealTransition(JSGlobalData
& globalData
, Structure
* structure
)
486 Structure
* transition
= preventExtensionsTransition(globalData
, structure
);
488 if (transition
->m_propertyTable
) {
489 PropertyTable::iterator end
= transition
->m_propertyTable
->end();
490 for (PropertyTable::iterator iter
= transition
->m_propertyTable
->begin(); iter
!= end
; ++iter
)
491 iter
->attributes
|= DontDelete
;
497 // In future we may want to cache this transition.
498 Structure
* Structure::freezeTransition(JSGlobalData
& globalData
, Structure
* structure
)
500 Structure
* transition
= preventExtensionsTransition(globalData
, structure
);
502 if (transition
->m_propertyTable
) {
503 PropertyTable::iterator end
= transition
->m_propertyTable
->end();
504 for (PropertyTable::iterator iter
= transition
->m_propertyTable
->begin(); iter
!= end
; ++iter
)
505 iter
->attributes
|= (DontDelete
| ReadOnly
);
511 // In future we may want to cache this transition.
512 Structure
* Structure::preventExtensionsTransition(JSGlobalData
& globalData
, Structure
* structure
)
514 Structure
* transition
= create(globalData
, structure
);
516 // Don't set m_offset, as one can not transition to this.
518 structure
->materializePropertyMapIfNecessary(globalData
);
519 transition
->m_propertyTable
= structure
->copyPropertyTable(globalData
, transition
);
520 transition
->m_isPinnedPropertyTable
= true;
521 transition
->m_preventExtensions
= true;
523 ASSERT(structure
->anonymousSlotCount() == transition
->anonymousSlotCount());
527 // In future we may want to cache this property.
528 bool Structure::isSealed(JSGlobalData
& globalData
)
533 materializePropertyMapIfNecessary(globalData
);
534 if (!m_propertyTable
)
537 PropertyTable::iterator end
= m_propertyTable
->end();
538 for (PropertyTable::iterator iter
= m_propertyTable
->begin(); iter
!= end
; ++iter
) {
539 if ((iter
->attributes
& DontDelete
) != DontDelete
)
545 // In future we may want to cache this property.
546 bool Structure::isFrozen(JSGlobalData
& globalData
)
551 materializePropertyMapIfNecessary(globalData
);
552 if (!m_propertyTable
)
555 PropertyTable::iterator end
= m_propertyTable
->end();
556 for (PropertyTable::iterator iter
= m_propertyTable
->begin(); iter
!= end
; ++iter
) {
557 if ((iter
->attributes
& (DontDelete
| ReadOnly
)) != (DontDelete
| ReadOnly
))
563 Structure
* Structure::flattenDictionaryStructure(JSGlobalData
& globalData
, JSObject
* object
)
565 ASSERT(isDictionary());
566 if (isUncacheableDictionary()) {
567 ASSERT(m_propertyTable
);
569 unsigned anonymousSlotCount
= m_anonymousSlotCount
;
570 size_t propertyCount
= m_propertyTable
->size();
571 Vector
<JSValue
> values(propertyCount
);
574 PropertyTable::iterator end
= m_propertyTable
->end();
575 for (PropertyTable::iterator iter
= m_propertyTable
->begin(); iter
!= end
; ++iter
, ++i
) {
576 values
[i
] = object
->getDirectOffset(iter
->offset
);
577 // Update property table to have the new property offsets
578 iter
->offset
= anonymousSlotCount
+ i
;
581 // Copy the original property values into their final locations
582 for (unsigned i
= 0; i
< propertyCount
; i
++)
583 object
->putDirectOffset(globalData
, anonymousSlotCount
+ i
, values
[i
]);
585 m_propertyTable
->clearDeletedOffsets();
588 m_dictionaryKind
= NoneDictionaryKind
;
592 size_t Structure::addPropertyWithoutTransition(JSGlobalData
& globalData
, const Identifier
& propertyName
, unsigned attributes
, JSCell
* specificValue
)
594 ASSERT(!m_enumerationCache
);
596 if (m_specificFunctionThrashCount
== maxSpecificFunctionThrashCount
)
599 materializePropertyMapIfNecessary(globalData
);
601 m_isPinnedPropertyTable
= true;
603 size_t offset
= putSpecificValue(globalData
, propertyName
, attributes
, specificValue
);
604 ASSERT(offset
>= m_anonymousSlotCount
);
605 if (propertyStorageSize() > propertyStorageCapacity())
606 growPropertyStorageCapacity();
610 size_t Structure::removePropertyWithoutTransition(JSGlobalData
& globalData
, const Identifier
& propertyName
)
612 ASSERT(isUncacheableDictionary());
613 ASSERT(!m_enumerationCache
);
615 materializePropertyMapIfNecessary(globalData
);
617 m_isPinnedPropertyTable
= true;
618 size_t offset
= remove(propertyName
);
619 ASSERT(offset
>= m_anonymousSlotCount
);
623 #if DUMP_PROPERTYMAP_STATS
625 struct PropertyMapStatisticsExitLogger
{
626 ~PropertyMapStatisticsExitLogger();
629 static PropertyMapStatisticsExitLogger logger
;
631 PropertyMapStatisticsExitLogger::~PropertyMapStatisticsExitLogger()
633 printf("\nJSC::PropertyMap statistics\n\n");
634 printf("%d probes\n", numProbes
);
635 printf("%d collisions (%.1f%%)\n", numCollisions
, 100.0 * numCollisions
/ numProbes
);
636 printf("%d rehashes\n", numRehashes
);
637 printf("%d removes\n", numRemoves
);
642 #if !DO_PROPERTYMAP_CONSTENCY_CHECK
644 inline void Structure::checkConsistency()
650 PassOwnPtr
<PropertyTable
> Structure::copyPropertyTable(JSGlobalData
& globalData
, Structure
* owner
)
652 return adoptPtr(m_propertyTable
? new PropertyTable(globalData
, owner
, *m_propertyTable
) : 0);
655 size_t Structure::get(JSGlobalData
& globalData
, StringImpl
* propertyName
, unsigned& attributes
, JSCell
*& specificValue
)
657 materializePropertyMapIfNecessary(globalData
);
658 if (!m_propertyTable
)
659 return WTF::notFound
;
661 PropertyMapEntry
* entry
= m_propertyTable
->find(propertyName
).first
;
663 return WTF::notFound
;
665 attributes
= entry
->attributes
;
666 specificValue
= entry
->specificValue
.get();
667 ASSERT(entry
->offset
>= m_anonymousSlotCount
);
668 return entry
->offset
;
671 bool Structure::despecifyFunction(JSGlobalData
& globalData
, const Identifier
& propertyName
)
673 materializePropertyMapIfNecessary(globalData
);
674 if (!m_propertyTable
)
677 ASSERT(!propertyName
.isNull());
678 PropertyMapEntry
* entry
= m_propertyTable
->find(propertyName
.impl()).first
;
682 ASSERT(entry
->specificValue
);
683 entry
->specificValue
.clear();
687 void Structure::despecifyAllFunctions(JSGlobalData
& globalData
)
689 materializePropertyMapIfNecessary(globalData
);
690 if (!m_propertyTable
)
693 PropertyTable::iterator end
= m_propertyTable
->end();
694 for (PropertyTable::iterator iter
= m_propertyTable
->begin(); iter
!= end
; ++iter
)
695 iter
->specificValue
.clear();
698 size_t Structure::putSpecificValue(JSGlobalData
& globalData
, const Identifier
& propertyName
, unsigned attributes
, JSCell
* specificValue
)
700 ASSERT(!propertyName
.isNull());
701 ASSERT(get(globalData
, propertyName
) == notFound
);
704 if (attributes
& DontEnum
)
705 m_hasNonEnumerableProperties
= true;
707 StringImpl
* rep
= propertyName
.impl();
709 if (!m_propertyTable
)
714 if (m_propertyTable
->hasDeletedOffset())
715 newOffset
= m_propertyTable
->getDeletedOffset();
717 newOffset
= m_propertyTable
->size() + m_anonymousSlotCount
;
718 ASSERT(newOffset
>= m_anonymousSlotCount
);
720 m_propertyTable
->add(PropertyMapEntry(globalData
, this, rep
, newOffset
, attributes
, specificValue
));
726 size_t Structure::remove(const Identifier
& propertyName
)
728 ASSERT(!propertyName
.isNull());
732 StringImpl
* rep
= propertyName
.impl();
734 if (!m_propertyTable
)
737 PropertyTable::find_iterator position
= m_propertyTable
->find(rep
);
741 size_t offset
= position
.first
->offset
;
742 ASSERT(offset
>= m_anonymousSlotCount
);
744 m_propertyTable
->remove(position
);
745 m_propertyTable
->addDeletedOffset(offset
);
751 void Structure::createPropertyMap(unsigned capacity
)
753 ASSERT(!m_propertyTable
);
756 m_propertyTable
= adoptPtr(new PropertyTable(capacity
));
760 void Structure::getPropertyNames(JSGlobalData
& globalData
, PropertyNameArray
& propertyNames
, EnumerationMode mode
)
762 materializePropertyMapIfNecessary(globalData
);
763 if (!m_propertyTable
)
766 bool knownUnique
= !propertyNames
.size();
768 PropertyTable::iterator end
= m_propertyTable
->end();
769 for (PropertyTable::iterator iter
= m_propertyTable
->begin(); iter
!= end
; ++iter
) {
770 ASSERT(m_hasNonEnumerableProperties
|| !(iter
->attributes
& DontEnum
));
771 if (!(iter
->attributes
& DontEnum
) || (mode
== IncludeDontEnumProperties
)) {
773 propertyNames
.addKnownUnique(iter
->key
);
775 propertyNames
.add(iter
->key
);
780 void Structure::visitChildren(SlotVisitor
& visitor
)
782 ASSERT_GC_OBJECT_INHERITS(this, &s_info
);
783 ASSERT(structure()->typeInfo().overridesVisitChildren());
784 JSCell::visitChildren(visitor
);
786 visitor
.append(&m_prototype
);
787 if (m_cachedPrototypeChain
)
788 visitor
.append(&m_cachedPrototypeChain
);
790 visitor
.append(&m_previous
);
791 if (m_specificValueInPrevious
)
792 visitor
.append(&m_specificValueInPrevious
);
793 if (m_enumerationCache
)
794 visitor
.append(&m_enumerationCache
);
795 if (m_propertyTable
) {
796 PropertyTable::iterator end
= m_propertyTable
->end();
797 for (PropertyTable::iterator ptr
= m_propertyTable
->begin(); ptr
!= end
; ++ptr
) {
798 if (ptr
->specificValue
)
799 visitor
.append(&ptr
->specificValue
);
804 #if DO_PROPERTYMAP_CONSTENCY_CHECK
806 void PropertyTable::checkConsistency()
808 ASSERT(m_indexSize
>= PropertyTable::MinimumTableSize
);
810 ASSERT(m_indexSize
== m_indexMask
+ 1);
811 ASSERT(!(m_indexSize
& m_indexMask
));
813 ASSERT(m_keyCount
<= m_indexSize
/ 2);
814 ASSERT(m_keyCount
+ m_deletedCount
<= m_indexSize
/ 2);
815 ASSERT(m_deletedCount
<= m_indexSize
/ 4);
817 unsigned indexCount
= 0;
818 unsigned deletedIndexCount
= 0;
819 for (unsigned a
= 0; a
!= m_indexSize
; ++a
) {
820 unsigned entryIndex
= m_index
[a
];
821 if (entryIndex
== PropertyTable::EmptyEntryIndex
)
823 if (entryIndex
== deletedEntryIndex()) {
827 ASSERT(entryIndex
< deletedEntryIndex());
828 ASSERT(entryIndex
- 1 <= usedCount());
831 for (unsigned b
= a
+ 1; b
!= m_indexSize
; ++b
)
832 ASSERT(m_index
[b
] != entryIndex
);
834 ASSERT(indexCount
== m_keyCount
);
835 ASSERT(deletedIndexCount
== m_deletedCount
);
837 ASSERT(!table()[deletedEntryIndex() - 1].key
);
839 unsigned nonEmptyEntryCount
= 0;
840 for (unsigned c
= 0; c
< usedCount(); ++c
) {
841 StringImpl
* rep
= table()[c
].key
;
842 if (rep
== PROPERTY_MAP_DELETED_ENTRY_KEY
)
844 ++nonEmptyEntryCount
;
845 unsigned i
= rep
->existingHash();
849 entryIndex
= m_index
[i
& m_indexMask
];
850 ASSERT(entryIndex
!= PropertyTable::EmptyEntryIndex
);
851 if (rep
== table()[entryIndex
- 1].key
)
854 k
= 1 | doubleHash(rep
->existingHash());
857 ASSERT(entryIndex
== c
+ 1);
860 ASSERT(nonEmptyEntryCount
== m_keyCount
);
863 void Structure::checkConsistency()
865 if (!m_propertyTable
)
868 if (!m_hasNonEnumerableProperties
) {
869 PropertyTable::iterator end
= m_propertyTable
->end();
870 for (PropertyTable::iterator iter
= m_propertyTable
->begin(); iter
!= end
; ++iter
) {
871 ASSERT(!(iter
->attributes
& DontEnum
));
872 ASSERT(iter
->offset
>= m_anonymousSlotCount
);
876 m_propertyTable
->checkConsistency();
879 #endif // DO_PROPERTYMAP_CONSTENCY_CHECK