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