/*
* Copyright (C) 2003, 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 ScopeChain_h
#define ScopeChain_h
#include "JSCell.h"
#include "Structure.h"
#include <wtf/FastAllocBase.h>
namespace JSC {
class JSGlobalData;
class JSGlobalObject;
class JSObject;
class MarkStack;
class ScopeChainIterator;
class ScopeChainNode : public JSCell {
public:
ScopeChainNode(ScopeChainNode* next, JSObject* object, JSGlobalData* globalData, JSGlobalObject* globalObject, JSObject* globalThis)
: JSCell(*globalData, globalData->scopeChainNodeStructure.get())
, globalData(globalData)
, next(*globalData, this, next)
, object(*globalData, this, object)
, globalObject(*globalData, this, globalObject)
, globalThis(*globalData, this, globalThis)
{
ASSERT(globalData);
ASSERT(globalObject);
}
JSGlobalData* globalData;
WriteBarrier<ScopeChainNode> next;
WriteBarrier<JSObject> object;
WriteBarrier<JSGlobalObject> globalObject;
WriteBarrier<JSObject> globalThis;
ScopeChainNode* push(JSObject*);
ScopeChainNode* pop();
ScopeChainIterator begin();
ScopeChainIterator end();
int localDepth();
#ifndef NDEBUG
void print();
#endif
static Structure* createStructure(JSGlobalData& globalData, JSValue proto) { return Structure::create(globalData, proto, TypeInfo(CompoundType, StructureFlags), AnonymousSlotCount, &s_info); }
virtual void markChildren(MarkStack&);
private:
static const unsigned StructureFlags = OverridesMarkChildren;
static const ClassInfo s_info;
};
inline ScopeChainNode* ScopeChainNode::push(JSObject* o)
{
ASSERT(o);
return new (globalData) ScopeChainNode(this, o, globalData, globalObject.get(), globalThis.get());
}
inline ScopeChainNode* ScopeChainNode::pop()
{
ASSERT(next);
return next.get();
}
class ScopeChainIterator {
public:
ScopeChainIterator(ScopeChainNode* node)
: m_node(node)
{
}
WriteBarrier<JSObject> const & operator*() const { return m_node->object; }
WriteBarrier<JSObject> const * operator->() const { return &(operator*()); }
ScopeChainIterator& operator++() { m_node = m_node->next.get(); return *this; }
// postfix ++ intentionally omitted
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:
ScopeChainNode* m_node;
};
inline ScopeChainIterator ScopeChainNode::begin()
{
return ScopeChainIterator(this);
}
inline ScopeChainIterator ScopeChainNode::end()
{
return ScopeChainIterator(0);
}
ALWAYS_INLINE JSGlobalData& ExecState::globalData() const
{
ASSERT(scopeChain()->globalData);
return *scopeChain()->globalData;
}
ALWAYS_INLINE JSGlobalObject* ExecState::lexicalGlobalObject() const
{
return scopeChain()->globalObject.get();
}
ALWAYS_INLINE JSObject* ExecState::globalThisValue() const
{
return scopeChain()->globalThis.get();
}
ALWAYS_INLINE ScopeChainNode* Register::scopeChain() const
{
return static_cast<ScopeChainNode*>(jsValue().asCell());
}
ALWAYS_INLINE Register& Register::operator=(ScopeChainNode* scopeChain)
{
*this = JSValue(scopeChain);
return *this;
}
} // namespace JSC
#endif // ScopeChain_h