2 * Copyright (C) 1999-2001 Harri Porten (porten@kde.org)
3 * Copyright (C) 2001 Peter Kelly (pmk@post.com)
4 * Copyright (C) 2003, 2004, 2005, 2007, 2008, 2009 Apple Inc. All rights reserved.
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Library General Public
8 * License as published by the Free Software Foundation; either
9 * version 2 of the License, or (at your option) any later version.
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Library General Public License for more details.
16 * You should have received a copy of the GNU Library General Public License
17 * along with this library; see the file COPYING.LIB. If not, write to
18 * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
19 * Boston, MA 02110-1301, USA.
26 #include "Collector.h"
27 #include "JSImmediate.h"
29 #include "MarkStack.h"
30 #include "Structure.h"
31 #include <wtf/Noncopyable.h>
35 class JSCell
: public NoncopyableCustomAllocated
{
36 friend class GetterSetter
;
39 friend class JSNumberCell
;
40 friend class JSObject
;
41 friend class JSPropertyNameIterator
;
42 friend class JSString
;
44 friend class JSAPIValueWrapper
;
45 friend class JSZombie
;
46 friend class JSGlobalData
;
49 explicit JSCell(Structure
*);
53 static PassRefPtr
<Structure
> createDummyStructure()
55 return Structure::create(jsNull(), TypeInfo(UnspecifiedType
), AnonymousSlotCount
);
60 bool isNumber() const;
62 bool isString() const;
63 bool isObject() const;
64 virtual bool isGetterSetter() const;
65 bool inherits(const ClassInfo
*) const;
66 virtual bool isAPIValueWrapper() const { return false; }
67 virtual bool isPropertyNameIterator() const { return false; }
69 Structure
* structure() const;
71 // Extracting the value.
72 bool getString(ExecState
* exec
, UString
&) const;
73 UString
getString(ExecState
* exec
) const; // null string if not a string
74 JSObject
* getObject(); // NULL if not an object
75 const JSObject
* getObject() const; // NULL if not an object
77 virtual CallType
getCallData(CallData
&);
78 virtual ConstructType
getConstructData(ConstructData
&);
80 // Extracting integer values.
81 // FIXME: remove these methods, can check isNumberCell in JSValue && then call asNumberCell::*.
82 virtual bool getUInt32(uint32_t&) const;
85 virtual JSValue
toPrimitive(ExecState
*, PreferredPrimitiveType
) const;
86 virtual bool getPrimitiveNumber(ExecState
*, double& number
, JSValue
&);
87 virtual bool toBoolean(ExecState
*) const;
88 virtual double toNumber(ExecState
*) const;
89 virtual UString
toString(ExecState
*) const;
90 virtual JSObject
* toObject(ExecState
*) const;
92 // Garbage collection.
93 void* operator new(size_t, ExecState
*);
94 void* operator new(size_t, JSGlobalData
*);
95 void* operator new(size_t, void* placementNewDestination
) { return placementNewDestination
; }
97 virtual void markChildren(MarkStack
&);
98 #if ENABLE(JSC_ZOMBIES)
99 virtual bool isZombie() const { return false; }
102 // Object operations, with the toObject operation included.
103 virtual const ClassInfo
* classInfo() const;
104 virtual void put(ExecState
*, const Identifier
& propertyName
, JSValue
, PutPropertySlot
&);
105 virtual void put(ExecState
*, unsigned propertyName
, JSValue
);
106 virtual bool deleteProperty(ExecState
*, const Identifier
& propertyName
);
107 virtual bool deleteProperty(ExecState
*, unsigned propertyName
);
109 virtual JSObject
* toThisObject(ExecState
*) const;
110 virtual JSValue
getJSNumber();
111 void* vptr() { return *reinterpret_cast<void**>(this); }
112 void setVPtr(void* vptr
) { *reinterpret_cast<void**>(this) = vptr
; }
114 // FIXME: Rename getOwnPropertySlot to virtualGetOwnPropertySlot, and
115 // fastGetOwnPropertySlot to getOwnPropertySlot. Callers should always
116 // call this function, not its slower virtual counterpart. (For integer
117 // property names, we want a similar interface with appropriate optimizations.)
118 bool fastGetOwnPropertySlot(ExecState
*, const Identifier
& propertyName
, PropertySlot
&);
121 static const unsigned AnonymousSlotCount
= 0;
124 // Base implementation; for non-object classes implements getPropertySlot.
125 virtual bool getOwnPropertySlot(ExecState
*, const Identifier
& propertyName
, PropertySlot
&);
126 virtual bool getOwnPropertySlot(ExecState
*, unsigned propertyName
, PropertySlot
&);
128 Structure
* m_structure
;
131 inline JSCell::JSCell(Structure
* structure
)
132 : m_structure(structure
)
136 inline JSCell::~JSCell()
141 inline bool JSCell::isNumber() const
143 return m_structure
->typeInfo().type() == NumberType
;
147 inline bool JSCell::isObject() const
149 return m_structure
->typeInfo().type() == ObjectType
;
152 inline bool JSCell::isString() const
154 return m_structure
->typeInfo().type() == StringType
;
157 inline Structure
* JSCell::structure() const
162 inline void JSCell::markChildren(MarkStack
&)
166 inline void* JSCell::operator new(size_t size
, JSGlobalData
* globalData
)
168 return globalData
->heap
.allocate(size
);
171 inline void* JSCell::operator new(size_t size
, ExecState
* exec
)
173 return exec
->heap()->allocate(size
);
176 // --- JSValue inlines ----------------------------
178 inline bool JSValue::isString() const
180 return isCell() && asCell()->isString();
183 inline bool JSValue::isGetterSetter() const
185 return isCell() && asCell()->isGetterSetter();
188 inline bool JSValue::isObject() const
190 return isCell() && asCell()->isObject();
193 inline bool JSValue::getString(ExecState
* exec
, UString
& s
) const
195 return isCell() && asCell()->getString(exec
, s
);
198 inline UString
JSValue::getString(ExecState
* exec
) const
200 return isCell() ? asCell()->getString(exec
) : UString();
203 inline JSObject
* JSValue::getObject() const
205 return isCell() ? asCell()->getObject() : 0;
208 inline CallType
JSValue::getCallData(CallData
& callData
)
210 return isCell() ? asCell()->getCallData(callData
) : CallTypeNone
;
213 inline ConstructType
JSValue::getConstructData(ConstructData
& constructData
)
215 return isCell() ? asCell()->getConstructData(constructData
) : ConstructTypeNone
;
218 ALWAYS_INLINE
bool JSValue::getUInt32(uint32_t& v
) const
221 int32_t i
= asInt32();
222 v
= static_cast<uint32_t>(i
);
226 double d
= asDouble();
227 v
= static_cast<uint32_t>(d
);
233 #if !USE(JSVALUE32_64)
234 ALWAYS_INLINE JSCell
* JSValue::asCell() const
239 #endif // !USE(JSVALUE32_64)
241 inline JSValue
JSValue::toPrimitive(ExecState
* exec
, PreferredPrimitiveType preferredType
) const
243 return isCell() ? asCell()->toPrimitive(exec
, preferredType
) : asValue();
246 inline bool JSValue::getPrimitiveNumber(ExecState
* exec
, double& number
, JSValue
& value
)
259 return asCell()->getPrimitiveNumber(exec
, number
, value
);
265 if (isFalse() || isNull()) {
270 ASSERT(isUndefined());
271 number
= nonInlineNaN();
276 inline bool JSValue::toBoolean(ExecState
* exec
) const
279 return asInt32() != 0;
281 return asDouble() > 0.0 || asDouble() < 0.0; // false for NaN
283 return asCell()->toBoolean(exec
);
284 return isTrue(); // false, null, and undefined all convert to false.
287 ALWAYS_INLINE
double JSValue::toNumber(ExecState
* exec
) const
294 return asCell()->toNumber(exec
);
297 return isUndefined() ? nonInlineNaN() : 0; // null and false both convert to 0.
300 inline bool JSValue::needsThisConversion() const
302 if (UNLIKELY(!isCell()))
304 return asCell()->structure()->typeInfo().needsThisConversion();
307 inline JSValue
JSValue::getJSNumber()
309 if (isInt32() || isDouble())
312 return asCell()->getJSNumber();
316 inline JSObject
* JSValue::toObject(ExecState
* exec
) const
318 return isCell() ? asCell()->toObject(exec
) : toObjectSlowCase(exec
);
321 inline JSObject
* JSValue::toThisObject(ExecState
* exec
) const
323 return isCell() ? asCell()->toThisObject(exec
) : toThisObjectSlowCase(exec
);
326 ALWAYS_INLINE
void MarkStack::append(JSCell
* cell
)
328 ASSERT(!m_isCheckingForDefaultMarkViolation
);
330 if (Heap::isCellMarked(cell
))
332 Heap::markCell(cell
);
333 if (cell
->structure()->typeInfo().type() >= CompoundType
)
334 m_values
.append(cell
);
337 ALWAYS_INLINE
void MarkStack::append(JSValue value
)
341 append(value
.asCell());
344 inline Heap
* Heap::heap(JSValue v
)
348 return heap(v
.asCell());
351 inline Heap
* Heap::heap(JSCell
* c
)
353 return cellBlock(c
)->heap
;
356 #if ENABLE(JSC_ZOMBIES)
357 inline bool JSValue::isZombie() const
359 return isCell() && asCell() && asCell()->isZombie();