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 "NumberPrototype.h"
30 #include "StringPrototype.h"
31 #include "StructureChain.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 NativeErrorConstructor
;
46 class ProgramCodeBlock
;
47 class RegExpConstructor
;
48 class RegExpPrototype
;
51 struct ActivationStackNode
;
54 typedef Vector
<ExecState
*, 16> ExecStateStack
;
56 class JSGlobalObject
: public JSVariableObject
{
58 typedef HashSet
<RefPtr
<OpaqueJSWeakObjectMap
> > WeakMapSet
;
60 RefPtr
<JSGlobalData
> m_globalData
;
62 size_t m_registerArraySize
;
63 Register m_globalCallFrame
[RegisterFile::CallFrameHeaderSize
];
65 WriteBarrier
<ScopeChainNode
> m_globalScopeChain
;
66 WriteBarrier
<JSObject
> m_methodCallDummy
;
68 WriteBarrier
<RegExpConstructor
> m_regExpConstructor
;
69 WriteBarrier
<ErrorConstructor
> m_errorConstructor
;
70 WriteBarrier
<NativeErrorConstructor
> m_evalErrorConstructor
;
71 WriteBarrier
<NativeErrorConstructor
> m_rangeErrorConstructor
;
72 WriteBarrier
<NativeErrorConstructor
> m_referenceErrorConstructor
;
73 WriteBarrier
<NativeErrorConstructor
> m_syntaxErrorConstructor
;
74 WriteBarrier
<NativeErrorConstructor
> m_typeErrorConstructor
;
75 WriteBarrier
<NativeErrorConstructor
> m_URIErrorConstructor
;
77 WriteBarrier
<JSFunction
> m_evalFunction
;
78 WriteBarrier
<JSFunction
> m_callFunction
;
79 WriteBarrier
<JSFunction
> m_applyFunction
;
81 WriteBarrier
<ObjectPrototype
> m_objectPrototype
;
82 WriteBarrier
<FunctionPrototype
> m_functionPrototype
;
83 WriteBarrier
<ArrayPrototype
> m_arrayPrototype
;
84 WriteBarrier
<BooleanPrototype
> m_booleanPrototype
;
85 WriteBarrier
<StringPrototype
> m_stringPrototype
;
86 WriteBarrier
<NumberPrototype
> m_numberPrototype
;
87 WriteBarrier
<DatePrototype
> m_datePrototype
;
88 WriteBarrier
<RegExpPrototype
> m_regExpPrototype
;
90 WriteBarrier
<Structure
> m_argumentsStructure
;
91 WriteBarrier
<Structure
> m_arrayStructure
;
92 WriteBarrier
<Structure
> m_booleanObjectStructure
;
93 WriteBarrier
<Structure
> m_callbackConstructorStructure
;
94 WriteBarrier
<Structure
> m_callbackFunctionStructure
;
95 WriteBarrier
<Structure
> m_callbackObjectStructure
;
96 WriteBarrier
<Structure
> m_dateStructure
;
97 WriteBarrier
<Structure
> m_emptyObjectStructure
;
98 WriteBarrier
<Structure
> m_nullPrototypeObjectStructure
;
99 WriteBarrier
<Structure
> m_errorStructure
;
100 WriteBarrier
<Structure
> m_functionStructure
;
101 WriteBarrier
<Structure
> m_numberObjectStructure
;
102 WriteBarrier
<Structure
> m_regExpMatchesArrayStructure
;
103 WriteBarrier
<Structure
> m_regExpStructure
;
104 WriteBarrier
<Structure
> m_stringObjectStructure
;
105 WriteBarrier
<Structure
> m_internalFunctionStructure
;
107 unsigned m_profileGroup
;
108 Debugger
* m_debugger
;
110 WeakMapSet m_weakMaps
;
111 Weak
<JSGlobalObject
> m_weakMapsFinalizer
;
112 class WeakMapsFinalizer
: public WeakHandleOwner
{
114 virtual void finalize(Handle
<Unknown
>, void* context
);
116 static WeakMapsFinalizer
* weakMapsFinalizer();
118 WeakRandom m_weakRandom
;
120 SymbolTable m_symbolTable
;
122 bool m_isEvalEnabled
;
125 void* operator new(size_t, JSGlobalData
*);
127 explicit JSGlobalObject(JSGlobalData
& globalData
, Structure
* structure
)
128 : JSVariableObject(globalData
, structure
, &m_symbolTable
, 0)
129 , m_registerArraySize(0)
130 , m_globalScopeChain()
131 , m_weakRandom(static_cast<unsigned>(randomNumber() * (std::numeric_limits
<unsigned>::max() + 1.0)))
132 , m_isEvalEnabled(true)
134 COMPILE_ASSERT(JSGlobalObject::AnonymousSlotCount
== 1, JSGlobalObject_has_only_a_single_slot
);
135 putThisToAnonymousValue(0);
139 static JS_EXPORTDATA
const ClassInfo s_info
;
142 JSGlobalObject(JSGlobalData
& globalData
, Structure
* structure
, JSObject
* thisValue
)
143 : JSVariableObject(globalData
, structure
, &m_symbolTable
, 0)
144 , m_registerArraySize(0)
145 , m_globalScopeChain()
146 , m_weakRandom(static_cast<unsigned>(randomNumber() * (std::numeric_limits
<unsigned>::max() + 1.0)))
147 , m_isEvalEnabled(true)
149 COMPILE_ASSERT(JSGlobalObject::AnonymousSlotCount
== 1, JSGlobalObject_has_only_a_single_slot
);
150 putThisToAnonymousValue(0);
155 virtual ~JSGlobalObject();
157 virtual void visitChildren(SlotVisitor
&);
159 virtual bool getOwnPropertySlot(ExecState
*, const Identifier
&, PropertySlot
&);
160 virtual bool getOwnPropertyDescriptor(ExecState
*, const Identifier
&, PropertyDescriptor
&);
161 virtual bool hasOwnPropertyForWrite(ExecState
*, const Identifier
&);
162 virtual void put(ExecState
*, const Identifier
&, JSValue
, PutPropertySlot
&);
163 virtual void putWithAttributes(ExecState
*, const Identifier
& propertyName
, JSValue value
, unsigned attributes
);
165 virtual void defineGetter(ExecState
*, const Identifier
& propertyName
, JSObject
* getterFunc
, unsigned attributes
);
166 virtual void defineSetter(ExecState
*, const Identifier
& propertyName
, JSObject
* setterFunc
, unsigned attributes
);
168 // We use this in the code generator as we perform symbol table
169 // lookups prior to initializing the properties
170 bool symbolTableHasProperty(const Identifier
& propertyName
);
172 // The following accessors return pristine values, even if a script
173 // replaces the global object's associated property.
175 RegExpConstructor
* regExpConstructor() const { return m_regExpConstructor
.get(); }
177 ErrorConstructor
* errorConstructor() const { return m_errorConstructor
.get(); }
178 NativeErrorConstructor
* evalErrorConstructor() const { return m_evalErrorConstructor
.get(); }
179 NativeErrorConstructor
* rangeErrorConstructor() const { return m_rangeErrorConstructor
.get(); }
180 NativeErrorConstructor
* referenceErrorConstructor() const { return m_referenceErrorConstructor
.get(); }
181 NativeErrorConstructor
* syntaxErrorConstructor() const { return m_syntaxErrorConstructor
.get(); }
182 NativeErrorConstructor
* typeErrorConstructor() const { return m_typeErrorConstructor
.get(); }
183 NativeErrorConstructor
* URIErrorConstructor() const { return m_URIErrorConstructor
.get(); }
185 JSFunction
* evalFunction() const { return m_evalFunction
.get(); }
186 JSFunction
* callFunction() const { return m_callFunction
.get(); }
187 JSFunction
* applyFunction() const { return m_applyFunction
.get(); }
189 ObjectPrototype
* objectPrototype() const { return m_objectPrototype
.get(); }
190 FunctionPrototype
* functionPrototype() const { return m_functionPrototype
.get(); }
191 ArrayPrototype
* arrayPrototype() const { return m_arrayPrototype
.get(); }
192 BooleanPrototype
* booleanPrototype() const { return m_booleanPrototype
.get(); }
193 StringPrototype
* stringPrototype() const { return m_stringPrototype
.get(); }
194 NumberPrototype
* numberPrototype() const { return m_numberPrototype
.get(); }
195 DatePrototype
* datePrototype() const { return m_datePrototype
.get(); }
196 RegExpPrototype
* regExpPrototype() const { return m_regExpPrototype
.get(); }
198 JSObject
* methodCallDummy() const { return m_methodCallDummy
.get(); }
200 Structure
* argumentsStructure() const { return m_argumentsStructure
.get(); }
201 Structure
* arrayStructure() const { return m_arrayStructure
.get(); }
202 Structure
* booleanObjectStructure() const { return m_booleanObjectStructure
.get(); }
203 Structure
* callbackConstructorStructure() const { return m_callbackConstructorStructure
.get(); }
204 Structure
* callbackFunctionStructure() const { return m_callbackFunctionStructure
.get(); }
205 Structure
* callbackObjectStructure() const { return m_callbackObjectStructure
.get(); }
206 Structure
* dateStructure() const { return m_dateStructure
.get(); }
207 Structure
* emptyObjectStructure() const { return m_emptyObjectStructure
.get(); }
208 Structure
* nullPrototypeObjectStructure() const { return m_nullPrototypeObjectStructure
.get(); }
209 Structure
* errorStructure() const { return m_errorStructure
.get(); }
210 Structure
* functionStructure() const { return m_functionStructure
.get(); }
211 Structure
* numberObjectStructure() const { return m_numberObjectStructure
.get(); }
212 Structure
* internalFunctionStructure() const { return m_internalFunctionStructure
.get(); }
213 Structure
* regExpMatchesArrayStructure() const { return m_regExpMatchesArrayStructure
.get(); }
214 Structure
* regExpStructure() const { return m_regExpStructure
.get(); }
215 Structure
* stringObjectStructure() const { return m_stringObjectStructure
.get(); }
217 void setProfileGroup(unsigned value
) { m_profileGroup
= value
; }
218 unsigned profileGroup() const { return m_profileGroup
; }
220 Debugger
* debugger() const { return m_debugger
; }
221 void setDebugger(Debugger
* debugger
) { m_debugger
= debugger
; }
223 virtual bool supportsProfiling() const { return false; }
224 virtual bool supportsRichSourceInfo() const { return true; }
226 ScopeChainNode
* globalScopeChain() { return m_globalScopeChain
.get(); }
228 virtual bool isGlobalObject() const { return true; }
230 virtual ExecState
* globalExec();
232 virtual bool shouldInterruptScriptBeforeTimeout() const { return false; }
233 virtual bool shouldInterruptScript() const { return true; }
235 virtual bool allowsAccessFrom(const JSGlobalObject
*) const { return true; }
237 virtual bool isDynamicScope(bool& requiresDynamicChecks
) const;
240 bool isEvalEnabled() { return m_isEvalEnabled
; }
242 void copyGlobalsFrom(RegisterFile
&);
243 void copyGlobalsTo(RegisterFile
&);
244 void resizeRegisters(int oldSize
, int newSize
);
246 void resetPrototype(JSGlobalData
&, JSValue prototype
);
248 JSGlobalData
& globalData() const { return *m_globalData
.get(); }
250 static Structure
* createStructure(JSGlobalData
& globalData
, JSValue prototype
)
252 return Structure::create(globalData
, prototype
, TypeInfo(ObjectType
, StructureFlags
), AnonymousSlotCount
, &s_info
);
255 void registerWeakMap(OpaqueJSWeakObjectMap
* map
)
257 if (!m_weakMapsFinalizer
)
258 m_weakMapsFinalizer
.set(globalData(), this, weakMapsFinalizer());
262 void deregisterWeakMap(OpaqueJSWeakObjectMap
* map
)
264 m_weakMaps
.remove(map
);
267 double weakRandomNumber() { return m_weakRandom
.get(); }
270 static const unsigned AnonymousSlotCount
= JSVariableObject::AnonymousSlotCount
+ 1;
271 static const unsigned StructureFlags
= OverridesGetOwnPropertySlot
| OverridesVisitChildren
| OverridesGetPropertyNames
| JSVariableObject::StructureFlags
;
273 struct GlobalPropertyInfo
{
274 GlobalPropertyInfo(const Identifier
& i
, JSValue v
, unsigned a
)
281 const Identifier identifier
;
285 void addStaticGlobals(GlobalPropertyInfo
*, int count
);
288 // FIXME: Fold reset into init.
289 void init(JSObject
* thisValue
);
290 void reset(JSValue prototype
);
292 void setRegisters(WriteBarrier
<Unknown
>* registers
, PassOwnArrayPtr
<WriteBarrier
<Unknown
> > registerArray
, size_t count
);
294 void* operator new(size_t); // can only be allocated with JSGlobalData
297 JSGlobalObject
* asGlobalObject(JSValue
);
299 inline JSGlobalObject
* asGlobalObject(JSValue value
)
301 ASSERT(asObject(value
)->isGlobalObject());
302 return static_cast<JSGlobalObject
*>(asObject(value
));
305 inline void JSGlobalObject::setRegisters(WriteBarrier
<Unknown
>* registers
, PassOwnArrayPtr
<WriteBarrier
<Unknown
> > registerArray
, size_t count
)
307 JSVariableObject::setRegisters(registers
, registerArray
);
308 m_registerArraySize
= count
;
311 inline bool JSGlobalObject::hasOwnPropertyForWrite(ExecState
* exec
, const Identifier
& propertyName
)
314 if (JSVariableObject::getOwnPropertySlot(exec
, propertyName
, slot
))
316 bool slotIsWriteable
;
317 return symbolTableGet(propertyName
, slot
, slotIsWriteable
);
320 inline bool JSGlobalObject::symbolTableHasProperty(const Identifier
& propertyName
)
322 SymbolTableEntry entry
= symbolTable().inlineGet(propertyName
.impl());
323 return !entry
.isNull();
326 inline JSValue
Structure::prototypeForLookup(ExecState
* exec
) const
328 if (typeInfo().type() == ObjectType
)
329 return m_prototype
.get();
331 ASSERT(typeInfo().type() == StringType
);
332 return exec
->lexicalGlobalObject()->stringPrototype();
335 inline StructureChain
* Structure::prototypeChain(ExecState
* exec
) const
337 // We cache our prototype chain so our clients can share it.
338 if (!isValid(exec
, m_cachedPrototypeChain
.get())) {
339 JSValue prototype
= prototypeForLookup(exec
);
340 m_cachedPrototypeChain
.set(exec
->globalData(), this, StructureChain::create(exec
->globalData(), prototype
.isNull() ? 0 : asObject(prototype
)->structure()));
342 return m_cachedPrototypeChain
.get();
345 inline bool Structure::isValid(ExecState
* exec
, StructureChain
* cachedPrototypeChain
) const
347 if (!cachedPrototypeChain
)
350 JSValue prototype
= prototypeForLookup(exec
);
351 WriteBarrier
<Structure
>* cachedStructure
= cachedPrototypeChain
->head();
352 while(*cachedStructure
&& !prototype
.isNull()) {
353 if (asObject(prototype
)->structure() != cachedStructure
->get())
356 prototype
= asObject(prototype
)->prototype();
358 return prototype
.isNull() && !*cachedStructure
;
361 inline JSGlobalObject
* ExecState::dynamicGlobalObject()
363 if (this == lexicalGlobalObject()->globalExec())
364 return lexicalGlobalObject();
366 // For any ExecState that's not a globalExec, the
367 // dynamic global object must be set since code is running
368 ASSERT(globalData().dynamicGlobalObject
);
369 return globalData().dynamicGlobalObject
;
372 inline JSObject
* constructEmptyObject(ExecState
* exec
, JSGlobalObject
* globalObject
)
374 return constructEmptyObject(exec
, globalObject
->emptyObjectStructure());
377 inline JSObject
* constructEmptyObject(ExecState
* exec
)
379 return constructEmptyObject(exec
, exec
->lexicalGlobalObject());
382 inline JSArray
* constructEmptyArray(ExecState
* exec
, JSGlobalObject
* globalObject
)
384 return new (exec
) JSArray(exec
->globalData(), globalObject
->arrayStructure());
387 inline JSArray
* constructEmptyArray(ExecState
* exec
)
389 return constructEmptyArray(exec
, exec
->lexicalGlobalObject());
392 inline JSArray
* constructEmptyArray(ExecState
* exec
, JSGlobalObject
* globalObject
, unsigned initialLength
)
394 return new (exec
) JSArray(exec
->globalData(), globalObject
->arrayStructure(), initialLength
, CreateInitialized
);
397 inline JSArray
* constructEmptyArray(ExecState
* exec
, unsigned initialLength
)
399 return constructEmptyArray(exec
, exec
->lexicalGlobalObject(), initialLength
);
402 inline JSArray
* constructArray(ExecState
* exec
, JSGlobalObject
* globalObject
, JSValue singleItemValue
)
404 MarkedArgumentBuffer values
;
405 values
.append(singleItemValue
);
406 return new (exec
) JSArray(exec
->globalData(), globalObject
->arrayStructure(), values
);
409 inline JSArray
* constructArray(ExecState
* exec
, JSValue singleItemValue
)
411 return constructArray(exec
, exec
->lexicalGlobalObject(), singleItemValue
);
414 inline JSArray
* constructArray(ExecState
* exec
, JSGlobalObject
* globalObject
, const ArgList
& values
)
416 return new (exec
) JSArray(exec
->globalData(), globalObject
->arrayStructure(), values
);
419 inline JSArray
* constructArray(ExecState
* exec
, const ArgList
& values
)
421 return constructArray(exec
, exec
->lexicalGlobalObject(), values
);
424 class DynamicGlobalObjectScope
{
425 WTF_MAKE_NONCOPYABLE(DynamicGlobalObjectScope
);
427 DynamicGlobalObjectScope(JSGlobalData
&, JSGlobalObject
*);
429 ~DynamicGlobalObjectScope()
431 m_dynamicGlobalObjectSlot
= m_savedDynamicGlobalObject
;
435 JSGlobalObject
*& m_dynamicGlobalObjectSlot
;
436 JSGlobalObject
* m_savedDynamicGlobalObject
;
441 #endif // JSGlobalObject_h