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.  
  29 #include "Identifier.h" 
  33 #include "PropertyMapHashTable.h" 
  34 #include "PropertyNameArray.h" 
  36 #include "StructureTransitionTable.h" 
  37 #include "JSTypeInfo.h" 
  40 #include <wtf/PassOwnPtr.h> 
  41 #include <wtf/PassRefPtr.h> 
  42 #include <wtf/RefCounted.h> 
  48     class PropertyNameArray
; 
  49     class PropertyNameArrayData
; 
  51     typedef MarkStack SlotVisitor
; 
  55     enum EnumerationMode 
{ 
  56         ExcludeDontEnumProperties
, 
  57         IncludeDontEnumProperties
 
  60     class Structure 
: public JSCell 
{ 
  62         friend class StructureTransitionTable
; 
  63         static Structure
* create(JSGlobalData
& globalData
, JSValue prototype
, const TypeInfo
& typeInfo
, unsigned anonymousSlotCount
, const ClassInfo
* classInfo
) 
  65             ASSERT(globalData
.structureStructure
); 
  67             return new (&globalData
) Structure(globalData
, prototype
, typeInfo
, anonymousSlotCount
, classInfo
); 
  70         static void dumpStatistics(); 
  72         static Structure
* addPropertyTransition(JSGlobalData
&, Structure
*, const Identifier
& propertyName
, unsigned attributes
, JSCell
* specificValue
, size_t& offset
); 
  73         static Structure
* addPropertyTransitionToExistingStructure(Structure
*, const Identifier
& propertyName
, unsigned attributes
, JSCell
* specificValue
, size_t& offset
); 
  74         static Structure
* removePropertyTransition(JSGlobalData
&, Structure
*, const Identifier
& propertyName
, size_t& offset
); 
  75         static Structure
* changePrototypeTransition(JSGlobalData
&, Structure
*, JSValue prototype
); 
  76         static Structure
* despecifyFunctionTransition(JSGlobalData
&, Structure
*, const Identifier
&); 
  77         static Structure
* getterSetterTransition(JSGlobalData
&, Structure
*); 
  78         static Structure
* toCacheableDictionaryTransition(JSGlobalData
&, Structure
*); 
  79         static Structure
* toUncacheableDictionaryTransition(JSGlobalData
&, Structure
*); 
  80         static Structure
* sealTransition(JSGlobalData
&, Structure
*); 
  81         static Structure
* freezeTransition(JSGlobalData
&, Structure
*); 
  82         static Structure
* preventExtensionsTransition(JSGlobalData
&, Structure
*); 
  84         bool isSealed(JSGlobalData
&); 
  85         bool isFrozen(JSGlobalData
&); 
  86         bool isExtensible() const { return !m_preventExtensions
; } 
  87         bool didTransition() const { return m_didTransition
; } 
  89         Structure
* flattenDictionaryStructure(JSGlobalData
&, JSObject
*); 
  93         // These should be used with caution.   
  94         size_t addPropertyWithoutTransition(JSGlobalData
&, const Identifier
& propertyName
, unsigned attributes
, JSCell
* specificValue
); 
  95         size_t removePropertyWithoutTransition(JSGlobalData
&, const Identifier
& propertyName
); 
  96         void setPrototypeWithoutTransition(JSGlobalData
& globalData
, JSValue prototype
) { m_prototype
.set(globalData
, this, prototype
); } 
  98         bool isDictionary() const { return m_dictionaryKind 
!= NoneDictionaryKind
; } 
  99         bool isUncacheableDictionary() const { return m_dictionaryKind 
== UncachedDictionaryKind
; } 
 101         const TypeInfo
& typeInfo() const { ASSERT(structure()->classInfo() == &s_info
); return m_typeInfo
; } 
 103         JSValue 
storedPrototype() const { return m_prototype
.get(); } 
 104         JSValue 
prototypeForLookup(ExecState
*) const; 
 105         StructureChain
* prototypeChain(ExecState
*) const; 
 106         void visitChildren(SlotVisitor
&); 
 108         Structure
* previousID() const { ASSERT(structure()->classInfo() == &s_info
); return m_previous
.get(); } 
 110         void growPropertyStorageCapacity(); 
 111         unsigned propertyStorageCapacity() const { ASSERT(structure()->classInfo() == &s_info
); return m_propertyStorageCapacity
; } 
 112         unsigned propertyStorageSize() const { ASSERT(structure()->classInfo() == &s_info
); return m_anonymousSlotCount 
+ (m_propertyTable 
? m_propertyTable
->propertyStorageSize() : static_cast<unsigned>(m_offset 
+ 1)); } 
 113         bool isUsingInlineStorage() const; 
 115         size_t get(JSGlobalData
&, const Identifier
& propertyName
); 
 116         size_t get(JSGlobalData
&, StringImpl
* propertyName
, unsigned& attributes
, JSCell
*& specificValue
); 
 117         size_t get(JSGlobalData
& globalData
, const Identifier
& propertyName
, unsigned& attributes
, JSCell
*& specificValue
) 
 119             ASSERT(!propertyName
.isNull()); 
 120             ASSERT(structure()->classInfo() == &s_info
); 
 121             return get(globalData
, propertyName
.impl(), attributes
, specificValue
); 
 124         bool hasGetterSetterProperties() const { return m_hasGetterSetterProperties
; } 
 125         void setHasGetterSetterProperties(bool hasGetterSetterProperties
) { m_hasGetterSetterProperties 
= hasGetterSetterProperties
; } 
 127         bool hasNonEnumerableProperties() const { return m_hasNonEnumerableProperties
; } 
 129         bool hasAnonymousSlots() const { return !!m_anonymousSlotCount
; } 
 130         unsigned anonymousSlotCount() const { return m_anonymousSlotCount
; } 
 132         bool isEmpty() const { return m_propertyTable 
? m_propertyTable
->isEmpty() : m_offset 
== noOffset
; } 
 134         void despecifyDictionaryFunction(JSGlobalData
&, const Identifier
& propertyName
); 
 135         void disableSpecificFunctionTracking() { m_specificFunctionThrashCount 
= maxSpecificFunctionThrashCount
; } 
 137         void setEnumerationCache(JSGlobalData
&, JSPropertyNameIterator
* enumerationCache
); // Defined in JSPropertyNameIterator.h. 
 138         JSPropertyNameIterator
* enumerationCache(); // Defined in JSPropertyNameIterator.h. 
 139         void getPropertyNames(JSGlobalData
&, PropertyNameArray
&, EnumerationMode mode
); 
 141         const ClassInfo
* classInfo() const { return m_classInfo
; } 
 143         static ptrdiff_t prototypeOffset() 
 145             return OBJECT_OFFSETOF(Structure
, m_prototype
); 
 148         static ptrdiff_t typeInfoFlagsOffset() 
 150             return OBJECT_OFFSETOF(Structure
, m_typeInfo
) + TypeInfo::flagsOffset(); 
 153         static ptrdiff_t typeInfoTypeOffset() 
 155             return OBJECT_OFFSETOF(Structure
, m_typeInfo
) + TypeInfo::typeOffset(); 
 158         static Structure
* createStructure(JSGlobalData
& globalData
) 
 160             ASSERT(!globalData
.structureStructure
); 
 161             return new (&globalData
) Structure(globalData
); 
 164         static JS_EXPORTDATA 
const ClassInfo s_info
; 
 167         Structure(JSGlobalData
&, JSValue prototype
, const TypeInfo
&, unsigned anonymousSlotCount
, const ClassInfo
*); 
 168         Structure(JSGlobalData
&); 
 169         Structure(JSGlobalData
&, const Structure
*); 
 171         static Structure
* create(JSGlobalData
& globalData
, const Structure
* structure
) 
 173             ASSERT(globalData
.structureStructure
); 
 174             return new (&globalData
) Structure(globalData
, structure
); 
 178             NoneDictionaryKind 
= 0, 
 179             CachedDictionaryKind 
= 1, 
 180             UncachedDictionaryKind 
= 2 
 182         static Structure
* toDictionaryTransition(JSGlobalData
&, Structure
*, DictionaryKind
); 
 184         size_t putSpecificValue(JSGlobalData
&, const Identifier
& propertyName
, unsigned attributes
, JSCell
* specificValue
); 
 185         size_t remove(const Identifier
& propertyName
); 
 187         void createPropertyMap(unsigned keyCount 
= 0); 
 188         void checkConsistency(); 
 190         bool despecifyFunction(JSGlobalData
&, const Identifier
&); 
 191         void despecifyAllFunctions(JSGlobalData
&); 
 193         PassOwnPtr
<PropertyTable
> copyPropertyTable(JSGlobalData
&, Structure
* owner
); 
 194         void materializePropertyMap(JSGlobalData
&); 
 195         void materializePropertyMapIfNecessary(JSGlobalData
& globalData
) 
 197             ASSERT(structure()->classInfo() == &s_info
); 
 198             if (!m_propertyTable 
&& m_previous
) 
 199                 materializePropertyMap(globalData
); 
 202         signed char transitionCount() const 
 204             // Since the number of transitions is always the same as m_offset, we keep the size of Structure down by not storing both. 
 205             return m_offset 
== noOffset 
? 0 : m_offset 
+ 1; 
 208         bool isValid(ExecState
*, StructureChain
* cachedPrototypeChain
) const; 
 210         static const signed char s_maxTransitionLength 
= 64; 
 212         static const signed char noOffset 
= -1; 
 214         static const unsigned maxSpecificFunctionThrashCount 
= 3; 
 218         WriteBarrier
<Unknown
> m_prototype
; 
 219         mutable WriteBarrier
<StructureChain
> m_cachedPrototypeChain
; 
 221         WriteBarrier
<Structure
> m_previous
; 
 222         RefPtr
<StringImpl
> m_nameInPrevious
; 
 223         WriteBarrier
<JSCell
> m_specificValueInPrevious
; 
 225         const ClassInfo
* m_classInfo
; 
 227         StructureTransitionTable m_transitionTable
; 
 229         WriteBarrier
<JSPropertyNameIterator
> m_enumerationCache
; 
 231         OwnPtr
<PropertyTable
> m_propertyTable
; 
 233         uint32_t m_propertyStorageCapacity
; 
 235         // m_offset does not account for anonymous slots 
 236         signed char m_offset
; 
 238         unsigned m_dictionaryKind 
: 2; 
 239         bool m_isPinnedPropertyTable 
: 1; 
 240         bool m_hasGetterSetterProperties 
: 1; 
 241         bool m_hasNonEnumerableProperties 
: 1; 
 243         // Workaround for Symbian WINSCW compiler that cannot resolve unsigned type of the declared  
 244         // bitfield, when used as argument in make_pair() function calls in structure.ccp. 
 245         // This bitfield optimization is insignificant for the Symbian emulator target. 
 246         unsigned m_attributesInPrevious
; 
 248         unsigned m_attributesInPrevious 
: 7; 
 250         unsigned m_specificFunctionThrashCount 
: 2; 
 251         unsigned m_anonymousSlotCount 
: 5; 
 252         unsigned m_preventExtensions 
: 1; 
 253         unsigned m_didTransition 
: 1; 
 257     inline size_t Structure::get(JSGlobalData
& globalData
, const Identifier
& propertyName
) 
 259         ASSERT(structure()->classInfo() == &s_info
); 
 260         materializePropertyMapIfNecessary(globalData
); 
 261         if (!m_propertyTable
) 
 264         PropertyMapEntry
* entry 
= m_propertyTable
->find(propertyName
.impl()).first
; 
 265         ASSERT(!entry 
|| entry
->offset 
>= m_anonymousSlotCount
); 
 266         return entry 
? entry
->offset 
: notFound
; 
 269     inline bool JSCell::isObject() const 
 271         return m_structure
->typeInfo().type() == ObjectType
; 
 274     inline bool JSCell::isString() const 
 276         return m_structure
->typeInfo().type() == StringType
; 
 279     inline const ClassInfo
* JSCell::classInfo() const 
 281 #if ENABLE(GC_VALIDATION) 
 282         return m_structure
.unvalidatedGet()->classInfo(); 
 284         return m_structure
->classInfo(); 
 288     inline Structure
* JSCell::createDummyStructure(JSGlobalData
& globalData
) 
 290         return Structure::create(globalData
, jsNull(), TypeInfo(UnspecifiedType
), AnonymousSlotCount
, &s_dummyCellInfo
); 
 293     inline bool JSValue::needsThisConversion() const 
 295         if (UNLIKELY(!isCell())) 
 297         return asCell()->structure()->typeInfo().needsThisConversion(); 
 300     ALWAYS_INLINE 
void MarkStack::internalAppend(JSCell
* cell
) 
 302         ASSERT(!m_isCheckingForDefaultMarkViolation
); 
 304         if (Heap::testAndSetMarked(cell
)) 
 306         if (cell
->structure() && cell
->structure()->typeInfo().type() >= CompoundType
) 
 307             m_values
.append(cell
); 
 310     inline StructureTransitionTable::Hash::Key 
StructureTransitionTable::keyForWeakGCMapFinalizer(void*, Structure
* structure
) 
 312         // Newer versions of the STL have an std::make_pair function that takes rvalue references. 
 313         // 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. 
 314         // See https://bugs.webkit.org/show_bug.cgi?id=59261 for more details. 
 315         return Hash::Key(structure
->m_nameInPrevious
.get(), +structure
->m_attributesInPrevious
); 
 320 #endif // Structure_h