Make ASan build not depend on asan.xcconfig
[WebKit-https.git] / Source / JavaScriptCore / interpreter / Register.h
1 /*
2  * Copyright (C) 2008, 2009 Apple Inc. All rights reserved.
3  *
4  * Redistribution and use in source and binary forms, with or without
5  * modification, are permitted provided that the following conditions
6  * are met:
7  *
8  * 1.  Redistributions of source code must retain the above copyright
9  *     notice, this list of conditions and the following disclaimer.
10  * 2.  Redistributions in binary form must reproduce the above copyright
11  *     notice, this list of conditions and the following disclaimer in the
12  *     documentation and/or other materials provided with the distribution.
13  * 3.  Neither the name of Apple Inc. ("Apple") nor the names of
14  *     its contributors may be used to endorse or promote products derived
15  *     from this software without specific prior written permission.
16  *
17  * THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
18  * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
19  * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
20  * DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
21  * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
22  * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
23  * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
24  * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
25  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
26  * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27  */
28
29 #ifndef Register_h
30 #define Register_h
31
32 #include "JSCJSValue.h"
33 #include <wtf/Assertions.h>
34 #include <wtf/VectorTraits.h>
35
36 namespace JSC {
37
38     class CodeBlock;
39     class ExecState;
40     class JSLexicalEnvironment;
41     class JSObject;
42     class JSScope;
43
44     typedef ExecState CallFrame;
45
46     class Register {
47         WTF_MAKE_FAST_ALLOCATED;
48     public:
49         Register();
50
51         Register(const JSValue&);
52         Register& operator=(const JSValue&);
53         JSValue jsValue() const;
54         JSValue asanUnsafeJSValue() const;
55         EncodedJSValue encodedJSValue() const;
56         
57         Register& operator=(CallFrame*);
58         Register& operator=(CodeBlock*);
59         Register& operator=(JSScope*);
60         Register& operator=(JSObject*);
61
62         int32_t i() const;
63         JSLexicalEnvironment* lexicalEnvironment() const;
64         CallFrame* callFrame() const;
65         CodeBlock* codeBlock() const;
66         JSObject* object() const;
67         JSScope* scope() const;
68         int32_t unboxedInt32() const;
69         int64_t unboxedInt52() const;
70         int64_t unboxedStrictInt52() const;
71         bool unboxedBoolean() const;
72         double unboxedDouble() const;
73         JSCell* unboxedCell() const;
74         int32_t payload() const;
75         int32_t tag() const;
76         int32_t& payload();
77         int32_t& tag();
78
79         static Register withInt(int32_t i)
80         {
81             Register r = jsNumber(i);
82             return r;
83         }
84
85     private:
86         union {
87             EncodedJSValue value;
88             CallFrame* callFrame;
89             CodeBlock* codeBlock;
90             EncodedValueDescriptor encodedValue;
91             double number;
92             int64_t integer;
93         } u;
94     };
95
96     ALWAYS_INLINE Register::Register()
97     {
98 #ifndef NDEBUG
99         *this = JSValue();
100 #endif
101     }
102
103     ALWAYS_INLINE Register::Register(const JSValue& v)
104     {
105         u.value = JSValue::encode(v);
106     }
107
108     ALWAYS_INLINE Register& Register::operator=(const JSValue& v)
109     {
110         u.value = JSValue::encode(v);
111         return *this;
112     }
113
114     // FIXME (rdar://problem/19379214): ASan only needs to be suppressed for Register::jsValue() when called from prepareOSREntry(), but there is currently no way to express this short of adding a separate copy of the function.
115     SUPPRESS_ASAN ALWAYS_INLINE JSValue Register::asanUnsafeJSValue() const
116     {
117         return JSValue::decode(u.value);
118     }
119
120     ALWAYS_INLINE JSValue Register::jsValue() const
121     {
122         return JSValue::decode(u.value);
123     }
124
125     ALWAYS_INLINE EncodedJSValue Register::encodedJSValue() const
126     {
127         return u.value;
128     }
129
130     // Interpreter functions
131
132     ALWAYS_INLINE Register& Register::operator=(CallFrame* callFrame)
133     {
134         u.callFrame = callFrame;
135         return *this;
136     }
137
138     ALWAYS_INLINE Register& Register::operator=(CodeBlock* codeBlock)
139     {
140         u.codeBlock = codeBlock;
141         return *this;
142     }
143
144     ALWAYS_INLINE int32_t Register::i() const
145     {
146         return jsValue().asInt32();
147     }
148
149     ALWAYS_INLINE CallFrame* Register::callFrame() const
150     {
151         return u.callFrame;
152     }
153     
154     ALWAYS_INLINE CodeBlock* Register::codeBlock() const
155     {
156         return u.codeBlock;
157     }
158
159     ALWAYS_INLINE int32_t Register::unboxedInt32() const
160     {
161         return payload();
162     }
163
164     ALWAYS_INLINE int64_t Register::unboxedInt52() const
165     {
166         return u.integer >> JSValue::int52ShiftAmount;
167     }
168
169     ALWAYS_INLINE int64_t Register::unboxedStrictInt52() const
170     {
171         return u.integer;
172     }
173
174     ALWAYS_INLINE bool Register::unboxedBoolean() const
175     {
176         return !!payload();
177     }
178
179     ALWAYS_INLINE double Register::unboxedDouble() const
180     {
181         return u.number;
182     }
183
184     ALWAYS_INLINE JSCell* Register::unboxedCell() const
185     {
186 #if USE(JSVALUE64)
187         return u.encodedValue.ptr;
188 #else
189         return bitwise_cast<JSCell*>(payload());
190 #endif
191     }
192
193     ALWAYS_INLINE int32_t Register::payload() const
194     {
195         return u.encodedValue.asBits.payload;
196     }
197
198     ALWAYS_INLINE int32_t Register::tag() const
199     {
200         return u.encodedValue.asBits.tag;
201     }
202
203     ALWAYS_INLINE int32_t& Register::payload()
204     {
205         return u.encodedValue.asBits.payload;
206     }
207
208     ALWAYS_INLINE int32_t& Register::tag()
209     {
210         return u.encodedValue.asBits.tag;
211     }
212
213 } // namespace JSC
214
215 namespace WTF {
216
217     template<> struct VectorTraits<JSC::Register> : VectorTraitsBase<true, JSC::Register> { };
218
219 } // namespace WTF
220
221 #endif // Register_h