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
55 // Choose a number for the following so that most property maps are smaller,
56 // but it's not going to blow out the stack to allocate this number of pointers.
57 static const int smallMapThreshold
= 1024;
59 // The point at which the function call overhead of the qsort implementation
60 // becomes small compared to the inefficiency of insertion sort.
61 static const unsigned tinyMapThreshold
= 20;
63 static const unsigned newTableSize
= 16;
66 static WTF::RefCountedLeakCounter
structureCounter("Structure");
68 #if ENABLE(JSC_MULTIPLE_THREADS)
69 static Mutex
& ignoreSetMutex
= *(new Mutex
);
72 static bool shouldIgnoreLeaks
;
73 static HashSet
<Structure
*>& ignoreSet
= *(new HashSet
<Structure
*>);
76 #if DUMP_STRUCTURE_ID_STATISTICS
77 static HashSet
<Structure
*>& liveStructureSet
= *(new HashSet
<Structure
*>);
80 static int comparePropertyMapEntryIndices(const void* a
, const void* b
);
82 inline void Structure::setTransitionTable(TransitionTable
* table
)
84 ASSERT(m_isUsingSingleSlot
);
86 setSingleTransition(0);
88 m_isUsingSingleSlot
= false;
89 m_transitions
.m_table
= table
;
90 // This implicitly clears the flag that indicates we're using a single transition
91 ASSERT(!m_isUsingSingleSlot
);
94 // The contains and get methods accept imprecise matches, so if an unspecialised transition exists
95 // for the given key they will consider that transition to be a match. If a specialised transition
96 // exists and it matches the provided specificValue, get will return the specific transition.
97 inline bool Structure::transitionTableContains(const StructureTransitionTableHash::Key
& key
, JSCell
* specificValue
)
99 if (m_isUsingSingleSlot
) {
100 Structure
* existingTransition
= singleTransition();
101 return existingTransition
&& existingTransition
->m_nameInPrevious
.get() == key
.first
102 && existingTransition
->m_attributesInPrevious
== key
.second
103 && (existingTransition
->m_specificValueInPrevious
== specificValue
|| existingTransition
->m_specificValueInPrevious
== 0);
105 TransitionTable::iterator find
= transitionTable()->find(key
);
106 if (find
== transitionTable()->end())
109 return find
->second
.first
|| find
->second
.second
->transitionedFor(specificValue
);
112 inline Structure
* Structure::transitionTableGet(const StructureTransitionTableHash::Key
& key
, JSCell
* specificValue
) const
114 if (m_isUsingSingleSlot
) {
115 Structure
* existingTransition
= singleTransition();
116 if (existingTransition
&& existingTransition
->m_nameInPrevious
.get() == key
.first
117 && existingTransition
->m_attributesInPrevious
== key
.second
118 && (existingTransition
->m_specificValueInPrevious
== specificValue
|| existingTransition
->m_specificValueInPrevious
== 0))
119 return existingTransition
;
123 Transition transition
= transitionTable()->get(key
);
124 if (transition
.second
&& transition
.second
->transitionedFor(specificValue
))
125 return transition
.second
;
126 return transition
.first
;
129 inline bool Structure::transitionTableHasTransition(const StructureTransitionTableHash::Key
& key
) const
131 if (m_isUsingSingleSlot
) {
132 Structure
* transition
= singleTransition();
133 return transition
&& transition
->m_nameInPrevious
== key
.first
134 && transition
->m_attributesInPrevious
== key
.second
;
136 return transitionTable()->contains(key
);
139 inline void Structure::transitionTableRemove(const StructureTransitionTableHash::Key
& key
, JSCell
* specificValue
)
141 if (m_isUsingSingleSlot
) {
142 ASSERT(transitionTableContains(key
, specificValue
));
143 setSingleTransition(0);
146 TransitionTable::iterator find
= transitionTable()->find(key
);
148 find
->second
.first
= 0;
150 find
->second
.second
= 0;
151 if (!find
->second
.first
&& !find
->second
.second
)
152 transitionTable()->remove(find
);
155 inline void Structure::transitionTableAdd(const StructureTransitionTableHash::Key
& key
, Structure
* structure
, JSCell
* specificValue
)
157 if (m_isUsingSingleSlot
) {
158 if (!singleTransition()) {
159 setSingleTransition(structure
);
162 Structure
* existingTransition
= singleTransition();
163 TransitionTable
* transitionTable
= new TransitionTable
;
164 setTransitionTable(transitionTable
);
165 if (existingTransition
)
166 transitionTableAdd(std::make_pair(existingTransition
->m_nameInPrevious
.get(), existingTransition
->m_attributesInPrevious
), existingTransition
, existingTransition
->m_specificValueInPrevious
);
168 if (!specificValue
) {
169 TransitionTable::iterator find
= transitionTable()->find(key
);
170 if (find
== transitionTable()->end())
171 transitionTable()->add(key
, Transition(structure
, static_cast<Structure
*>(0)));
173 find
->second
.first
= structure
;
175 // If we're adding a transition to a specific value, then there cannot be
176 // an existing transition
177 ASSERT(!transitionTable()->contains(key
));
178 transitionTable()->add(key
, Transition(static_cast<Structure
*>(0), structure
));
182 void Structure::dumpStatistics()
184 #if DUMP_STRUCTURE_ID_STATISTICS
185 unsigned numberLeaf
= 0;
186 unsigned numberUsingSingleSlot
= 0;
187 unsigned numberSingletons
= 0;
188 unsigned numberWithPropertyMaps
= 0;
189 unsigned totalPropertyMapsSize
= 0;
191 HashSet
<Structure
*>::const_iterator end
= liveStructureSet
.end();
192 for (HashSet
<Structure
*>::const_iterator it
= liveStructureSet
.begin(); it
!= end
; ++it
) {
193 Structure
* structure
= *it
;
194 if (structure
->m_usingSingleTransitionSlot
) {
195 if (!structure
->m_transitions
.singleTransition
)
198 ++numberUsingSingleSlot
;
200 if (!structure
->m_previous
&& !structure
->m_transitions
.singleTransition
)
204 if (structure
->m_propertyTable
) {
205 ++numberWithPropertyMaps
;
206 totalPropertyMapsSize
+= PropertyMapHashTable::allocationSize(structure
->m_propertyTable
->size
);
207 if (structure
->m_propertyTable
->deletedOffsets
)
208 totalPropertyMapsSize
+= (structure
->m_propertyTable
->deletedOffsets
->capacity() * sizeof(unsigned));
212 printf("Number of live Structures: %d\n", liveStructureSet
.size());
213 printf("Number of Structures using the single item optimization for transition map: %d\n", numberUsingSingleSlot
);
214 printf("Number of Structures that are leaf nodes: %d\n", numberLeaf
);
215 printf("Number of Structures that singletons: %d\n", numberSingletons
);
216 printf("Number of Structures with PropertyMaps: %d\n", numberWithPropertyMaps
);
218 printf("Size of a single Structures: %d\n", static_cast<unsigned>(sizeof(Structure
)));
219 printf("Size of sum of all property maps: %d\n", totalPropertyMapsSize
);
220 printf("Size of average of all property maps: %f\n", static_cast<double>(totalPropertyMapsSize
) / static_cast<double>(liveStructureSet
.size()));
222 printf("Dumping Structure statistics is not enabled.\n");
226 Structure::Structure(JSValue prototype
, const TypeInfo
& typeInfo
, unsigned anonymousSlotCount
)
227 : m_typeInfo(typeInfo
)
228 , m_prototype(prototype
)
229 , m_specificValueInPrevious(0)
231 , m_propertyStorageCapacity(JSObject::inlineStorageCapacity
)
233 , m_dictionaryKind(NoneDictionaryKind
)
234 , m_isPinnedPropertyTable(false)
235 , m_hasGetterSetterProperties(false)
236 , m_attributesInPrevious(0)
237 , m_specificFunctionThrashCount(0)
238 , m_anonymousSlotCount(anonymousSlotCount
)
239 , m_isUsingSingleSlot(true)
241 m_transitions
.m_singleTransition
= 0;
244 ASSERT(m_prototype
.isObject() || m_prototype
.isNull());
247 #if ENABLE(JSC_MULTIPLE_THREADS)
248 MutexLocker
protect(ignoreSetMutex
);
250 if (shouldIgnoreLeaks
)
253 structureCounter
.increment();
256 #if DUMP_STRUCTURE_ID_STATISTICS
257 liveStructureSet
.add(this);
261 Structure::~Structure()
264 ASSERT(m_nameInPrevious
);
265 m_previous
->transitionTableRemove(make_pair(m_nameInPrevious
.get(), m_attributesInPrevious
), m_specificValueInPrevious
);
268 ASSERT(!m_enumerationCache
.hasDeadObject());
270 if (m_propertyTable
) {
271 unsigned entryCount
= m_propertyTable
->keyCount
+ m_propertyTable
->deletedSentinelCount
;
272 for (unsigned i
= 1; i
<= entryCount
; i
++) {
273 if (UString::Rep
* key
= m_propertyTable
->entries()[i
].key
)
277 delete m_propertyTable
->deletedOffsets
;
278 fastFree(m_propertyTable
);
281 if (!m_isUsingSingleSlot
)
282 delete transitionTable();
285 #if ENABLE(JSC_MULTIPLE_THREADS)
286 MutexLocker
protect(ignoreSetMutex
);
288 HashSet
<Structure
*>::iterator it
= ignoreSet
.find(this);
289 if (it
!= ignoreSet
.end())
290 ignoreSet
.remove(it
);
292 structureCounter
.decrement();
295 #if DUMP_STRUCTURE_ID_STATISTICS
296 liveStructureSet
.remove(this);
300 void Structure::startIgnoringLeaks()
303 shouldIgnoreLeaks
= true;
307 void Structure::stopIgnoringLeaks()
310 shouldIgnoreLeaks
= false;
314 static bool isPowerOf2(unsigned v
)
316 // Taken from http://www.cs.utk.edu/~vose/c-stuff/bithacks.html
318 return !(v
& (v
- 1)) && v
;
321 static unsigned nextPowerOf2(unsigned v
)
323 // Taken from http://www.cs.utk.edu/~vose/c-stuff/bithacks.html
324 // Devised by Sean Anderson, Sepember 14, 2001
337 static unsigned sizeForKeyCount(size_t keyCount
)
339 if (keyCount
== notFound
)
345 if (isPowerOf2(keyCount
))
348 return nextPowerOf2(keyCount
) * 2;
351 void Structure::materializePropertyMap()
353 ASSERT(!m_propertyTable
);
355 Vector
<Structure
*, 8> structures
;
356 structures
.append(this);
358 Structure
* structure
= this;
360 // Search for the last Structure with a property table.
361 while ((structure
= structure
->previousID())) {
362 if (structure
->m_isPinnedPropertyTable
) {
363 ASSERT(structure
->m_propertyTable
);
364 ASSERT(!structure
->m_previous
);
366 m_propertyTable
= structure
->copyPropertyTable();
370 structures
.append(structure
);
373 if (!m_propertyTable
)
374 createPropertyMapHashTable(sizeForKeyCount(m_offset
+ 1));
376 if (sizeForKeyCount(m_offset
+ 1) > m_propertyTable
->size
)
377 rehashPropertyMapHashTable(sizeForKeyCount(m_offset
+ 1)); // This could be made more efficient by combining with the copy above.
380 for (ptrdiff_t i
= structures
.size() - 2; i
>= 0; --i
) {
381 structure
= structures
[i
];
382 structure
->m_nameInPrevious
->ref();
383 PropertyMapEntry
entry(structure
->m_nameInPrevious
.get(), m_anonymousSlotCount
+ structure
->m_offset
, structure
->m_attributesInPrevious
, structure
->m_specificValueInPrevious
, ++m_propertyTable
->lastIndexUsed
);
384 insertIntoPropertyMapHashTable(entry
);
388 void Structure::growPropertyStorageCapacity()
390 if (m_propertyStorageCapacity
== JSObject::inlineStorageCapacity
)
391 m_propertyStorageCapacity
= JSObject::nonInlineBaseStorageCapacity
;
393 m_propertyStorageCapacity
*= 2;
396 void Structure::despecifyDictionaryFunction(const Identifier
& propertyName
)
398 const UString::Rep
* rep
= propertyName
._ustring
.rep();
400 materializePropertyMapIfNecessary();
402 ASSERT(isDictionary());
403 ASSERT(m_propertyTable
);
405 unsigned i
= rep
->existingHash();
407 #if DUMP_PROPERTYMAP_STATS
411 unsigned entryIndex
= m_propertyTable
->entryIndices
[i
& m_propertyTable
->sizeMask
];
412 ASSERT(entryIndex
!= emptyEntryIndex
);
414 if (rep
== m_propertyTable
->entries()[entryIndex
- 1].key
) {
415 m_propertyTable
->entries()[entryIndex
- 1].specificValue
= 0;
419 #if DUMP_PROPERTYMAP_STATS
423 unsigned k
= 1 | doubleHash(rep
->existingHash());
428 #if DUMP_PROPERTYMAP_STATS
432 entryIndex
= m_propertyTable
->entryIndices
[i
& m_propertyTable
->sizeMask
];
433 ASSERT(entryIndex
!= emptyEntryIndex
);
435 if (rep
== m_propertyTable
->entries()[entryIndex
- 1].key
) {
436 m_propertyTable
->entries()[entryIndex
- 1].specificValue
= 0;
442 PassRefPtr
<Structure
> Structure::addPropertyTransitionToExistingStructure(Structure
* structure
, const Identifier
& propertyName
, unsigned attributes
, JSCell
* specificValue
, size_t& offset
)
444 ASSERT(!structure
->isDictionary());
445 ASSERT(structure
->typeInfo().type() == ObjectType
);
447 if (Structure
* existingTransition
= structure
->transitionTableGet(make_pair(propertyName
.ustring().rep(), attributes
), specificValue
)) {
448 ASSERT(existingTransition
->m_offset
!= noOffset
);
449 offset
= existingTransition
->m_offset
+ existingTransition
->m_anonymousSlotCount
;
450 ASSERT(offset
>= structure
->m_anonymousSlotCount
);
451 ASSERT(structure
->m_anonymousSlotCount
== existingTransition
->m_anonymousSlotCount
);
452 return existingTransition
;
458 PassRefPtr
<Structure
> Structure::addPropertyTransition(Structure
* structure
, const Identifier
& propertyName
, unsigned attributes
, JSCell
* specificValue
, size_t& offset
)
460 ASSERT(!structure
->isDictionary());
461 ASSERT(structure
->typeInfo().type() == ObjectType
);
462 ASSERT(!Structure::addPropertyTransitionToExistingStructure(structure
, propertyName
, attributes
, specificValue
, offset
));
464 if (structure
->m_specificFunctionThrashCount
== maxSpecificFunctionThrashCount
)
467 if (structure
->transitionCount() > s_maxTransitionLength
) {
468 RefPtr
<Structure
> transition
= toCacheableDictionaryTransition(structure
);
469 ASSERT(structure
!= transition
);
470 offset
= transition
->put(propertyName
, attributes
, specificValue
);
471 ASSERT(offset
>= structure
->m_anonymousSlotCount
);
472 ASSERT(structure
->m_anonymousSlotCount
== transition
->m_anonymousSlotCount
);
473 if (transition
->propertyStorageSize() > transition
->propertyStorageCapacity())
474 transition
->growPropertyStorageCapacity();
475 return transition
.release();
478 RefPtr
<Structure
> transition
= create(structure
->m_prototype
, structure
->typeInfo(), structure
->anonymousSlotCount());
480 transition
->m_cachedPrototypeChain
= structure
->m_cachedPrototypeChain
;
481 transition
->m_previous
= structure
;
482 transition
->m_nameInPrevious
= propertyName
.ustring().rep();
483 transition
->m_attributesInPrevious
= attributes
;
484 transition
->m_specificValueInPrevious
= specificValue
;
485 transition
->m_propertyStorageCapacity
= structure
->m_propertyStorageCapacity
;
486 transition
->m_hasGetterSetterProperties
= structure
->m_hasGetterSetterProperties
;
487 transition
->m_hasNonEnumerableProperties
= structure
->m_hasNonEnumerableProperties
;
488 transition
->m_specificFunctionThrashCount
= structure
->m_specificFunctionThrashCount
;
490 if (structure
->m_propertyTable
) {
491 if (structure
->m_isPinnedPropertyTable
)
492 transition
->m_propertyTable
= structure
->copyPropertyTable();
494 transition
->m_propertyTable
= structure
->m_propertyTable
;
495 structure
->m_propertyTable
= 0;
498 if (structure
->m_previous
)
499 transition
->materializePropertyMap();
501 transition
->createPropertyMapHashTable();
504 offset
= transition
->put(propertyName
, attributes
, specificValue
);
505 ASSERT(offset
>= structure
->m_anonymousSlotCount
);
506 ASSERT(structure
->m_anonymousSlotCount
== transition
->m_anonymousSlotCount
);
507 if (transition
->propertyStorageSize() > transition
->propertyStorageCapacity())
508 transition
->growPropertyStorageCapacity();
510 transition
->m_offset
= offset
- structure
->m_anonymousSlotCount
;
511 ASSERT(structure
->anonymousSlotCount() == transition
->anonymousSlotCount());
512 structure
->transitionTableAdd(make_pair(propertyName
.ustring().rep(), attributes
), transition
.get(), specificValue
);
513 return transition
.release();
516 PassRefPtr
<Structure
> Structure::removePropertyTransition(Structure
* structure
, const Identifier
& propertyName
, size_t& offset
)
518 ASSERT(!structure
->isUncacheableDictionary());
520 RefPtr
<Structure
> transition
= toUncacheableDictionaryTransition(structure
);
522 offset
= transition
->remove(propertyName
);
523 ASSERT(offset
>= structure
->m_anonymousSlotCount
);
524 ASSERT(structure
->m_anonymousSlotCount
== transition
->m_anonymousSlotCount
);
526 return transition
.release();
529 PassRefPtr
<Structure
> Structure::changePrototypeTransition(Structure
* structure
, JSValue prototype
)
531 RefPtr
<Structure
> transition
= create(prototype
, structure
->typeInfo(), structure
->anonymousSlotCount());
533 transition
->m_propertyStorageCapacity
= structure
->m_propertyStorageCapacity
;
534 transition
->m_hasGetterSetterProperties
= structure
->m_hasGetterSetterProperties
;
535 transition
->m_hasNonEnumerableProperties
= structure
->m_hasNonEnumerableProperties
;
536 transition
->m_specificFunctionThrashCount
= structure
->m_specificFunctionThrashCount
;
538 // Don't set m_offset, as one can not transition to this.
540 structure
->materializePropertyMapIfNecessary();
541 transition
->m_propertyTable
= structure
->copyPropertyTable();
542 transition
->m_isPinnedPropertyTable
= true;
544 ASSERT(structure
->anonymousSlotCount() == transition
->anonymousSlotCount());
545 return transition
.release();
548 PassRefPtr
<Structure
> Structure::despecifyFunctionTransition(Structure
* structure
, const Identifier
& replaceFunction
)
550 ASSERT(structure
->m_specificFunctionThrashCount
< maxSpecificFunctionThrashCount
);
551 RefPtr
<Structure
> transition
= create(structure
->storedPrototype(), structure
->typeInfo(), structure
->anonymousSlotCount());
553 transition
->m_propertyStorageCapacity
= structure
->m_propertyStorageCapacity
;
554 transition
->m_hasGetterSetterProperties
= structure
->m_hasGetterSetterProperties
;
555 transition
->m_hasNonEnumerableProperties
= structure
->m_hasNonEnumerableProperties
;
556 transition
->m_specificFunctionThrashCount
= structure
->m_specificFunctionThrashCount
+ 1;
558 // Don't set m_offset, as one can not transition to this.
560 structure
->materializePropertyMapIfNecessary();
561 transition
->m_propertyTable
= structure
->copyPropertyTable();
562 transition
->m_isPinnedPropertyTable
= true;
564 if (transition
->m_specificFunctionThrashCount
== maxSpecificFunctionThrashCount
)
565 transition
->despecifyAllFunctions();
567 bool removed
= transition
->despecifyFunction(replaceFunction
);
568 ASSERT_UNUSED(removed
, removed
);
571 ASSERT(structure
->anonymousSlotCount() == transition
->anonymousSlotCount());
572 return transition
.release();
575 PassRefPtr
<Structure
> Structure::getterSetterTransition(Structure
* structure
)
577 RefPtr
<Structure
> transition
= create(structure
->storedPrototype(), structure
->typeInfo(), structure
->anonymousSlotCount());
578 transition
->m_propertyStorageCapacity
= structure
->m_propertyStorageCapacity
;
579 transition
->m_hasGetterSetterProperties
= transition
->m_hasGetterSetterProperties
;
580 transition
->m_hasNonEnumerableProperties
= structure
->m_hasNonEnumerableProperties
;
581 transition
->m_specificFunctionThrashCount
= structure
->m_specificFunctionThrashCount
;
583 // Don't set m_offset, as one can not transition to this.
585 structure
->materializePropertyMapIfNecessary();
586 transition
->m_propertyTable
= structure
->copyPropertyTable();
587 transition
->m_isPinnedPropertyTable
= true;
589 ASSERT(structure
->anonymousSlotCount() == transition
->anonymousSlotCount());
590 return transition
.release();
593 PassRefPtr
<Structure
> Structure::toDictionaryTransition(Structure
* structure
, DictionaryKind kind
)
595 ASSERT(!structure
->isUncacheableDictionary());
597 RefPtr
<Structure
> transition
= create(structure
->m_prototype
, structure
->typeInfo(), structure
->anonymousSlotCount());
598 transition
->m_dictionaryKind
= kind
;
599 transition
->m_propertyStorageCapacity
= structure
->m_propertyStorageCapacity
;
600 transition
->m_hasGetterSetterProperties
= structure
->m_hasGetterSetterProperties
;
601 transition
->m_hasNonEnumerableProperties
= structure
->m_hasNonEnumerableProperties
;
602 transition
->m_specificFunctionThrashCount
= structure
->m_specificFunctionThrashCount
;
604 structure
->materializePropertyMapIfNecessary();
605 transition
->m_propertyTable
= structure
->copyPropertyTable();
606 transition
->m_isPinnedPropertyTable
= true;
608 ASSERT(structure
->anonymousSlotCount() == transition
->anonymousSlotCount());
609 return transition
.release();
612 PassRefPtr
<Structure
> Structure::toCacheableDictionaryTransition(Structure
* structure
)
614 return toDictionaryTransition(structure
, CachedDictionaryKind
);
617 PassRefPtr
<Structure
> Structure::toUncacheableDictionaryTransition(Structure
* structure
)
619 return toDictionaryTransition(structure
, UncachedDictionaryKind
);
622 PassRefPtr
<Structure
> Structure::flattenDictionaryStructure(JSObject
* object
)
624 ASSERT(isDictionary());
625 if (isUncacheableDictionary()) {
626 ASSERT(m_propertyTable
);
627 Vector
<PropertyMapEntry
*> sortedPropertyEntries(m_propertyTable
->keyCount
);
628 PropertyMapEntry
** p
= sortedPropertyEntries
.data();
629 unsigned entryCount
= m_propertyTable
->keyCount
+ m_propertyTable
->deletedSentinelCount
;
630 for (unsigned i
= 1; i
<= entryCount
; i
++) {
631 if (m_propertyTable
->entries()[i
].key
)
632 *p
++ = &m_propertyTable
->entries()[i
];
634 size_t propertyCount
= p
- sortedPropertyEntries
.data();
635 qsort(sortedPropertyEntries
.data(), propertyCount
, sizeof(PropertyMapEntry
*), comparePropertyMapEntryIndices
);
636 sortedPropertyEntries
.resize(propertyCount
);
638 // We now have the properties currently defined on this object
639 // in the order that they are expected to be in, but we need to
640 // reorder the storage, so we have to copy the current values out
641 Vector
<JSValue
> values(propertyCount
);
642 unsigned anonymousSlotCount
= m_anonymousSlotCount
;
643 for (unsigned i
= 0; i
< propertyCount
; i
++) {
644 PropertyMapEntry
* entry
= sortedPropertyEntries
[i
];
645 values
[i
] = object
->getDirectOffset(entry
->offset
);
646 // Update property table to have the new property offsets
647 entry
->offset
= anonymousSlotCount
+ i
;
651 // Copy the original property values into their final locations
652 for (unsigned i
= 0; i
< propertyCount
; i
++)
653 object
->putDirectOffset(anonymousSlotCount
+ i
, values
[i
]);
655 if (m_propertyTable
->deletedOffsets
) {
656 delete m_propertyTable
->deletedOffsets
;
657 m_propertyTable
->deletedOffsets
= 0;
661 m_dictionaryKind
= NoneDictionaryKind
;
665 size_t Structure::addPropertyWithoutTransition(const Identifier
& propertyName
, unsigned attributes
, JSCell
* specificValue
)
667 ASSERT(!m_enumerationCache
);
669 if (m_specificFunctionThrashCount
== maxSpecificFunctionThrashCount
)
672 materializePropertyMapIfNecessary();
674 m_isPinnedPropertyTable
= true;
676 size_t offset
= put(propertyName
, attributes
, specificValue
);
677 ASSERT(offset
>= m_anonymousSlotCount
);
678 if (propertyStorageSize() > propertyStorageCapacity())
679 growPropertyStorageCapacity();
683 size_t Structure::removePropertyWithoutTransition(const Identifier
& propertyName
)
685 ASSERT(isUncacheableDictionary());
686 ASSERT(!m_enumerationCache
);
688 materializePropertyMapIfNecessary();
690 m_isPinnedPropertyTable
= true;
691 size_t offset
= remove(propertyName
);
692 ASSERT(offset
>= m_anonymousSlotCount
);
696 #if DUMP_PROPERTYMAP_STATS
698 static int numProbes
;
699 static int numCollisions
;
700 static int numRehashes
;
701 static int numRemoves
;
703 struct PropertyMapStatisticsExitLogger
{
704 ~PropertyMapStatisticsExitLogger();
707 static PropertyMapStatisticsExitLogger logger
;
709 PropertyMapStatisticsExitLogger::~PropertyMapStatisticsExitLogger()
711 printf("\nJSC::PropertyMap statistics\n\n");
712 printf("%d probes\n", numProbes
);
713 printf("%d collisions (%.1f%%)\n", numCollisions
, 100.0 * numCollisions
/ numProbes
);
714 printf("%d rehashes\n", numRehashes
);
715 printf("%d removes\n", numRemoves
);
720 static const unsigned deletedSentinelIndex
= 1;
722 #if !DO_PROPERTYMAP_CONSTENCY_CHECK
724 inline void Structure::checkConsistency()
730 PropertyMapHashTable
* Structure::copyPropertyTable()
732 if (!m_propertyTable
)
735 size_t tableSize
= PropertyMapHashTable::allocationSize(m_propertyTable
->size
);
736 PropertyMapHashTable
* newTable
= static_cast<PropertyMapHashTable
*>(fastMalloc(tableSize
));
737 memcpy(newTable
, m_propertyTable
, tableSize
);
739 unsigned entryCount
= m_propertyTable
->keyCount
+ m_propertyTable
->deletedSentinelCount
;
740 for (unsigned i
= 1; i
<= entryCount
; ++i
) {
741 if (UString::Rep
* key
= newTable
->entries()[i
].key
)
745 // Copy the deletedOffsets vector.
746 if (m_propertyTable
->deletedOffsets
)
747 newTable
->deletedOffsets
= new Vector
<unsigned>(*m_propertyTable
->deletedOffsets
);
752 size_t Structure::get(const UString::Rep
* rep
, unsigned& attributes
, JSCell
*& specificValue
)
754 materializePropertyMapIfNecessary();
755 if (!m_propertyTable
)
758 unsigned i
= rep
->existingHash();
760 #if DUMP_PROPERTYMAP_STATS
764 unsigned entryIndex
= m_propertyTable
->entryIndices
[i
& m_propertyTable
->sizeMask
];
765 if (entryIndex
== emptyEntryIndex
)
768 if (rep
== m_propertyTable
->entries()[entryIndex
- 1].key
) {
769 attributes
= m_propertyTable
->entries()[entryIndex
- 1].attributes
;
770 specificValue
= m_propertyTable
->entries()[entryIndex
- 1].specificValue
;
771 ASSERT(m_propertyTable
->entries()[entryIndex
- 1].offset
>= m_anonymousSlotCount
);
772 return m_propertyTable
->entries()[entryIndex
- 1].offset
;
775 #if DUMP_PROPERTYMAP_STATS
779 unsigned k
= 1 | doubleHash(rep
->existingHash());
784 #if DUMP_PROPERTYMAP_STATS
788 entryIndex
= m_propertyTable
->entryIndices
[i
& m_propertyTable
->sizeMask
];
789 if (entryIndex
== emptyEntryIndex
)
792 if (rep
== m_propertyTable
->entries()[entryIndex
- 1].key
) {
793 attributes
= m_propertyTable
->entries()[entryIndex
- 1].attributes
;
794 specificValue
= m_propertyTable
->entries()[entryIndex
- 1].specificValue
;
795 ASSERT(m_propertyTable
->entries()[entryIndex
- 1].offset
>= m_anonymousSlotCount
);
796 return m_propertyTable
->entries()[entryIndex
- 1].offset
;
801 bool Structure::despecifyFunction(const Identifier
& propertyName
)
803 ASSERT(!propertyName
.isNull());
805 materializePropertyMapIfNecessary();
806 if (!m_propertyTable
)
809 UString::Rep
* rep
= propertyName
._ustring
.rep();
811 unsigned i
= rep
->existingHash();
813 #if DUMP_PROPERTYMAP_STATS
817 unsigned entryIndex
= m_propertyTable
->entryIndices
[i
& m_propertyTable
->sizeMask
];
818 if (entryIndex
== emptyEntryIndex
)
821 if (rep
== m_propertyTable
->entries()[entryIndex
- 1].key
) {
822 ASSERT(m_propertyTable
->entries()[entryIndex
- 1].specificValue
);
823 m_propertyTable
->entries()[entryIndex
- 1].specificValue
= 0;
827 #if DUMP_PROPERTYMAP_STATS
831 unsigned k
= 1 | doubleHash(rep
->existingHash());
836 #if DUMP_PROPERTYMAP_STATS
840 entryIndex
= m_propertyTable
->entryIndices
[i
& m_propertyTable
->sizeMask
];
841 if (entryIndex
== emptyEntryIndex
)
844 if (rep
== m_propertyTable
->entries()[entryIndex
- 1].key
) {
845 ASSERT(m_propertyTable
->entries()[entryIndex
- 1].specificValue
);
846 m_propertyTable
->entries()[entryIndex
- 1].specificValue
= 0;
852 void Structure::despecifyAllFunctions()
854 materializePropertyMapIfNecessary();
855 if (!m_propertyTable
)
858 unsigned entryCount
= m_propertyTable
->keyCount
+ m_propertyTable
->deletedSentinelCount
;
859 for (unsigned i
= 1; i
<= entryCount
; ++i
)
860 m_propertyTable
->entries()[i
].specificValue
= 0;
863 size_t Structure::put(const Identifier
& propertyName
, unsigned attributes
, JSCell
* specificValue
)
865 ASSERT(!propertyName
.isNull());
866 ASSERT(get(propertyName
) == notFound
);
870 if (attributes
& DontEnum
)
871 m_hasNonEnumerableProperties
= true;
873 UString::Rep
* rep
= propertyName
._ustring
.rep();
875 if (!m_propertyTable
)
876 createPropertyMapHashTable();
878 // FIXME: Consider a fast case for tables with no deleted sentinels.
880 unsigned i
= rep
->existingHash();
882 bool foundDeletedElement
= false;
883 unsigned deletedElementIndex
= 0; // initialize to make the compiler happy
885 #if DUMP_PROPERTYMAP_STATS
890 unsigned entryIndex
= m_propertyTable
->entryIndices
[i
& m_propertyTable
->sizeMask
];
891 if (entryIndex
== emptyEntryIndex
)
894 if (entryIndex
== deletedSentinelIndex
) {
895 // If we find a deleted-element sentinel, remember it for use later.
896 if (!foundDeletedElement
) {
897 foundDeletedElement
= true;
898 deletedElementIndex
= i
;
903 k
= 1 | doubleHash(rep
->existingHash());
904 #if DUMP_PROPERTYMAP_STATS
911 #if DUMP_PROPERTYMAP_STATS
916 // Figure out which entry to use.
917 unsigned entryIndex
= m_propertyTable
->keyCount
+ m_propertyTable
->deletedSentinelCount
+ 2;
918 if (foundDeletedElement
) {
919 i
= deletedElementIndex
;
920 --m_propertyTable
->deletedSentinelCount
;
922 // Since we're not making the table bigger, we can't use the entry one past
923 // the end that we were planning on using, so search backwards for the empty
924 // slot that we can use. We know it will be there because we did at least one
925 // deletion in the past that left an entry empty.
926 while (m_propertyTable
->entries()[--entryIndex
- 1].key
) { }
929 // Create a new hash table entry.
930 m_propertyTable
->entryIndices
[i
& m_propertyTable
->sizeMask
] = entryIndex
;
932 // Create a new hash table entry.
934 m_propertyTable
->entries()[entryIndex
- 1].key
= rep
;
935 m_propertyTable
->entries()[entryIndex
- 1].attributes
= attributes
;
936 m_propertyTable
->entries()[entryIndex
- 1].specificValue
= specificValue
;
937 m_propertyTable
->entries()[entryIndex
- 1].index
= ++m_propertyTable
->lastIndexUsed
;
940 if (m_propertyTable
->deletedOffsets
&& !m_propertyTable
->deletedOffsets
->isEmpty()) {
941 newOffset
= m_propertyTable
->deletedOffsets
->last();
942 m_propertyTable
->deletedOffsets
->removeLast();
944 newOffset
= m_propertyTable
->keyCount
+ m_anonymousSlotCount
;
945 m_propertyTable
->entries()[entryIndex
- 1].offset
= newOffset
;
947 ASSERT(newOffset
>= m_anonymousSlotCount
);
948 ++m_propertyTable
->keyCount
;
950 if ((m_propertyTable
->keyCount
+ m_propertyTable
->deletedSentinelCount
) * 2 >= m_propertyTable
->size
)
951 expandPropertyMapHashTable();
957 bool Structure::hasTransition(UString::Rep
* rep
, unsigned attributes
)
959 return transitionTableHasTransition(make_pair(rep
, attributes
));
962 size_t Structure::remove(const Identifier
& propertyName
)
964 ASSERT(!propertyName
.isNull());
968 UString::Rep
* rep
= propertyName
._ustring
.rep();
970 if (!m_propertyTable
)
973 #if DUMP_PROPERTYMAP_STATS
978 // Find the thing to remove.
979 unsigned i
= rep
->existingHash();
982 UString::Rep
* key
= 0;
984 entryIndex
= m_propertyTable
->entryIndices
[i
& m_propertyTable
->sizeMask
];
985 if (entryIndex
== emptyEntryIndex
)
988 key
= m_propertyTable
->entries()[entryIndex
- 1].key
;
993 k
= 1 | doubleHash(rep
->existingHash());
994 #if DUMP_PROPERTYMAP_STATS
1001 #if DUMP_PROPERTYMAP_STATS
1006 // Replace this one element with the deleted sentinel. Also clear out
1007 // the entry so we can iterate all the entries as needed.
1008 m_propertyTable
->entryIndices
[i
& m_propertyTable
->sizeMask
] = deletedSentinelIndex
;
1010 size_t offset
= m_propertyTable
->entries()[entryIndex
- 1].offset
;
1011 ASSERT(offset
>= m_anonymousSlotCount
);
1014 m_propertyTable
->entries()[entryIndex
- 1].key
= 0;
1015 m_propertyTable
->entries()[entryIndex
- 1].attributes
= 0;
1016 m_propertyTable
->entries()[entryIndex
- 1].specificValue
= 0;
1017 m_propertyTable
->entries()[entryIndex
- 1].offset
= 0;
1019 if (!m_propertyTable
->deletedOffsets
)
1020 m_propertyTable
->deletedOffsets
= new Vector
<unsigned>;
1021 m_propertyTable
->deletedOffsets
->append(offset
);
1023 ASSERT(m_propertyTable
->keyCount
>= 1);
1024 --m_propertyTable
->keyCount
;
1025 ++m_propertyTable
->deletedSentinelCount
;
1027 if (m_propertyTable
->deletedSentinelCount
* 4 >= m_propertyTable
->size
)
1028 rehashPropertyMapHashTable();
1034 void Structure::insertIntoPropertyMapHashTable(const PropertyMapEntry
& entry
)
1036 ASSERT(m_propertyTable
);
1037 ASSERT(entry
.offset
>= m_anonymousSlotCount
);
1038 unsigned i
= entry
.key
->existingHash();
1041 #if DUMP_PROPERTYMAP_STATS
1046 unsigned entryIndex
= m_propertyTable
->entryIndices
[i
& m_propertyTable
->sizeMask
];
1047 if (entryIndex
== emptyEntryIndex
)
1051 k
= 1 | doubleHash(entry
.key
->existingHash());
1052 #if DUMP_PROPERTYMAP_STATS
1059 #if DUMP_PROPERTYMAP_STATS
1064 unsigned entryIndex
= m_propertyTable
->keyCount
+ 2;
1065 m_propertyTable
->entryIndices
[i
& m_propertyTable
->sizeMask
] = entryIndex
;
1066 m_propertyTable
->entries()[entryIndex
- 1] = entry
;
1068 ++m_propertyTable
->keyCount
;
1071 void Structure::createPropertyMapHashTable()
1073 ASSERT(sizeForKeyCount(7) == newTableSize
);
1074 createPropertyMapHashTable(newTableSize
);
1077 void Structure::createPropertyMapHashTable(unsigned newTableSize
)
1079 ASSERT(!m_propertyTable
);
1080 ASSERT(isPowerOf2(newTableSize
));
1084 m_propertyTable
= static_cast<PropertyMapHashTable
*>(fastZeroedMalloc(PropertyMapHashTable::allocationSize(newTableSize
)));
1085 m_propertyTable
->size
= newTableSize
;
1086 m_propertyTable
->sizeMask
= newTableSize
- 1;
1091 void Structure::expandPropertyMapHashTable()
1093 ASSERT(m_propertyTable
);
1094 rehashPropertyMapHashTable(m_propertyTable
->size
* 2);
1097 void Structure::rehashPropertyMapHashTable()
1099 ASSERT(m_propertyTable
);
1100 ASSERT(m_propertyTable
->size
);
1101 rehashPropertyMapHashTable(m_propertyTable
->size
);
1104 void Structure::rehashPropertyMapHashTable(unsigned newTableSize
)
1106 ASSERT(m_propertyTable
);
1107 ASSERT(isPowerOf2(newTableSize
));
1111 PropertyMapHashTable
* oldTable
= m_propertyTable
;
1113 m_propertyTable
= static_cast<PropertyMapHashTable
*>(fastZeroedMalloc(PropertyMapHashTable::allocationSize(newTableSize
)));
1114 m_propertyTable
->size
= newTableSize
;
1115 m_propertyTable
->sizeMask
= newTableSize
- 1;
1117 unsigned lastIndexUsed
= 0;
1118 unsigned entryCount
= oldTable
->keyCount
+ oldTable
->deletedSentinelCount
;
1119 for (unsigned i
= 1; i
<= entryCount
; ++i
) {
1120 if (oldTable
->entries()[i
].key
) {
1121 lastIndexUsed
= max(oldTable
->entries()[i
].index
, lastIndexUsed
);
1122 insertIntoPropertyMapHashTable(oldTable
->entries()[i
]);
1125 m_propertyTable
->lastIndexUsed
= lastIndexUsed
;
1126 m_propertyTable
->deletedOffsets
= oldTable
->deletedOffsets
;
1133 int comparePropertyMapEntryIndices(const void* a
, const void* b
)
1135 unsigned ia
= static_cast<PropertyMapEntry
* const*>(a
)[0]->index
;
1136 unsigned ib
= static_cast<PropertyMapEntry
* const*>(b
)[0]->index
;
1144 void Structure::getPropertyNames(PropertyNameArray
& propertyNames
, EnumerationMode mode
)
1146 materializePropertyMapIfNecessary();
1147 if (!m_propertyTable
)
1150 if (m_propertyTable
->keyCount
< tinyMapThreshold
) {
1151 PropertyMapEntry
* a
[tinyMapThreshold
];
1153 unsigned entryCount
= m_propertyTable
->keyCount
+ m_propertyTable
->deletedSentinelCount
;
1154 for (unsigned k
= 1; k
<= entryCount
; k
++) {
1155 ASSERT(m_hasNonEnumerableProperties
|| !(m_propertyTable
->entries()[k
].attributes
& DontEnum
));
1156 if (m_propertyTable
->entries()[k
].key
&& (!(m_propertyTable
->entries()[k
].attributes
& DontEnum
) || (mode
== IncludeDontEnumProperties
))) {
1157 PropertyMapEntry
* value
= &m_propertyTable
->entries()[k
];
1159 for (j
= i
- 1; j
>= 0 && a
[j
]->index
> value
->index
; --j
)
1165 if (!propertyNames
.size()) {
1166 for (int k
= 0; k
< i
; ++k
)
1167 propertyNames
.addKnownUnique(a
[k
]->key
);
1169 for (int k
= 0; k
< i
; ++k
)
1170 propertyNames
.add(a
[k
]->key
);
1176 // Allocate a buffer to use to sort the keys.
1177 Vector
<PropertyMapEntry
*, smallMapThreshold
> sortedEnumerables(m_propertyTable
->keyCount
);
1179 // Get pointers to the enumerable entries in the buffer.
1180 PropertyMapEntry
** p
= sortedEnumerables
.data();
1181 unsigned entryCount
= m_propertyTable
->keyCount
+ m_propertyTable
->deletedSentinelCount
;
1182 for (unsigned i
= 1; i
<= entryCount
; i
++) {
1183 if (m_propertyTable
->entries()[i
].key
&& (!(m_propertyTable
->entries()[i
].attributes
& DontEnum
) || (mode
== IncludeDontEnumProperties
)))
1184 *p
++ = &m_propertyTable
->entries()[i
];
1187 size_t enumerableCount
= p
- sortedEnumerables
.data();
1188 // Sort the entries by index.
1189 qsort(sortedEnumerables
.data(), enumerableCount
, sizeof(PropertyMapEntry
*), comparePropertyMapEntryIndices
);
1190 sortedEnumerables
.resize(enumerableCount
);
1192 // Put the keys of the sorted entries into the list.
1193 if (!propertyNames
.size()) {
1194 for (size_t i
= 0; i
< sortedEnumerables
.size(); ++i
)
1195 propertyNames
.addKnownUnique(sortedEnumerables
[i
]->key
);
1197 for (size_t i
= 0; i
< sortedEnumerables
.size(); ++i
)
1198 propertyNames
.add(sortedEnumerables
[i
]->key
);
1202 #if DO_PROPERTYMAP_CONSTENCY_CHECK
1204 void Structure::checkConsistency()
1206 if (!m_propertyTable
)
1209 ASSERT(m_propertyTable
->size
>= newTableSize
);
1210 ASSERT(m_propertyTable
->sizeMask
);
1211 ASSERT(m_propertyTable
->size
== m_propertyTable
->sizeMask
+ 1);
1212 ASSERT(!(m_propertyTable
->size
& m_propertyTable
->sizeMask
));
1214 ASSERT(m_propertyTable
->keyCount
<= m_propertyTable
->size
/ 2);
1215 ASSERT(m_propertyTable
->deletedSentinelCount
<= m_propertyTable
->size
/ 4);
1217 ASSERT(m_propertyTable
->keyCount
+ m_propertyTable
->deletedSentinelCount
<= m_propertyTable
->size
/ 2);
1219 unsigned indexCount
= 0;
1220 unsigned deletedIndexCount
= 0;
1221 for (unsigned a
= 0; a
!= m_propertyTable
->size
; ++a
) {
1222 unsigned entryIndex
= m_propertyTable
->entryIndices
[a
];
1223 if (entryIndex
== emptyEntryIndex
)
1225 if (entryIndex
== deletedSentinelIndex
) {
1226 ++deletedIndexCount
;
1229 ASSERT(entryIndex
> deletedSentinelIndex
);
1230 ASSERT(entryIndex
- 1 <= m_propertyTable
->keyCount
+ m_propertyTable
->deletedSentinelCount
);
1233 for (unsigned b
= a
+ 1; b
!= m_propertyTable
->size
; ++b
)
1234 ASSERT(m_propertyTable
->entryIndices
[b
] != entryIndex
);
1236 ASSERT(indexCount
== m_propertyTable
->keyCount
);
1237 ASSERT(deletedIndexCount
== m_propertyTable
->deletedSentinelCount
);
1239 ASSERT(m_propertyTable
->entries()[0].key
== 0);
1241 unsigned nonEmptyEntryCount
= 0;
1242 for (unsigned c
= 1; c
<= m_propertyTable
->keyCount
+ m_propertyTable
->deletedSentinelCount
; ++c
) {
1243 ASSERT(m_hasNonEnumerableProperties
|| !(m_propertyTable
->entries()[c
].attributes
& DontEnum
));
1244 UString::Rep
* rep
= m_propertyTable
->entries()[c
].key
;
1245 ASSERT(m_propertyTable
->entries()[c
].offset
>= m_anonymousSlotCount
);
1248 ++nonEmptyEntryCount
;
1249 unsigned i
= rep
->existingHash();
1251 unsigned entryIndex
;
1253 entryIndex
= m_propertyTable
->entryIndices
[i
& m_propertyTable
->sizeMask
];
1254 ASSERT(entryIndex
!= emptyEntryIndex
);
1255 if (rep
== m_propertyTable
->entries()[entryIndex
- 1].key
)
1258 k
= 1 | doubleHash(rep
->existingHash());
1261 ASSERT(entryIndex
== c
+ 1);
1264 ASSERT(nonEmptyEntryCount
== m_propertyTable
->keyCount
);
1267 #endif // DO_PROPERTYMAP_CONSTENCY_CHECK