| /* |
| * Copyright (C) 2007 Eric Seidel <eric@webkit.org> |
| * Copyright (C) 2007, 2008, 2009 Apple Inc. All rights reserved. |
| * |
| * This library is free software; you can redistribute it and/or |
| * modify it under the terms of the GNU Library General Public |
| * License as published by the Free Software Foundation; either |
| * version 2 of the License, or (at your option) any later version. |
| * |
| * This library is distributed in the hope that it will be useful, |
| * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
| * Library General Public License for more details. |
| * |
| * You should have received a copy of the GNU Library General Public License |
| * along with this library; see the file COPYING.LIB. If not, write to |
| * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, |
| * Boston, MA 02110-1301, USA. |
| * |
| */ |
| |
| #ifndef JSGlobalObject_h |
| #define JSGlobalObject_h |
| |
| #include "ArrayAllocationProfile.h" |
| #include "JSArray.h" |
| #include "JSGlobalData.h" |
| #include "JSSegmentedVariableObject.h" |
| #include "JSWeakObjectMapRefInternal.h" |
| #include "NumberPrototype.h" |
| #include "SpecialPointer.h" |
| #include "StringPrototype.h" |
| #include "StructureChain.h" |
| #include "Watchpoint.h" |
| #include <wtf/HashSet.h> |
| #include <wtf/OwnPtr.h> |
| #include <wtf/RandomNumber.h> |
| |
| namespace JSC { |
| |
| class ArrayPrototype; |
| class BooleanPrototype; |
| class DatePrototype; |
| class Debugger; |
| class ErrorConstructor; |
| class ErrorPrototype; |
| class EvalCodeBlock; |
| class EvalExecutable; |
| class FunctionCodeBlock; |
| class FunctionExecutable; |
| class FunctionPrototype; |
| class GetterSetter; |
| class GlobalCodeBlock; |
| class JSStack; |
| class LLIntOffsetsExtractor; |
| class NativeErrorConstructor; |
| class ProgramCodeBlock; |
| class ProgramExecutable; |
| class RegExpConstructor; |
| class RegExpPrototype; |
| class SourceCode; |
| struct ActivationStackNode; |
| struct HashTable; |
| |
| typedef Vector<ExecState*, 16> ExecStateStack; |
| |
| struct GlobalObjectMethodTable { |
| typedef bool (*AllowsAccessFromFunctionPtr)(const JSGlobalObject*, ExecState*); |
| AllowsAccessFromFunctionPtr allowsAccessFrom; |
| |
| typedef bool (*SupportsProfilingFunctionPtr)(const JSGlobalObject*); |
| SupportsProfilingFunctionPtr supportsProfiling; |
| |
| typedef bool (*SupportsRichSourceInfoFunctionPtr)(const JSGlobalObject*); |
| SupportsRichSourceInfoFunctionPtr supportsRichSourceInfo; |
| |
| typedef bool (*ShouldInterruptScriptFunctionPtr)(const JSGlobalObject*); |
| ShouldInterruptScriptFunctionPtr shouldInterruptScript; |
| |
| typedef bool (*JavaScriptExperimentsEnabledFunctionPtr)(const JSGlobalObject*); |
| JavaScriptExperimentsEnabledFunctionPtr javaScriptExperimentsEnabled; |
| }; |
| |
| #if defined(COBALT) |
| class ReportEvalCallback { |
| public: |
| virtual void reportEval() = 0; |
| }; |
| #endif |
| |
| class JSGlobalObject : public JSSegmentedVariableObject { |
| private: |
| typedef HashSet<RefPtr<OpaqueJSWeakObjectMap> > WeakMapSet; |
| |
| struct JSGlobalObjectRareData { |
| JSGlobalObjectRareData() |
| : profileGroup(0) |
| { |
| } |
| |
| WeakMapSet weakMaps; |
| unsigned profileGroup; |
| }; |
| |
| protected: |
| |
| Register m_globalCallFrame[JSStack::CallFrameHeaderSize]; |
| |
| WriteBarrier<JSObject> m_globalThis; |
| |
| WriteBarrier<RegExpConstructor> m_regExpConstructor; |
| WriteBarrier<ErrorConstructor> m_errorConstructor; |
| WriteBarrier<NativeErrorConstructor> m_evalErrorConstructor; |
| WriteBarrier<NativeErrorConstructor> m_rangeErrorConstructor; |
| WriteBarrier<NativeErrorConstructor> m_referenceErrorConstructor; |
| WriteBarrier<NativeErrorConstructor> m_syntaxErrorConstructor; |
| WriteBarrier<NativeErrorConstructor> m_typeErrorConstructor; |
| WriteBarrier<NativeErrorConstructor> m_URIErrorConstructor; |
| |
| WriteBarrier<JSFunction> m_evalFunction; |
| WriteBarrier<JSFunction> m_callFunction; |
| WriteBarrier<JSFunction> m_applyFunction; |
| WriteBarrier<GetterSetter> m_throwTypeErrorGetterSetter; |
| |
| WriteBarrier<ObjectPrototype> m_objectPrototype; |
| WriteBarrier<FunctionPrototype> m_functionPrototype; |
| WriteBarrier<ArrayPrototype> m_arrayPrototype; |
| WriteBarrier<BooleanPrototype> m_booleanPrototype; |
| WriteBarrier<StringPrototype> m_stringPrototype; |
| WriteBarrier<NumberPrototype> m_numberPrototype; |
| WriteBarrier<DatePrototype> m_datePrototype; |
| WriteBarrier<RegExpPrototype> m_regExpPrototype; |
| WriteBarrier<ErrorPrototype> m_errorPrototype; |
| |
| WriteBarrier<Structure> m_withScopeStructure; |
| WriteBarrier<Structure> m_strictEvalActivationStructure; |
| WriteBarrier<Structure> m_activationStructure; |
| WriteBarrier<Structure> m_nameScopeStructure; |
| WriteBarrier<Structure> m_argumentsStructure; |
| |
| // Lists the actual structures used for having these particular indexing shapes. |
| WriteBarrier<Structure> m_originalArrayStructureForIndexingShape[NumberOfIndexingShapes]; |
| // Lists the structures we should use during allocation for these particular indexing shapes. |
| WriteBarrier<Structure> m_arrayStructureForIndexingShapeDuringAllocation[NumberOfIndexingShapes]; |
| |
| WriteBarrier<Structure> m_booleanObjectStructure; |
| WriteBarrier<Structure> m_callbackConstructorStructure; |
| WriteBarrier<Structure> m_callbackFunctionStructure; |
| WriteBarrier<Structure> m_callbackObjectStructure; |
| WriteBarrier<Structure> m_dateStructure; |
| WriteBarrier<Structure> m_emptyObjectStructure; |
| WriteBarrier<Structure> m_nullPrototypeObjectStructure; |
| WriteBarrier<Structure> m_errorStructure; |
| WriteBarrier<Structure> m_functionStructure; |
| WriteBarrier<Structure> m_boundFunctionStructure; |
| WriteBarrier<Structure> m_namedFunctionStructure; |
| PropertyOffset m_functionNameOffset; |
| WriteBarrier<Structure> m_numberObjectStructure; |
| WriteBarrier<Structure> m_privateNameStructure; |
| WriteBarrier<Structure> m_regExpMatchesArrayStructure; |
| WriteBarrier<Structure> m_regExpStructure; |
| WriteBarrier<Structure> m_stringObjectStructure; |
| WriteBarrier<Structure> m_internalFunctionStructure; |
| |
| void* m_specialPointers[Special::TableSize]; // Special pointers used by the LLInt and JIT. |
| |
| Debugger* m_debugger; |
| |
| RefPtr<WatchpointSet> m_masqueradesAsUndefinedWatchpoint; |
| RefPtr<WatchpointSet> m_havingABadTimeWatchpoint; |
| |
| OwnPtr<JSGlobalObjectRareData> m_rareData; |
| |
| WeakRandom m_weakRandom; |
| |
| bool m_evalEnabled; |
| String m_evalDisabledErrorMessage; |
| bool m_experimentsEnabled; |
| #if defined(COBALT) |
| ReportEvalCallback* m_reportEvalCallback; |
| #endif |
| |
| static JS_EXPORTDATA const GlobalObjectMethodTable s_globalObjectMethodTable; |
| const GlobalObjectMethodTable* m_globalObjectMethodTable; |
| |
| void createRareDataIfNeeded() |
| { |
| if (m_rareData) |
| return; |
| m_rareData = adoptPtr(new JSGlobalObjectRareData); |
| } |
| |
| public: |
| typedef JSSegmentedVariableObject Base; |
| |
| static JSGlobalObject* create(JSGlobalData& globalData, Structure* structure) |
| { |
| JSGlobalObject* globalObject = new (NotNull, allocateCell<JSGlobalObject>(globalData.heap)) JSGlobalObject(globalData, structure); |
| globalObject->finishCreation(globalData); |
| globalData.heap.addFinalizer(globalObject, destroy); |
| return globalObject; |
| } |
| |
| DECLARE_EXPORTED_CLASSINFO(); |
| |
| bool hasDebugger() const { return m_debugger; } |
| bool hasProfiler() const { return globalObjectMethodTable()->supportsProfiling(this); } |
| |
| protected: |
| JS_EXPORT_PRIVATE explicit JSGlobalObject(JSGlobalData&, Structure*, const GlobalObjectMethodTable* = 0); |
| |
| void finishCreation(JSGlobalData& globalData) |
| { |
| Base::finishCreation(globalData); |
| structure()->setGlobalObject(globalData, this); |
| m_experimentsEnabled = m_globalObjectMethodTable->javaScriptExperimentsEnabled(this); |
| init(this); |
| } |
| |
| void finishCreation(JSGlobalData& globalData, JSObject* thisValue) |
| { |
| Base::finishCreation(globalData); |
| structure()->setGlobalObject(globalData, this); |
| m_experimentsEnabled = m_globalObjectMethodTable->javaScriptExperimentsEnabled(this); |
| init(thisValue); |
| } |
| |
| public: |
| JS_EXPORT_PRIVATE ~JSGlobalObject(); |
| JS_EXPORT_PRIVATE static void destroy(JSCell*); |
| // We don't need a destructor because we use a finalizer instead. |
| static const bool needsDestruction = false; |
| |
| JS_EXPORT_PRIVATE static void visitChildrenInternal(JSCell*, SlotVisitor&); |
| static void visitChildren(JSCell* cell, SlotVisitor& visitor) { |
| return visitChildrenInternal(cell, visitor); |
| } |
| |
| JS_EXPORT_PRIVATE static bool getOwnPropertySlotInternal(JSCell*, ExecState*, PropertyName, PropertySlot&); |
| static bool getOwnPropertySlot(JSCell* cell, ExecState* exec, PropertyName name, PropertySlot& slot) { |
| return getOwnPropertySlotInternal(cell, exec, name, slot); |
| } |
| |
| JS_EXPORT_PRIVATE static bool getOwnPropertyDescriptorInternal(JSObject*, ExecState*, PropertyName, PropertyDescriptor&); |
| static bool getOwnPropertyDescriptor(JSObject* object, ExecState* exec, PropertyName name, PropertyDescriptor& descriptor) { |
| return getOwnPropertyDescriptorInternal(object, exec, name, descriptor); |
| } |
| |
| bool hasOwnPropertyForWrite(ExecState*, PropertyName); |
| |
| JS_EXPORT_PRIVATE static void putInternal(JSCell*, ExecState*, PropertyName, JSValue, PutPropertySlot&); |
| static void put(JSCell* cell, ExecState* exec, PropertyName name, JSValue value, PutPropertySlot& slot) { |
| return putInternal(cell, exec, name, value, slot); |
| } |
| |
| JS_EXPORT_PRIVATE static void putDirectVirtualInternal(JSObject*, ExecState*, PropertyName, JSValue, unsigned attributes); |
| static void putDirectVirtual(JSObject* object, ExecState* exec, PropertyName name, JSValue value, unsigned attributes) { |
| return putDirectVirtualInternal(object, exec, name, value, attributes); |
| } |
| |
| JS_EXPORT_PRIVATE static void defineGetter(JSObject*, ExecState*, PropertyName, JSObject* getterFunc, unsigned attributes); |
| JS_EXPORT_PRIVATE static void defineSetter(JSObject*, ExecState*, PropertyName, JSObject* setterFunc, unsigned attributes); |
| |
| JS_EXPORT_PRIVATE static bool defineOwnPropertyInternal(JSObject*, ExecState*, PropertyName, PropertyDescriptor&, bool shouldThrow); |
| static bool defineOwnProperty(JSObject* object, ExecState* exec, PropertyName name, PropertyDescriptor& descriptor, bool shouldThrow) { |
| return defineOwnPropertyInternal(object, exec, name, descriptor, shouldThrow); |
| } |
| |
| // We use this in the code generator as we perform symbol table |
| // lookups prior to initializing the properties |
| bool symbolTableHasProperty(PropertyName); |
| |
| // The following accessors return pristine values, even if a script |
| // replaces the global object's associated property. |
| |
| RegExpConstructor* regExpConstructor() const { return m_regExpConstructor.get(); } |
| |
| ErrorConstructor* errorConstructor() const { return m_errorConstructor.get(); } |
| NativeErrorConstructor* evalErrorConstructor() const { return m_evalErrorConstructor.get(); } |
| NativeErrorConstructor* rangeErrorConstructor() const { return m_rangeErrorConstructor.get(); } |
| NativeErrorConstructor* referenceErrorConstructor() const { return m_referenceErrorConstructor.get(); } |
| NativeErrorConstructor* syntaxErrorConstructor() const { return m_syntaxErrorConstructor.get(); } |
| NativeErrorConstructor* typeErrorConstructor() const { return m_typeErrorConstructor.get(); } |
| NativeErrorConstructor* URIErrorConstructor() const { return m_URIErrorConstructor.get(); } |
| |
| JSFunction* evalFunction() const { return m_evalFunction.get(); } |
| JSFunction* callFunction() const { return m_callFunction.get(); } |
| JSFunction* applyFunction() const { return m_applyFunction.get(); } |
| GetterSetter* throwTypeErrorGetterSetter(ExecState* exec) |
| { |
| if (!m_throwTypeErrorGetterSetter) |
| createThrowTypeError(exec); |
| return m_throwTypeErrorGetterSetter.get(); |
| } |
| |
| ObjectPrototype* objectPrototype() const { return m_objectPrototype.get(); } |
| FunctionPrototype* functionPrototype() const { return m_functionPrototype.get(); } |
| ArrayPrototype* arrayPrototype() const { return m_arrayPrototype.get(); } |
| BooleanPrototype* booleanPrototype() const { return m_booleanPrototype.get(); } |
| StringPrototype* stringPrototype() const { return m_stringPrototype.get(); } |
| NumberPrototype* numberPrototype() const { return m_numberPrototype.get(); } |
| DatePrototype* datePrototype() const { return m_datePrototype.get(); } |
| RegExpPrototype* regExpPrototype() const { return m_regExpPrototype.get(); } |
| ErrorPrototype* errorPrototype() const { return m_errorPrototype.get(); } |
| |
| Structure* withScopeStructure() const { return m_withScopeStructure.get(); } |
| Structure* strictEvalActivationStructure() const { return m_strictEvalActivationStructure.get(); } |
| Structure* activationStructure() const { return m_activationStructure.get(); } |
| Structure* nameScopeStructure() const { return m_nameScopeStructure.get(); } |
| Structure* argumentsStructure() const { return m_argumentsStructure.get(); } |
| Structure* originalArrayStructureForIndexingType(IndexingType indexingType) const |
| { |
| ASSERT(indexingType & IsArray); |
| return m_originalArrayStructureForIndexingShape[(indexingType & IndexingShapeMask) >> IndexingShapeShift].get(); |
| } |
| Structure* arrayStructureForIndexingTypeDuringAllocation(IndexingType indexingType) const |
| { |
| ASSERT(indexingType & IsArray); |
| return m_arrayStructureForIndexingShapeDuringAllocation[(indexingType & IndexingShapeMask) >> IndexingShapeShift].get(); |
| } |
| Structure* arrayStructureForProfileDuringAllocation(ArrayAllocationProfile* profile) const |
| { |
| return arrayStructureForIndexingTypeDuringAllocation(ArrayAllocationProfile::selectIndexingTypeFor(profile)); |
| } |
| |
| bool isOriginalArrayStructure(Structure* structure) |
| { |
| return originalArrayStructureForIndexingType(structure->indexingType() | IsArray) == structure; |
| } |
| |
| Structure* booleanObjectStructure() const { return m_booleanObjectStructure.get(); } |
| Structure* callbackConstructorStructure() const { return m_callbackConstructorStructure.get(); } |
| Structure* callbackFunctionStructure() const { return m_callbackFunctionStructure.get(); } |
| Structure* callbackObjectStructure() const { return m_callbackObjectStructure.get(); } |
| Structure* dateStructure() const { return m_dateStructure.get(); } |
| Structure* emptyObjectStructure() const { return m_emptyObjectStructure.get(); } |
| Structure* nullPrototypeObjectStructure() const { return m_nullPrototypeObjectStructure.get(); } |
| Structure* errorStructure() const { return m_errorStructure.get(); } |
| Structure* functionStructure() const { return m_functionStructure.get(); } |
| Structure* boundFunctionStructure() const { return m_boundFunctionStructure.get(); } |
| Structure* namedFunctionStructure() const { return m_namedFunctionStructure.get(); } |
| PropertyOffset functionNameOffset() const { return m_functionNameOffset; } |
| Structure* numberObjectStructure() const { return m_numberObjectStructure.get(); } |
| Structure* privateNameStructure() const { return m_privateNameStructure.get(); } |
| Structure* internalFunctionStructure() const { return m_internalFunctionStructure.get(); } |
| Structure* regExpMatchesArrayStructure() const { return m_regExpMatchesArrayStructure.get(); } |
| Structure* regExpStructure() const { return m_regExpStructure.get(); } |
| Structure* stringObjectStructure() const { return m_stringObjectStructure.get(); } |
| |
| void* actualPointerFor(Special::Pointer pointer) |
| { |
| ASSERT(pointer < Special::TableSize); |
| return m_specialPointers[pointer]; |
| } |
| |
| WatchpointSet* masqueradesAsUndefinedWatchpoint() { return m_masqueradesAsUndefinedWatchpoint.get(); } |
| WatchpointSet* havingABadTimeWatchpoint() { return m_havingABadTimeWatchpoint.get(); } |
| |
| bool isHavingABadTime() const |
| { |
| return m_havingABadTimeWatchpoint->hasBeenInvalidated(); |
| } |
| |
| void haveABadTime(JSGlobalData&); |
| |
| bool arrayPrototypeChainIsSane(); |
| |
| void setProfileGroup(unsigned value) { createRareDataIfNeeded(); m_rareData->profileGroup = value; } |
| unsigned profileGroup() const |
| { |
| if (!m_rareData) |
| return 0; |
| return m_rareData->profileGroup; |
| } |
| |
| Debugger* debugger() const { return m_debugger; } |
| void setDebugger(Debugger* debugger) { m_debugger = debugger; } |
| |
| const GlobalObjectMethodTable* globalObjectMethodTable() const { return m_globalObjectMethodTable; } |
| |
| static bool allowsAccessFrom(const JSGlobalObject*, ExecState*) { return true; } |
| static bool supportsProfiling(const JSGlobalObject*) { return false; } |
| static bool supportsRichSourceInfo(const JSGlobalObject*) { return true; } |
| |
| JS_EXPORT_PRIVATE ExecState* globalExec(); |
| |
| static bool shouldInterruptScript(const JSGlobalObject*) { return true; } |
| static bool javaScriptExperimentsEnabled(const JSGlobalObject*) { return false; } |
| |
| bool isDynamicScope(bool& requiresDynamicChecks) const; |
| |
| #if defined(COBALT) |
| void setReportEvalCallback(ReportEvalCallback* callback) { |
| m_reportEvalCallback = callback; |
| } |
| bool checkEvalEnabledAndReport() const { |
| if (m_reportEvalCallback) { |
| m_reportEvalCallback->reportEval(); |
| } |
| return m_evalEnabled; |
| } |
| #else |
| bool evalEnabled() const { return m_evalEnabled; } |
| #endif |
| const String& evalDisabledErrorMessage() const { return m_evalDisabledErrorMessage; } |
| void setEvalEnabled(bool enabled, const String& errorMessage = String()) |
| { |
| m_evalEnabled = enabled; |
| m_evalDisabledErrorMessage = errorMessage; |
| } |
| |
| void resetPrototype(JSGlobalData&, JSValue prototype); |
| |
| JSGlobalData& globalData() const { return *Heap::heap(this)->globalData(); } |
| JSObject* globalThis() const; |
| |
| static Structure* createStructure(JSGlobalData& globalData, JSValue prototype) |
| { |
| return Structure::create(globalData, 0, prototype, TypeInfo(GlobalObjectType, StructureFlags), &s_info); |
| } |
| |
| void registerWeakMap(OpaqueJSWeakObjectMap* map) |
| { |
| createRareDataIfNeeded(); |
| m_rareData->weakMaps.add(map); |
| } |
| |
| void unregisterWeakMap(OpaqueJSWeakObjectMap* map) |
| { |
| if (m_rareData) |
| m_rareData->weakMaps.remove(map); |
| } |
| |
| double weakRandomNumber() { return m_weakRandom.get(); } |
| unsigned weakRandomInteger() { return m_weakRandom.getUint32(); } |
| |
| UnlinkedProgramCodeBlock* createProgramCodeBlock(CallFrame*, ProgramExecutable*, JSObject** exception); |
| UnlinkedEvalCodeBlock* createEvalCodeBlock(CallFrame*, EvalExecutable*, JSObject** exception); |
| UnlinkedFunctionExecutable* createFunctionExecutableFromGlobalCode(CallFrame*, const Identifier&, const SourceCode&, JSObject** exception); |
| |
| protected: |
| |
| static const unsigned StructureFlags = OverridesGetOwnPropertySlot | OverridesVisitChildren | OverridesGetPropertyNames | Base::StructureFlags; |
| |
| struct GlobalPropertyInfo { |
| GlobalPropertyInfo(const Identifier& i, JSValue v, unsigned a) |
| : identifier(i) |
| , value(v) |
| , attributes(a) |
| { |
| } |
| |
| const Identifier identifier; |
| JSValue value; |
| unsigned attributes; |
| }; |
| JS_EXPORT_PRIVATE void addStaticGlobals(GlobalPropertyInfo*, int count); |
| |
| JS_EXPORT_PRIVATE static JSC::JSObject* toThisObjectInternal(JSC::JSCell*, JSC::ExecState*); |
| static JSC::JSObject* toThisObject(JSC::JSCell* cell, JSC::ExecState* exec) { |
| return toThisObjectInternal(cell, exec); |
| } |
| |
| JS_EXPORT_PRIVATE void setGlobalThis(JSGlobalData&, JSObject* globalThis); |
| |
| private: |
| friend class LLIntOffsetsExtractor; |
| |
| // FIXME: Fold reset into init. |
| JS_EXPORT_PRIVATE void init(JSObject* thisValue); |
| void reset(JSValue prototype); |
| |
| void createThrowTypeError(ExecState*); |
| |
| JS_EXPORT_PRIVATE static void clearRareData(JSCell*); |
| }; |
| |
| JSGlobalObject* asGlobalObject(JSValue); |
| |
| inline JSGlobalObject* asGlobalObject(JSValue value) |
| { |
| ASSERT(asObject(value)->isGlobalObject()); |
| return jsCast<JSGlobalObject*>(asObject(value)); |
| } |
| |
| inline bool JSGlobalObject::hasOwnPropertyForWrite(ExecState* exec, PropertyName propertyName) |
| { |
| PropertySlot slot; |
| if (Base::getOwnPropertySlot(this, exec, propertyName, slot)) |
| return true; |
| bool slotIsWriteable; |
| return symbolTableGet(this, propertyName, slot, slotIsWriteable); |
| } |
| |
| inline bool JSGlobalObject::symbolTableHasProperty(PropertyName propertyName) |
| { |
| SymbolTableEntry entry = symbolTable()->inlineGet(propertyName.publicName()); |
| return !entry.isNull(); |
| } |
| |
| inline JSValue Structure::prototypeForLookup(JSGlobalObject* globalObject) const |
| { |
| if (isObject()) |
| return m_prototype.get(); |
| |
| ASSERT(typeInfo().type() == StringType); |
| return globalObject->stringPrototype(); |
| } |
| |
| inline JSValue Structure::prototypeForLookup(ExecState* exec) const |
| { |
| return prototypeForLookup(exec->lexicalGlobalObject()); |
| } |
| |
| inline StructureChain* Structure::prototypeChain(JSGlobalData& globalData, JSGlobalObject* globalObject) const |
| { |
| // We cache our prototype chain so our clients can share it. |
| if (!isValid(globalObject, m_cachedPrototypeChain.get())) { |
| JSValue prototype = prototypeForLookup(globalObject); |
| m_cachedPrototypeChain.set(globalData, this, StructureChain::create(globalData, prototype.isNull() ? 0 : asObject(prototype)->structure())); |
| } |
| return m_cachedPrototypeChain.get(); |
| } |
| |
| inline StructureChain* Structure::prototypeChain(ExecState* exec) const |
| { |
| return prototypeChain(exec->globalData(), exec->lexicalGlobalObject()); |
| } |
| |
| inline bool Structure::isValid(JSGlobalObject* globalObject, StructureChain* cachedPrototypeChain) const |
| { |
| if (!cachedPrototypeChain) |
| return false; |
| |
| JSValue prototype = prototypeForLookup(globalObject); |
| WriteBarrier<Structure>* cachedStructure = cachedPrototypeChain->head(); |
| while(*cachedStructure && !prototype.isNull()) { |
| if (asObject(prototype)->structure() != cachedStructure->get()) |
| return false; |
| ++cachedStructure; |
| prototype = asObject(prototype)->prototype(); |
| } |
| return prototype.isNull() && !*cachedStructure; |
| } |
| |
| inline bool Structure::isValid(ExecState* exec, StructureChain* cachedPrototypeChain) const |
| { |
| return isValid(exec->lexicalGlobalObject(), cachedPrototypeChain); |
| } |
| |
| inline JSGlobalObject* ExecState::dynamicGlobalObject() |
| { |
| if (this == lexicalGlobalObject()->globalExec()) |
| return lexicalGlobalObject(); |
| |
| // For any ExecState that's not a globalExec, the |
| // dynamic global object must be set since code is running |
| ASSERT(globalData().dynamicGlobalObject); |
| return globalData().dynamicGlobalObject; |
| } |
| |
| inline JSObject* constructEmptyObject(ExecState* exec, JSGlobalObject* globalObject) |
| { |
| return constructEmptyObject(exec, globalObject->emptyObjectStructure()); |
| } |
| |
| inline JSObject* constructEmptyObject(ExecState* exec) |
| { |
| return constructEmptyObject(exec, exec->lexicalGlobalObject()); |
| } |
| |
| inline JSArray* constructEmptyArray(ExecState* exec, ArrayAllocationProfile* profile, JSGlobalObject* globalObject, unsigned initialLength = 0) |
| { |
| return ArrayAllocationProfile::updateLastAllocationFor(profile, JSArray::create(exec->globalData(), initialLength >= MIN_SPARSE_ARRAY_INDEX ? globalObject->arrayStructureForIndexingTypeDuringAllocation(ArrayWithArrayStorage) : globalObject->arrayStructureForProfileDuringAllocation(profile), initialLength)); |
| } |
| |
| inline JSArray* constructEmptyArray(ExecState* exec, ArrayAllocationProfile* profile, unsigned initialLength = 0) |
| { |
| return constructEmptyArray(exec, profile, exec->lexicalGlobalObject(), initialLength); |
| } |
| |
| inline JSArray* constructArray(ExecState* exec, ArrayAllocationProfile* profile, JSGlobalObject* globalObject, const ArgList& values) |
| { |
| return ArrayAllocationProfile::updateLastAllocationFor(profile, constructArray(exec, globalObject->arrayStructureForProfileDuringAllocation(profile), values)); |
| } |
| |
| inline JSArray* constructArray(ExecState* exec, ArrayAllocationProfile* profile, const ArgList& values) |
| { |
| return constructArray(exec, profile, exec->lexicalGlobalObject(), values); |
| } |
| |
| inline JSArray* constructArray(ExecState* exec, ArrayAllocationProfile* profile, JSGlobalObject* globalObject, const JSValue* values, unsigned length) |
| { |
| return ArrayAllocationProfile::updateLastAllocationFor(profile, constructArray(exec, globalObject->arrayStructureForProfileDuringAllocation(profile), values, length)); |
| } |
| |
| inline JSArray* constructArray(ExecState* exec, ArrayAllocationProfile* profile, const JSValue* values, unsigned length) |
| { |
| return constructArray(exec, profile, exec->lexicalGlobalObject(), values, length); |
| } |
| |
| class DynamicGlobalObjectScope { |
| WTF_MAKE_NONCOPYABLE(DynamicGlobalObjectScope); |
| public: |
| JS_EXPORT_PRIVATE DynamicGlobalObjectScope(JSGlobalData&, JSGlobalObject*); |
| |
| ~DynamicGlobalObjectScope() |
| { |
| m_dynamicGlobalObjectSlot = m_savedDynamicGlobalObject; |
| } |
| |
| private: |
| JSGlobalObject*& m_dynamicGlobalObjectSlot; |
| JSGlobalObject* m_savedDynamicGlobalObject; |
| }; |
| |
| inline bool JSGlobalObject::isDynamicScope(bool&) const |
| { |
| return true; |
| } |
| |
| inline JSObject* JSScope::globalThis() |
| { |
| return globalObject()->globalThis(); |
| } |
| |
| inline JSObject* JSGlobalObject::globalThis() const |
| { |
| return m_globalThis.get(); |
| } |
| |
| } // namespace JSC |
| |
| #endif // JSGlobalObject_h |