]> git.saurik.com Git - apple/javascriptcore.git/blobdiff - interpreter/Register.h
JavaScriptCore-521.tar.gz
[apple/javascriptcore.git] / interpreter / Register.h
diff --git a/interpreter/Register.h b/interpreter/Register.h
new file mode 100644 (file)
index 0000000..ff36bbc
--- /dev/null
@@ -0,0 +1,299 @@
+/*
+ * Copyright (C) 2008 Apple Inc. All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions
+ * are met:
+ *
+ * 1.  Redistributions of source code must retain the above copyright
+ *     notice, this list of conditions and the following disclaimer.
+ * 2.  Redistributions in binary form must reproduce the above copyright
+ *     notice, this list of conditions and the following disclaimer in the
+ *     documentation and/or other materials provided with the distribution.
+ * 3.  Neither the name of Apple Computer, Inc. ("Apple") nor the names of
+ *     its contributors may be used to endorse or promote products derived
+ *     from this software without specific prior written permission.
+ *
+ * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
+ * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
+ * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
+ * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
+ * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
+ * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
+ * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
+ * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
+ * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
+ * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+#ifndef Register_h
+#define Register_h
+
+#include "JSValue.h"
+#include <wtf/VectorTraits.h>
+
+namespace JSC {
+
+    class Arguments;
+    class CodeBlock;
+    class ExecState;
+    class JSActivation;
+    class JSFunction;
+    class JSPropertyNameIterator;
+    class ScopeChainNode;
+
+    struct Instruction;
+
+    typedef ExecState CallFrame;
+
+    class Register {
+    public:
+        Register();
+        Register(JSValuePtr);
+
+        JSValuePtr jsValue(CallFrame*) const;
+        JSValuePtr getJSValue() const;
+
+        bool marked() const;
+        void mark();
+        
+    private:
+        friend class ExecState;
+        friend class Interpreter;
+
+        // Only CallFrame and Interpreter should use these functions.
+
+        Register(intptr_t);
+
+        Register(JSActivation*);
+        Register(Arguments*);
+        Register(CallFrame*);
+        Register(CodeBlock*);
+        Register(JSFunction*);
+        Register(JSPropertyNameIterator*);
+        Register(ScopeChainNode*);
+        Register(Instruction*);
+
+        intptr_t i() const;
+        void* v() const;
+
+        JSActivation* activation() const;
+        Arguments* arguments() const;
+        CallFrame* callFrame() const;
+        CodeBlock* codeBlock() const;
+        JSFunction* function() const;
+        JSPropertyNameIterator* propertyNameIterator() const;
+        ScopeChainNode* scopeChain() const;
+        Instruction* vPC() const;
+
+        union {
+            intptr_t i;
+            void* v;
+            JSValueEncodedAsPointer* value;
+
+            JSActivation* activation;
+            Arguments* arguments;
+            CallFrame* callFrame;
+            CodeBlock* codeBlock;
+            JSFunction* function;
+            JSPropertyNameIterator* propertyNameIterator;
+            ScopeChainNode* scopeChain;
+            Instruction* vPC;
+        } u;
+
+#ifndef NDEBUG
+        enum {
+            EmptyType,
+
+            IntType,
+            ValueType,
+
+            ActivationType,
+            ArgumentsType,
+            CallFrameType,
+            CodeBlockType,
+            FunctionType,
+            InstructionType,
+            PropertyNameIteratorType,
+            RegisterType,
+            ScopeChainNodeType
+        } m_type;
+#endif
+    };
+
+#ifndef NDEBUG
+    #define SET_TYPE(type) m_type = (type)
+    // FIXME: The CTI code to put value into registers doesn't set m_type.
+    // Once it does, we can turn this assertion back on.
+    #define ASSERT_TYPE(type)
+#else
+    #define SET_TYPE(type)
+    #define ASSERT_TYPE(type)
+#endif
+
+    ALWAYS_INLINE Register::Register()
+    {
+#ifndef NDEBUG
+        SET_TYPE(EmptyType);
+        u.value = JSValuePtr::encode(noValue());
+#endif
+    }
+
+    ALWAYS_INLINE Register::Register(JSValuePtr v)
+    {
+        SET_TYPE(ValueType);
+        u.value = JSValuePtr::encode(v);
+    }
+
+    // This function is scaffolding for legacy clients. It will eventually go away.
+    ALWAYS_INLINE JSValuePtr Register::jsValue(CallFrame*) const
+    {
+        // Once registers hold doubles, this function will allocate a JSValue*
+        // if the register doesn't hold one already. 
+        ASSERT_TYPE(ValueType);
+        return JSValuePtr::decode(u.value);
+    }
+    
+    ALWAYS_INLINE JSValuePtr Register::getJSValue() const
+    {
+        ASSERT_TYPE(JSValueType);
+        return JSValuePtr::decode(u.value);
+    }
+    
+    ALWAYS_INLINE bool Register::marked() const
+    {
+        return getJSValue().marked();
+    }
+
+    ALWAYS_INLINE void Register::mark()
+    {
+        getJSValue().mark();
+    }
+    
+    // Interpreter functions
+
+    ALWAYS_INLINE Register::Register(Arguments* arguments)
+    {
+        SET_TYPE(ArgumentsType);
+        u.arguments = arguments;
+    }
+
+    ALWAYS_INLINE Register::Register(JSActivation* activation)
+    {
+        SET_TYPE(ActivationType);
+        u.activation = activation;
+    }
+
+    ALWAYS_INLINE Register::Register(CallFrame* callFrame)
+    {
+        SET_TYPE(CallFrameType);
+        u.callFrame = callFrame;
+    }
+
+    ALWAYS_INLINE Register::Register(CodeBlock* codeBlock)
+    {
+        SET_TYPE(CodeBlockType);
+        u.codeBlock = codeBlock;
+    }
+
+    ALWAYS_INLINE Register::Register(JSFunction* function)
+    {
+        SET_TYPE(FunctionType);
+        u.function = function;
+    }
+
+    ALWAYS_INLINE Register::Register(Instruction* vPC)
+    {
+        SET_TYPE(InstructionType);
+        u.vPC = vPC;
+    }
+
+    ALWAYS_INLINE Register::Register(ScopeChainNode* scopeChain)
+    {
+        SET_TYPE(ScopeChainNodeType);
+        u.scopeChain = scopeChain;
+    }
+
+    ALWAYS_INLINE Register::Register(JSPropertyNameIterator* propertyNameIterator)
+    {
+        SET_TYPE(PropertyNameIteratorType);
+        u.propertyNameIterator = propertyNameIterator;
+    }
+
+    ALWAYS_INLINE Register::Register(intptr_t i)
+    {
+        SET_TYPE(IntType);
+        u.i = i;
+    }
+
+    ALWAYS_INLINE intptr_t Register::i() const
+    {
+        ASSERT_TYPE(IntType);
+        return u.i;
+    }
+    
+    ALWAYS_INLINE void* Register::v() const
+    {
+        return u.v;
+    }
+
+    ALWAYS_INLINE JSActivation* Register::activation() const
+    {
+        ASSERT_TYPE(ActivationType);
+        return u.activation;
+    }
+    
+    ALWAYS_INLINE Arguments* Register::arguments() const
+    {
+        ASSERT_TYPE(ArgumentsType);
+        return u.arguments;
+    }
+    
+    ALWAYS_INLINE CallFrame* Register::callFrame() const
+    {
+        ASSERT_TYPE(CallFrameType);
+        return u.callFrame;
+    }
+    
+    ALWAYS_INLINE CodeBlock* Register::codeBlock() const
+    {
+        ASSERT_TYPE(CodeBlockType);
+        return u.codeBlock;
+    }
+    
+    ALWAYS_INLINE JSFunction* Register::function() const
+    {
+        ASSERT_TYPE(FunctionType);
+        return u.function;
+    }
+    
+    ALWAYS_INLINE JSPropertyNameIterator* Register::propertyNameIterator() const
+    {
+        ASSERT_TYPE(PropertyNameIteratorType);
+        return u.propertyNameIterator;
+    }
+    
+    ALWAYS_INLINE ScopeChainNode* Register::scopeChain() const
+    {
+        ASSERT_TYPE(ScopeChainNodeType);
+        return u.scopeChain;
+    }
+    
+    ALWAYS_INLINE Instruction* Register::vPC() const
+    {
+        ASSERT_TYPE(InstructionType);
+        return u.vPC;
+    }
+
+    #undef SET_TYPE
+    #undef ASSERT_TYPE
+
+} // namespace JSC
+
+namespace WTF {
+
+    template<> struct VectorTraits<JSC::Register> : VectorTraitsBase<true, JSC::Register> { };
+
+} // namespace WTF
+
+#endif // Register_h