2 * Copyright (C) 2007 Eric Seidel <eric@webkit.org>
3 * Copyright (C) 2007, 2008, 2009 Apple Inc. All rights reserved.
5 * This library is free software; you can redistribute it and/or
6 * modify it under the terms of the GNU Library General Public
7 * License as published by the Free Software Foundation; either
8 * version 2 of the License, or (at your option) any later version.
10 * This library is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 * Library General Public License for more details.
15 * You should have received a copy of the GNU Library General Public License
16 * along with this library; see the file COPYING.LIB. If not, write to
17 * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
18 * Boston, MA 02110-1301, USA.
22 #ifndef JSGlobalObject_h
23 #define JSGlobalObject_h
26 #include "JSGlobalData.h"
27 #include "JSVariableObject.h"
28 #include "JSWeakObjectMapRefInternal.h"
29 #include "NativeFunctionWrapper.h"
30 #include "NumberPrototype.h"
31 #include "StringPrototype.h"
32 #include <wtf/HashSet.h>
33 #include <wtf/OwnPtr.h>
34 #include <wtf/RandomNumber.h>
39 class BooleanPrototype
;
42 class ErrorConstructor
;
43 class FunctionPrototype
;
44 class GlobalCodeBlock
;
45 class GlobalEvalFunction
;
46 class NativeErrorConstructor
;
47 class ProgramCodeBlock
;
48 class PrototypeFunction
;
49 class RegExpConstructor
;
50 class RegExpPrototype
;
53 struct ActivationStackNode
;
56 typedef Vector
<ExecState
*, 16> ExecStateStack
;
58 class JSGlobalObject
: public JSVariableObject
{
60 using JSVariableObject::JSVariableObjectData
;
61 typedef HashSet
<RefPtr
<OpaqueJSWeakObjectMap
> > WeakMapSet
;
63 struct JSGlobalObjectData
: public JSVariableObjectData
{
64 // We use an explicit destructor function pointer instead of a
65 // virtual destructor because we want to avoid adding a vtable
66 // pointer to this struct. Adding a vtable pointer would force the
67 // compiler to emit costly pointer fixup code when casting from
68 // JSVariableObjectData* to JSGlobalObjectData*.
69 typedef void (*Destructor
)(void*);
71 JSGlobalObjectData(Destructor destructor
)
72 : JSVariableObjectData(&symbolTable
, 0)
73 , destructor(destructor
)
74 , registerArraySize(0)
75 , globalScopeChain(NoScopeChain())
76 , regExpConstructor(0)
78 , evalErrorConstructor(0)
79 , rangeErrorConstructor(0)
80 , referenceErrorConstructor(0)
81 , syntaxErrorConstructor(0)
82 , typeErrorConstructor(0)
83 , URIErrorConstructor(0)
88 , functionPrototype(0)
96 , weakRandom(static_cast<unsigned>(randomNumber() * (std::numeric_limits
<unsigned>::max() + 1.0)))
100 Destructor destructor
;
102 size_t registerArraySize
;
104 JSGlobalObject
* next
;
105 JSGlobalObject
* prev
;
109 ScopeChain globalScopeChain
;
110 Register globalCallFrame
[RegisterFile::CallFrameHeaderSize
];
114 RegExpConstructor
* regExpConstructor
;
115 ErrorConstructor
* errorConstructor
;
116 NativeErrorConstructor
* evalErrorConstructor
;
117 NativeErrorConstructor
* rangeErrorConstructor
;
118 NativeErrorConstructor
* referenceErrorConstructor
;
119 NativeErrorConstructor
* syntaxErrorConstructor
;
120 NativeErrorConstructor
* typeErrorConstructor
;
121 NativeErrorConstructor
* URIErrorConstructor
;
123 GlobalEvalFunction
* evalFunction
;
124 NativeFunctionWrapper
* callFunction
;
125 NativeFunctionWrapper
* applyFunction
;
127 ObjectPrototype
* objectPrototype
;
128 FunctionPrototype
* functionPrototype
;
129 ArrayPrototype
* arrayPrototype
;
130 BooleanPrototype
* booleanPrototype
;
131 StringPrototype
* stringPrototype
;
132 NumberPrototype
* numberPrototype
;
133 DatePrototype
* datePrototype
;
134 RegExpPrototype
* regExpPrototype
;
136 JSObject
* methodCallDummy
;
138 RefPtr
<Structure
> argumentsStructure
;
139 RefPtr
<Structure
> arrayStructure
;
140 RefPtr
<Structure
> booleanObjectStructure
;
141 RefPtr
<Structure
> callbackConstructorStructure
;
142 RefPtr
<Structure
> callbackFunctionStructure
;
143 RefPtr
<Structure
> callbackObjectStructure
;
144 RefPtr
<Structure
> dateStructure
;
145 RefPtr
<Structure
> emptyObjectStructure
;
146 RefPtr
<Structure
> errorStructure
;
147 RefPtr
<Structure
> functionStructure
;
148 RefPtr
<Structure
> numberObjectStructure
;
149 RefPtr
<Structure
> prototypeFunctionStructure
;
150 RefPtr
<Structure
> regExpMatchesArrayStructure
;
151 RefPtr
<Structure
> regExpStructure
;
152 RefPtr
<Structure
> stringObjectStructure
;
154 SymbolTable symbolTable
;
155 unsigned profileGroup
;
157 RefPtr
<JSGlobalData
> globalData
;
159 HashSet
<GlobalCodeBlock
*> codeBlocks
;
161 WeakRandom weakRandom
;
165 void* operator new(size_t, JSGlobalData
*);
167 explicit JSGlobalObject()
168 : JSVariableObject(JSGlobalObject::createStructure(jsNull()), new JSGlobalObjectData(destroyJSGlobalObjectData
))
174 JSGlobalObject(NonNullPassRefPtr
<Structure
> structure
, JSGlobalObjectData
* data
, JSObject
* thisValue
)
175 : JSVariableObject(structure
, data
)
181 virtual ~JSGlobalObject();
183 virtual void markChildren(MarkStack
&);
185 virtual bool getOwnPropertySlot(ExecState
*, const Identifier
&, PropertySlot
&);
186 virtual bool getOwnPropertyDescriptor(ExecState
*, const Identifier
&, PropertyDescriptor
&);
187 virtual bool hasOwnPropertyForWrite(ExecState
*, const Identifier
&);
188 virtual void put(ExecState
*, const Identifier
&, JSValue
, PutPropertySlot
&);
189 virtual void putWithAttributes(ExecState
*, const Identifier
& propertyName
, JSValue value
, unsigned attributes
);
191 virtual void defineGetter(ExecState
*, const Identifier
& propertyName
, JSObject
* getterFunc
, unsigned attributes
);
192 virtual void defineSetter(ExecState
*, const Identifier
& propertyName
, JSObject
* setterFunc
, unsigned attributes
);
194 // Linked list of all global objects that use the same JSGlobalData.
195 JSGlobalObject
*& head() { return d()->globalData
->head
; }
196 JSGlobalObject
* next() { return d()->next
; }
198 // The following accessors return pristine values, even if a script
199 // replaces the global object's associated property.
201 RegExpConstructor
* regExpConstructor() const { return d()->regExpConstructor
; }
203 ErrorConstructor
* errorConstructor() const { return d()->errorConstructor
; }
204 NativeErrorConstructor
* evalErrorConstructor() const { return d()->evalErrorConstructor
; }
205 NativeErrorConstructor
* rangeErrorConstructor() const { return d()->rangeErrorConstructor
; }
206 NativeErrorConstructor
* referenceErrorConstructor() const { return d()->referenceErrorConstructor
; }
207 NativeErrorConstructor
* syntaxErrorConstructor() const { return d()->syntaxErrorConstructor
; }
208 NativeErrorConstructor
* typeErrorConstructor() const { return d()->typeErrorConstructor
; }
209 NativeErrorConstructor
* URIErrorConstructor() const { return d()->URIErrorConstructor
; }
211 GlobalEvalFunction
* evalFunction() const { return d()->evalFunction
; }
213 ObjectPrototype
* objectPrototype() const { return d()->objectPrototype
; }
214 FunctionPrototype
* functionPrototype() const { return d()->functionPrototype
; }
215 ArrayPrototype
* arrayPrototype() const { return d()->arrayPrototype
; }
216 BooleanPrototype
* booleanPrototype() const { return d()->booleanPrototype
; }
217 StringPrototype
* stringPrototype() const { return d()->stringPrototype
; }
218 NumberPrototype
* numberPrototype() const { return d()->numberPrototype
; }
219 DatePrototype
* datePrototype() const { return d()->datePrototype
; }
220 RegExpPrototype
* regExpPrototype() const { return d()->regExpPrototype
; }
222 JSObject
* methodCallDummy() const { return d()->methodCallDummy
; }
224 Structure
* argumentsStructure() const { return d()->argumentsStructure
.get(); }
225 Structure
* arrayStructure() const { return d()->arrayStructure
.get(); }
226 Structure
* booleanObjectStructure() const { return d()->booleanObjectStructure
.get(); }
227 Structure
* callbackConstructorStructure() const { return d()->callbackConstructorStructure
.get(); }
228 Structure
* callbackFunctionStructure() const { return d()->callbackFunctionStructure
.get(); }
229 Structure
* callbackObjectStructure() const { return d()->callbackObjectStructure
.get(); }
230 Structure
* dateStructure() const { return d()->dateStructure
.get(); }
231 Structure
* emptyObjectStructure() const { return d()->emptyObjectStructure
.get(); }
232 Structure
* errorStructure() const { return d()->errorStructure
.get(); }
233 Structure
* functionStructure() const { return d()->functionStructure
.get(); }
234 Structure
* numberObjectStructure() const { return d()->numberObjectStructure
.get(); }
235 Structure
* prototypeFunctionStructure() const { return d()->prototypeFunctionStructure
.get(); }
236 Structure
* regExpMatchesArrayStructure() const { return d()->regExpMatchesArrayStructure
.get(); }
237 Structure
* regExpStructure() const { return d()->regExpStructure
.get(); }
238 Structure
* stringObjectStructure() const { return d()->stringObjectStructure
.get(); }
240 void setProfileGroup(unsigned value
) { d()->profileGroup
= value
; }
241 unsigned profileGroup() const { return d()->profileGroup
; }
243 Debugger
* debugger() const { return d()->debugger
; }
244 void setDebugger(Debugger
* debugger
) { d()->debugger
= debugger
; }
246 virtual bool supportsProfiling() const { return false; }
248 int recursion() { return d()->recursion
; }
249 void incRecursion() { ++d()->recursion
; }
250 void decRecursion() { --d()->recursion
; }
252 ScopeChain
& globalScopeChain() { return d()->globalScopeChain
; }
254 virtual bool isGlobalObject() const { return true; }
256 virtual ExecState
* globalExec();
258 virtual bool shouldInterruptScriptBeforeTimeout() const { return false; }
259 virtual bool shouldInterruptScript() const { return true; }
261 virtual bool allowsAccessFrom(const JSGlobalObject
*) const { return true; }
263 virtual bool isDynamicScope(bool& requiresDynamicChecks
) const;
265 HashSet
<GlobalCodeBlock
*>& codeBlocks() { return d()->codeBlocks
; }
267 void copyGlobalsFrom(RegisterFile
&);
268 void copyGlobalsTo(RegisterFile
&);
270 void resetPrototype(JSValue prototype
);
272 JSGlobalData
* globalData() { return d()->globalData
.get(); }
273 JSGlobalObjectData
* d() const { return static_cast<JSGlobalObjectData
*>(JSVariableObject::d
); }
275 static PassRefPtr
<Structure
> createStructure(JSValue prototype
)
277 return Structure::create(prototype
, TypeInfo(ObjectType
, StructureFlags
), AnonymousSlotCount
);
280 void registerWeakMap(OpaqueJSWeakObjectMap
* map
)
282 d()->weakMaps
.add(map
);
285 void deregisterWeakMap(OpaqueJSWeakObjectMap
* map
)
287 d()->weakMaps
.remove(map
);
290 double weakRandomNumber() { return d()->weakRandom
.get(); }
293 static const unsigned StructureFlags
= OverridesGetOwnPropertySlot
| OverridesMarkChildren
| OverridesGetPropertyNames
| JSVariableObject::StructureFlags
;
295 struct GlobalPropertyInfo
{
296 GlobalPropertyInfo(const Identifier
& i
, JSValue v
, unsigned a
)
303 const Identifier identifier
;
307 void addStaticGlobals(GlobalPropertyInfo
*, int count
);
310 static void destroyJSGlobalObjectData(void*);
312 // FIXME: Fold reset into init.
313 void init(JSObject
* thisValue
);
314 void reset(JSValue prototype
);
316 void setRegisters(Register
* registers
, Register
* registerArray
, size_t count
);
318 void* operator new(size_t); // can only be allocated with JSGlobalData
321 JSGlobalObject
* asGlobalObject(JSValue
);
323 inline JSGlobalObject
* asGlobalObject(JSValue value
)
325 ASSERT(asObject(value
)->isGlobalObject());
326 return static_cast<JSGlobalObject
*>(asObject(value
));
329 inline void JSGlobalObject::setRegisters(Register
* registers
, Register
* registerArray
, size_t count
)
331 JSVariableObject::setRegisters(registers
, registerArray
);
332 d()->registerArraySize
= count
;
335 inline void JSGlobalObject::addStaticGlobals(GlobalPropertyInfo
* globals
, int count
)
337 size_t oldSize
= d()->registerArraySize
;
338 size_t newSize
= oldSize
+ count
;
339 Register
* registerArray
= new Register
[newSize
];
340 if (d()->registerArray
)
341 memcpy(registerArray
+ count
, d()->registerArray
.get(), oldSize
* sizeof(Register
));
342 setRegisters(registerArray
+ newSize
, registerArray
, newSize
);
344 for (int i
= 0, index
= -static_cast<int>(oldSize
) - 1; i
< count
; ++i
, --index
) {
345 GlobalPropertyInfo
& global
= globals
[i
];
346 ASSERT(global
.attributes
& DontDelete
);
347 SymbolTableEntry
newEntry(index
, global
.attributes
);
348 symbolTable().add(global
.identifier
.ustring().rep(), newEntry
);
349 registerAt(index
) = global
.value
;
353 inline bool JSGlobalObject::getOwnPropertySlot(ExecState
* exec
, const Identifier
& propertyName
, PropertySlot
& slot
)
355 if (JSVariableObject::getOwnPropertySlot(exec
, propertyName
, slot
))
357 return symbolTableGet(propertyName
, slot
);
360 inline bool JSGlobalObject::getOwnPropertyDescriptor(ExecState
* exec
, const Identifier
& propertyName
, PropertyDescriptor
& descriptor
)
362 if (symbolTableGet(propertyName
, descriptor
))
364 return JSVariableObject::getOwnPropertyDescriptor(exec
, propertyName
, descriptor
);
367 inline bool JSGlobalObject::hasOwnPropertyForWrite(ExecState
* exec
, const Identifier
& propertyName
)
370 if (JSVariableObject::getOwnPropertySlot(exec
, propertyName
, slot
))
372 bool slotIsWriteable
;
373 return symbolTableGet(propertyName
, slot
, slotIsWriteable
);
376 inline JSValue
Structure::prototypeForLookup(ExecState
* exec
) const
378 if (typeInfo().type() == ObjectType
)
382 if (typeInfo().type() == StringType
)
383 return exec
->lexicalGlobalObject()->stringPrototype();
385 ASSERT(typeInfo().type() == NumberType
);
386 return exec
->lexicalGlobalObject()->numberPrototype();
388 ASSERT(typeInfo().type() == StringType
);
389 return exec
->lexicalGlobalObject()->stringPrototype();
393 inline StructureChain
* Structure::prototypeChain(ExecState
* exec
) const
395 // We cache our prototype chain so our clients can share it.
396 if (!isValid(exec
, m_cachedPrototypeChain
.get())) {
397 JSValue prototype
= prototypeForLookup(exec
);
398 m_cachedPrototypeChain
= StructureChain::create(prototype
.isNull() ? 0 : asObject(prototype
)->structure());
400 return m_cachedPrototypeChain
.get();
403 inline bool Structure::isValid(ExecState
* exec
, StructureChain
* cachedPrototypeChain
) const
405 if (!cachedPrototypeChain
)
408 JSValue prototype
= prototypeForLookup(exec
);
409 RefPtr
<Structure
>* cachedStructure
= cachedPrototypeChain
->head();
410 while(*cachedStructure
&& !prototype
.isNull()) {
411 if (asObject(prototype
)->structure() != *cachedStructure
)
414 prototype
= asObject(prototype
)->prototype();
416 return prototype
.isNull() && !*cachedStructure
;
419 inline JSGlobalObject
* ExecState::dynamicGlobalObject()
421 if (this == lexicalGlobalObject()->globalExec())
422 return lexicalGlobalObject();
424 // For any ExecState that's not a globalExec, the
425 // dynamic global object must be set since code is running
426 ASSERT(globalData().dynamicGlobalObject
);
427 return globalData().dynamicGlobalObject
;
430 inline JSObject
* constructEmptyObject(ExecState
* exec
)
432 return new (exec
) JSObject(exec
->lexicalGlobalObject()->emptyObjectStructure());
435 inline JSObject
* constructEmptyObject(ExecState
* exec
, JSGlobalObject
* globalObject
)
437 return new (exec
) JSObject(globalObject
->emptyObjectStructure());
440 inline JSArray
* constructEmptyArray(ExecState
* exec
)
442 return new (exec
) JSArray(exec
->lexicalGlobalObject()->arrayStructure());
445 inline JSArray
* constructEmptyArray(ExecState
* exec
, JSGlobalObject
* globalObject
)
447 return new (exec
) JSArray(globalObject
->arrayStructure());
450 inline JSArray
* constructEmptyArray(ExecState
* exec
, unsigned initialLength
)
452 return new (exec
) JSArray(exec
->lexicalGlobalObject()->arrayStructure(), initialLength
);
455 inline JSArray
* constructArray(ExecState
* exec
, JSValue singleItemValue
)
457 MarkedArgumentBuffer values
;
458 values
.append(singleItemValue
);
459 return new (exec
) JSArray(exec
->lexicalGlobalObject()->arrayStructure(), values
);
462 inline JSArray
* constructArray(ExecState
* exec
, const ArgList
& values
)
464 return new (exec
) JSArray(exec
->lexicalGlobalObject()->arrayStructure(), values
);
467 class DynamicGlobalObjectScope
: public Noncopyable
{
469 DynamicGlobalObjectScope(CallFrame
* callFrame
, JSGlobalObject
* dynamicGlobalObject
)
470 : m_dynamicGlobalObjectSlot(callFrame
->globalData().dynamicGlobalObject
)
471 , m_savedDynamicGlobalObject(m_dynamicGlobalObjectSlot
)
473 if (!m_dynamicGlobalObjectSlot
) {
474 m_dynamicGlobalObjectSlot
= dynamicGlobalObject
;
476 // Reset the date cache between JS invocations to force the VM
477 // to observe time zone changes.
478 callFrame
->globalData().resetDateCache();
482 ~DynamicGlobalObjectScope()
484 m_dynamicGlobalObjectSlot
= m_savedDynamicGlobalObject
;
488 JSGlobalObject
*& m_dynamicGlobalObjectSlot
;
489 JSGlobalObject
* m_savedDynamicGlobalObject
;
494 #endif // JSGlobalObject_h