#ifndef JSScope_h
#define JSScope_h
#include "JSObject.h"
#include "ResolveOperation.h"
namespace JSC {
class ScopeChainIterator;
class JSScope : public JSNonFinalObject {
public:
typedef JSNonFinalObject Base;
friend class LLIntOffsetsExtractor;
static size_t offsetOfNext();
JS_EXPORT_PRIVATE static JSObject* objectAtScope(JSScope*);
static JSValue resolve(CallFrame*, const Identifier&, ResolveOperations*);
static JSValue resolveBase(CallFrame*, const Identifier&, bool isStrict, ResolveOperations*, PutToBaseOperation*);
static JSValue resolveWithBase(CallFrame*, const Identifier&, Register* base, ResolveOperations*, PutToBaseOperation*);
static JSValue resolveWithThis(CallFrame*, const Identifier&, Register* base, ResolveOperations*);
static JSValue resolveGlobal(CallFrame*, const Identifier&, JSGlobalObject*, ResolveOperation*);
static void resolvePut(CallFrame*, JSValue base, const Identifier&, JSValue, PutToBaseOperation*);
static void visitChildren(JSCell*, SlotVisitor&);
bool isDynamicScope(bool& requiresDynamicChecks) const;
ScopeChainIterator begin();
ScopeChainIterator end();
JSScope* next();
int localDepth();
JSGlobalObject* globalObject();
VM* vm();
JSObject* globalThis();
protected:
JSScope(VM&, Structure*, JSScope* next);
static const unsigned StructureFlags = OverridesVisitChildren | Base::StructureFlags;
private:
WriteBarrier<JSScope> m_next;
enum ReturnValues {
ReturnValue = 1,
ReturnBase = 2,
ReturnThis = 4,
ReturnBaseAndValue = ReturnValue | ReturnBase,
ReturnThisAndValue = ReturnValue | ReturnThis,
};
enum LookupMode { UnknownResolve, KnownResolve };
template <LookupMode, ReturnValues> static JSObject* resolveContainingScopeInternal(CallFrame*, const Identifier&, PropertySlot&, ResolveOperations*, PutToBaseOperation*, bool isStrict);
template <ReturnValues> static JSObject* resolveContainingScope(CallFrame*, const Identifier&, PropertySlot&, ResolveOperations*, PutToBaseOperation*, bool isStrict);
};
inline JSScope::JSScope(VM& vm, Structure* structure, JSScope* next)
: Base(vm, structure)
, m_next(vm, this, next, WriteBarrier<JSScope>::MayBeNull)
{
}
class ScopeChainIterator {
public:
ScopeChainIterator(JSScope* node)
: m_node(node)
{
}
JSObject* get() const { return JSScope::objectAtScope(m_node); }
JSObject* operator->() const { return JSScope::objectAtScope(m_node); }
ScopeChainIterator& operator++() { m_node = m_node->next(); return *this; }
bool operator==(const ScopeChainIterator& other) const { return m_node == other.m_node; }
bool operator!=(const ScopeChainIterator& other) const { return m_node != other.m_node; }
private:
JSScope* m_node;
};
inline ScopeChainIterator JSScope::begin()
{
return ScopeChainIterator(this);
}
inline ScopeChainIterator JSScope::end()
{
return ScopeChainIterator(0);
}
inline JSScope* JSScope::next()
{
return m_next.get();
}
inline JSGlobalObject* JSScope::globalObject()
{
return structure()->globalObject();
}
inline VM* JSScope::vm()
{
return Heap::heap(this)->vm();
}
inline Register& Register::operator=(JSScope* scope)
{
*this = JSValue(scope);
return *this;
}
inline JSScope* Register::scope() const
{
return jsCast<JSScope*>(jsValue());
}
inline VM& ExecState::vm() const
{
ASSERT(scope()->vm());
return *scope()->vm();
}
inline JSGlobalObject* ExecState::lexicalGlobalObject() const
{
return scope()->globalObject();
}
inline JSObject* ExecState::globalThisValue() const
{
return scope()->globalThis();
}
inline size_t JSScope::offsetOfNext()
{
return OBJECT_OFFSETOF(JSScope, m_next);
}
}
#endif // JSScope_h