2 * Copyright (C) 2007 Eric Seidel <eric@webkit.org>
3 * Copyright (C) 2007, 2008 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
25 #include "JSGlobalData.h"
26 #include "JSVariableObject.h"
27 #include "NumberPrototype.h"
28 #include "StringPrototype.h"
29 #include <wtf/HashSet.h>
30 #include <wtf/OwnPtr.h>
35 class BooleanPrototype
;
38 class ErrorConstructor
;
39 class FunctionPrototype
;
40 class GlobalEvalFunction
;
41 class NativeErrorConstructor
;
42 class ProgramCodeBlock
;
43 class RegExpConstructor
;
44 class RegExpPrototype
;
47 struct ActivationStackNode
;
50 typedef Vector
<ExecState
*, 16> ExecStateStack
;
52 class JSGlobalObject
: public JSVariableObject
{
54 using JSVariableObject::JSVariableObjectData
;
56 struct JSGlobalObjectData
: public JSVariableObjectData
{
58 : JSVariableObjectData(&symbolTable
, 0)
59 , registerArraySize(0)
60 , globalScopeChain(NoScopeChain())
61 , regExpConstructor(0)
63 , evalErrorConstructor(0)
64 , rangeErrorConstructor(0)
65 , referenceErrorConstructor(0)
66 , syntaxErrorConstructor(0)
67 , typeErrorConstructor(0)
68 , URIErrorConstructor(0)
71 , functionPrototype(0)
81 virtual ~JSGlobalObjectData()
85 size_t registerArraySize
;
92 ScopeChain globalScopeChain
;
93 Register globalCallFrame
[RegisterFile::CallFrameHeaderSize
];
97 RegExpConstructor
* regExpConstructor
;
98 ErrorConstructor
* errorConstructor
;
99 NativeErrorConstructor
* evalErrorConstructor
;
100 NativeErrorConstructor
* rangeErrorConstructor
;
101 NativeErrorConstructor
* referenceErrorConstructor
;
102 NativeErrorConstructor
* syntaxErrorConstructor
;
103 NativeErrorConstructor
* typeErrorConstructor
;
104 NativeErrorConstructor
* URIErrorConstructor
;
106 GlobalEvalFunction
* evalFunction
;
108 ObjectPrototype
* objectPrototype
;
109 FunctionPrototype
* functionPrototype
;
110 ArrayPrototype
* arrayPrototype
;
111 BooleanPrototype
* booleanPrototype
;
112 StringPrototype
* stringPrototype
;
113 NumberPrototype
* numberPrototype
;
114 DatePrototype
* datePrototype
;
115 RegExpPrototype
* regExpPrototype
;
117 RefPtr
<Structure
> argumentsStructure
;
118 RefPtr
<Structure
> arrayStructure
;
119 RefPtr
<Structure
> booleanObjectStructure
;
120 RefPtr
<Structure
> callbackConstructorStructure
;
121 RefPtr
<Structure
> callbackFunctionStructure
;
122 RefPtr
<Structure
> callbackObjectStructure
;
123 RefPtr
<Structure
> dateStructure
;
124 RefPtr
<Structure
> emptyObjectStructure
;
125 RefPtr
<Structure
> errorStructure
;
126 RefPtr
<Structure
> functionStructure
;
127 RefPtr
<Structure
> numberObjectStructure
;
128 RefPtr
<Structure
> prototypeFunctionStructure
;
129 RefPtr
<Structure
> regExpMatchesArrayStructure
;
130 RefPtr
<Structure
> regExpStructure
;
131 RefPtr
<Structure
> stringObjectStructure
;
133 SymbolTable symbolTable
;
134 unsigned profileGroup
;
136 RefPtr
<JSGlobalData
> globalData
;
138 HashSet
<ProgramCodeBlock
*> codeBlocks
;
142 void* operator new(size_t, JSGlobalData
*);
144 explicit JSGlobalObject()
145 : JSVariableObject(JSGlobalObject::createStructure(jsNull()), new JSGlobalObjectData
)
151 JSGlobalObject(PassRefPtr
<Structure
> structure
, JSGlobalObjectData
* data
, JSObject
* thisValue
)
152 : JSVariableObject(structure
, data
)
158 virtual ~JSGlobalObject();
162 virtual bool getOwnPropertySlot(ExecState
*, const Identifier
&, PropertySlot
&);
163 virtual bool getOwnPropertySlot(ExecState
*, const Identifier
&, PropertySlot
&, bool& slotIsWriteable
);
164 virtual void put(ExecState
*, const Identifier
&, JSValuePtr
, PutPropertySlot
&);
165 virtual void putWithAttributes(ExecState
*, const Identifier
& propertyName
, JSValuePtr value
, unsigned attributes
);
167 virtual void defineGetter(ExecState
*, const Identifier
& propertyName
, JSObject
* getterFunc
);
168 virtual void defineSetter(ExecState
*, const Identifier
& propertyName
, JSObject
* setterFunc
);
170 // Linked list of all global objects that use the same JSGlobalData.
171 JSGlobalObject
*& head() { return d()->globalData
->head
; }
172 JSGlobalObject
* next() { return d()->next
; }
174 // The following accessors return pristine values, even if a script
175 // replaces the global object's associated property.
177 RegExpConstructor
* regExpConstructor() const { return d()->regExpConstructor
; }
179 ErrorConstructor
* errorConstructor() const { return d()->errorConstructor
; }
180 NativeErrorConstructor
* evalErrorConstructor() const { return d()->evalErrorConstructor
; }
181 NativeErrorConstructor
* rangeErrorConstructor() const { return d()->rangeErrorConstructor
; }
182 NativeErrorConstructor
* referenceErrorConstructor() const { return d()->referenceErrorConstructor
; }
183 NativeErrorConstructor
* syntaxErrorConstructor() const { return d()->syntaxErrorConstructor
; }
184 NativeErrorConstructor
* typeErrorConstructor() const { return d()->typeErrorConstructor
; }
185 NativeErrorConstructor
* URIErrorConstructor() const { return d()->URIErrorConstructor
; }
187 GlobalEvalFunction
* evalFunction() const { return d()->evalFunction
; }
189 ObjectPrototype
* objectPrototype() const { return d()->objectPrototype
; }
190 FunctionPrototype
* functionPrototype() const { return d()->functionPrototype
; }
191 ArrayPrototype
* arrayPrototype() const { return d()->arrayPrototype
; }
192 BooleanPrototype
* booleanPrototype() const { return d()->booleanPrototype
; }
193 StringPrototype
* stringPrototype() const { return d()->stringPrototype
; }
194 NumberPrototype
* numberPrototype() const { return d()->numberPrototype
; }
195 DatePrototype
* datePrototype() const { return d()->datePrototype
; }
196 RegExpPrototype
* regExpPrototype() const { return d()->regExpPrototype
; }
198 Structure
* argumentsStructure() const { return d()->argumentsStructure
.get(); }
199 Structure
* arrayStructure() const { return d()->arrayStructure
.get(); }
200 Structure
* booleanObjectStructure() const { return d()->booleanObjectStructure
.get(); }
201 Structure
* callbackConstructorStructure() const { return d()->callbackConstructorStructure
.get(); }
202 Structure
* callbackFunctionStructure() const { return d()->callbackFunctionStructure
.get(); }
203 Structure
* callbackObjectStructure() const { return d()->callbackObjectStructure
.get(); }
204 Structure
* dateStructure() const { return d()->dateStructure
.get(); }
205 Structure
* emptyObjectStructure() const { return d()->emptyObjectStructure
.get(); }
206 Structure
* errorStructure() const { return d()->errorStructure
.get(); }
207 Structure
* functionStructure() const { return d()->functionStructure
.get(); }
208 Structure
* numberObjectStructure() const { return d()->numberObjectStructure
.get(); }
209 Structure
* prototypeFunctionStructure() const { return d()->prototypeFunctionStructure
.get(); }
210 Structure
* regExpMatchesArrayStructure() const { return d()->regExpMatchesArrayStructure
.get(); }
211 Structure
* regExpStructure() const { return d()->regExpStructure
.get(); }
212 Structure
* stringObjectStructure() const { return d()->stringObjectStructure
.get(); }
214 void setProfileGroup(unsigned value
) { d()->profileGroup
= value
; }
215 unsigned profileGroup() const { return d()->profileGroup
; }
217 void setTimeoutTime(unsigned timeoutTime
);
218 void startTimeoutCheck();
219 void stopTimeoutCheck();
221 Debugger
* debugger() const { return d()->debugger
; }
222 void setDebugger(Debugger
* debugger
) { d()->debugger
= debugger
; }
224 virtual bool supportsProfiling() const { return false; }
226 int recursion() { return d()->recursion
; }
227 void incRecursion() { ++d()->recursion
; }
228 void decRecursion() { --d()->recursion
; }
230 ScopeChain
& globalScopeChain() { return d()->globalScopeChain
; }
232 virtual bool isGlobalObject() const { return true; }
234 virtual ExecState
* globalExec();
236 virtual bool shouldInterruptScriptBeforeTimeout() const { return false; }
237 virtual bool shouldInterruptScript() const { return true; }
239 virtual bool allowsAccessFrom(const JSGlobalObject
*) const { return true; }
241 virtual bool isDynamicScope() const;
243 HashSet
<ProgramCodeBlock
*>& codeBlocks() { return d()->codeBlocks
; }
245 void copyGlobalsFrom(RegisterFile
&);
246 void copyGlobalsTo(RegisterFile
&);
248 void resetPrototype(JSValuePtr prototype
);
250 JSGlobalData
* globalData() { return d()->globalData
.get(); }
251 JSGlobalObjectData
* d() const { return static_cast<JSGlobalObjectData
*>(JSVariableObject::d
); }
253 static PassRefPtr
<Structure
> createStructure(JSValuePtr prototype
)
255 return Structure::create(prototype
, TypeInfo(ObjectType
));
259 struct GlobalPropertyInfo
{
260 GlobalPropertyInfo(const Identifier
& i
, JSValuePtr v
, unsigned a
)
267 const Identifier identifier
;
271 void addStaticGlobals(GlobalPropertyInfo
*, int count
);
274 // FIXME: Fold reset into init.
275 void init(JSObject
* thisValue
);
276 void reset(JSValuePtr prototype
);
278 void setRegisters(Register
* registers
, Register
* registerArray
, size_t count
);
280 void* operator new(size_t); // can only be allocated with JSGlobalData
283 JSGlobalObject
* asGlobalObject(JSValuePtr
);
285 inline JSGlobalObject
* asGlobalObject(JSValuePtr value
)
287 ASSERT(asObject(value
)->isGlobalObject());
288 return static_cast<JSGlobalObject
*>(asObject(value
));
291 inline void JSGlobalObject::setRegisters(Register
* registers
, Register
* registerArray
, size_t count
)
293 JSVariableObject::setRegisters(registers
, registerArray
);
294 d()->registerArraySize
= count
;
297 inline void JSGlobalObject::addStaticGlobals(GlobalPropertyInfo
* globals
, int count
)
299 size_t oldSize
= d()->registerArraySize
;
300 size_t newSize
= oldSize
+ count
;
301 Register
* registerArray
= new Register
[newSize
];
302 if (d()->registerArray
)
303 memcpy(registerArray
+ count
, d()->registerArray
.get(), oldSize
* sizeof(Register
));
304 setRegisters(registerArray
+ newSize
, registerArray
, newSize
);
306 for (int i
= 0, index
= -static_cast<int>(oldSize
) - 1; i
< count
; ++i
, --index
) {
307 GlobalPropertyInfo
& global
= globals
[i
];
308 ASSERT(global
.attributes
& DontDelete
);
309 SymbolTableEntry
newEntry(index
, global
.attributes
);
310 symbolTable().add(global
.identifier
.ustring().rep(), newEntry
);
311 registerAt(index
) = global
.value
;
315 inline bool JSGlobalObject::getOwnPropertySlot(ExecState
* exec
, const Identifier
& propertyName
, PropertySlot
& slot
)
317 if (JSVariableObject::getOwnPropertySlot(exec
, propertyName
, slot
))
319 return symbolTableGet(propertyName
, slot
);
322 inline bool JSGlobalObject::getOwnPropertySlot(ExecState
* exec
, const Identifier
& propertyName
, PropertySlot
& slot
, bool& slotIsWriteable
)
324 if (JSVariableObject::getOwnPropertySlotForWrite(exec
, propertyName
, slot
, slotIsWriteable
))
326 return symbolTableGet(propertyName
, slot
, slotIsWriteable
);
329 inline JSGlobalObject
* ScopeChainNode::globalObject() const
331 const ScopeChainNode
* n
= this;
334 return asGlobalObject(n
->object
);
337 inline JSValuePtr
Structure::prototypeForLookup(ExecState
* exec
) const
339 if (typeInfo().type() == ObjectType
)
342 if (typeInfo().type() == StringType
)
343 return exec
->lexicalGlobalObject()->stringPrototype();
345 ASSERT(typeInfo().type() == NumberType
);
346 return exec
->lexicalGlobalObject()->numberPrototype();
349 inline StructureChain
* Structure::prototypeChain(ExecState
* exec
) const
351 // We cache our prototype chain so our clients can share it.
352 if (!isValid(exec
, m_cachedPrototypeChain
.get())) {
353 JSValuePtr prototype
= prototypeForLookup(exec
);
354 m_cachedPrototypeChain
= StructureChain::create(prototype
.isNull() ? 0 : asObject(prototype
)->structure());
356 return m_cachedPrototypeChain
.get();
359 inline bool Structure::isValid(ExecState
* exec
, StructureChain
* cachedPrototypeChain
) const
361 if (!cachedPrototypeChain
)
364 JSValuePtr prototype
= prototypeForLookup(exec
);
365 RefPtr
<Structure
>* cachedStructure
= cachedPrototypeChain
->head();
366 while(*cachedStructure
&& !prototype
.isNull()) {
367 if (asObject(prototype
)->structure() != *cachedStructure
)
370 prototype
= asObject(prototype
)->prototype();
372 return prototype
.isNull() && !*cachedStructure
;
375 inline JSGlobalObject
* ExecState::dynamicGlobalObject()
377 if (this == lexicalGlobalObject()->globalExec())
378 return lexicalGlobalObject();
380 // For any ExecState that's not a globalExec, the
381 // dynamic global object must be set since code is running
382 ASSERT(globalData().dynamicGlobalObject
);
383 return globalData().dynamicGlobalObject
;
386 class DynamicGlobalObjectScope
: Noncopyable
{
388 DynamicGlobalObjectScope(CallFrame
* callFrame
, JSGlobalObject
* dynamicGlobalObject
)
389 : m_dynamicGlobalObjectSlot(callFrame
->globalData().dynamicGlobalObject
)
390 , m_savedDynamicGlobalObject(m_dynamicGlobalObjectSlot
)
392 m_dynamicGlobalObjectSlot
= dynamicGlobalObject
;
395 ~DynamicGlobalObjectScope()
397 m_dynamicGlobalObjectSlot
= m_savedDynamicGlobalObject
;
401 JSGlobalObject
*& m_dynamicGlobalObjectSlot
;
402 JSGlobalObject
* m_savedDynamicGlobalObject
;
407 #endif // JSGlobalObject_h