blob: c86f316ef315b4949fa94a428aee2cb08e228e0c [file] [log] [blame]
ulan@chromium.org65a89c22012-02-14 11:46:07 +00001// Copyright 2012 the V8 project authors. All rights reserved.
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +00002// Redistribution and use in source and binary forms, with or without
3// modification, are permitted provided that the following conditions are
4// met:
5//
6// * Redistributions of source code must retain the above copyright
7// notice, this list of conditions and the following disclaimer.
8// * Redistributions in binary form must reproduce the above
9// copyright notice, this list of conditions and the following
10// disclaimer in the documentation and/or other materials provided
11// with the distribution.
12// * Neither the name of Google Inc. nor the names of its
13// contributors may be used to endorse or promote products derived
14// from this software without specific prior written permission.
15//
16// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
17// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
18// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
19// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
20// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
21// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
22// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
23// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
24// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
25// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
26// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27
28#ifndef V8_IC_H_
29#define V8_IC_H_
30
kasperl@chromium.orga5551262010-12-07 12:49:48 +000031#include "macro-assembler.h"
danno@chromium.org40cb8782011-05-25 07:58:50 +000032#include "type-info.h"
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +000033
kasperl@chromium.org71affb52009-05-26 05:44:31 +000034namespace v8 {
35namespace internal {
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +000036
kmillikin@chromium.org13bd2942009-12-16 15:36:05 +000037
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +000038// IC_UTIL_LIST defines all utility functions called from generated
39// inline caching code. The argument for the macro, ICU, is the function name.
sgjesse@chromium.org0b6db592009-07-30 14:48:31 +000040#define IC_UTIL_LIST(ICU) \
41 ICU(LoadIC_Miss) \
42 ICU(KeyedLoadIC_Miss) \
ager@chromium.orgea91cc52011-05-23 06:06:11 +000043 ICU(KeyedLoadIC_MissForceGeneric) \
sgjesse@chromium.org0b6db592009-07-30 14:48:31 +000044 ICU(CallIC_Miss) \
lrn@chromium.org1af7e1b2010-06-07 11:12:01 +000045 ICU(KeyedCallIC_Miss) \
sgjesse@chromium.org0b6db592009-07-30 14:48:31 +000046 ICU(StoreIC_Miss) \
ager@chromium.org5c838252010-02-19 08:53:10 +000047 ICU(StoreIC_ArrayLength) \
sgjesse@chromium.org0b6db592009-07-30 14:48:31 +000048 ICU(SharedStoreIC_ExtendStorage) \
49 ICU(KeyedStoreIC_Miss) \
ager@chromium.orgea91cc52011-05-23 06:06:11 +000050 ICU(KeyedStoreIC_MissForceGeneric) \
51 ICU(KeyedStoreIC_Slow) \
sgjesse@chromium.org0b6db592009-07-30 14:48:31 +000052 /* Utilities for IC stubs. */ \
53 ICU(LoadCallbackProperty) \
54 ICU(StoreCallbackProperty) \
55 ICU(LoadPropertyWithInterceptorOnly) \
56 ICU(LoadPropertyWithInterceptorForLoad) \
57 ICU(LoadPropertyWithInterceptorForCall) \
ager@chromium.org5c838252010-02-19 08:53:10 +000058 ICU(KeyedLoadPropertyWithInterceptor) \
ager@chromium.orgce5e87b2010-03-10 10:24:18 +000059 ICU(StoreInterceptorProperty) \
danno@chromium.org40cb8782011-05-25 07:58:50 +000060 ICU(UnaryOp_Patch) \
61 ICU(BinaryOp_Patch) \
ricow@chromium.org9fa09672011-07-25 11:05:35 +000062 ICU(CompareIC_Miss) \
63 ICU(ToBoolean_Patch)
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +000064//
kasperl@chromium.orge959c182009-07-27 08:59:04 +000065// IC is the base class for LoadIC, StoreIC, CallIC, KeyedLoadIC,
66// and KeyedStoreIC.
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +000067//
68class IC {
69 public:
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +000070 // The ids for utility called from the generated code.
71 enum UtilityId {
72 #define CONST_NAME(name) k##name,
73 IC_UTIL_LIST(CONST_NAME)
74 #undef CONST_NAME
75 kUtilityCount
76 };
77
78 // Looks up the address of the named utility.
79 static Address AddressFromUtilityId(UtilityId id);
80
81 // Alias the inline cache state type to make the IC code more readable.
82 typedef InlineCacheState State;
83
84 // The IC code is either invoked with no extra frames on the stack
85 // or with a single extra frame for supporting calls.
86 enum FrameDepth {
87 NO_EXTRA_FRAME = 0,
88 EXTRA_CALL_FRAME = 1
89 };
90
91 // Construct the IC structure with the given number of extra
92 // JavaScript frames on the stack.
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +000093 IC(FrameDepth depth, Isolate* isolate);
ricow@chromium.org64e3a4b2011-12-13 08:07:27 +000094 virtual ~IC() {}
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +000095
96 // Get the call-site target; used for determining the state.
ricow@chromium.org64e3a4b2011-12-13 08:07:27 +000097 Code* target() const { return GetTargetAtAddress(address()); }
98 inline Address address() const;
99
100 virtual bool IsGeneric() const { return false; }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000101
ager@chromium.orgce5e87b2010-03-10 10:24:18 +0000102 // Compute the current IC state based on the target stub, receiver and name.
103 static State StateFrom(Code* target, Object* receiver, Object* name);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000104
105 // Clear the inline cache to initial state.
106 static void Clear(Address address);
107
108 // Computes the reloc info for this IC. This is a fairly expensive
109 // operation as it has to search through the heap to find the code
110 // object that contains this IC site.
ager@chromium.org236ad962008-09-25 09:45:57 +0000111 RelocInfo::Mode ComputeMode();
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000112
113 // Returns if this IC is for contextual (no explicit receiver)
114 // access to properties.
kmillikin@chromium.org13bd2942009-12-16 15:36:05 +0000115 bool IsContextual(Handle<Object> receiver) {
116 if (receiver->IsGlobalObject()) {
117 return SlowIsContextual();
118 } else {
119 ASSERT(!SlowIsContextual());
120 return false;
121 }
122 }
123
124 bool SlowIsContextual() {
ager@chromium.org236ad962008-09-25 09:45:57 +0000125 return ComputeMode() == RelocInfo::CODE_TARGET_CONTEXT;
126 }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000127
kmillikin@chromium.org69ea3962010-07-05 11:01:40 +0000128 // Determines which map must be used for keeping the code stub.
129 // These methods should not be called with undefined or null.
130 static inline InlineCacheHolderFlag GetCodeCacheForObject(Object* object,
131 JSObject* holder);
132 static inline InlineCacheHolderFlag GetCodeCacheForObject(JSObject* object,
133 JSObject* holder);
ricow@chromium.org65fae842010-08-25 15:26:24 +0000134 static inline JSObject* GetCodeCacheHolder(Object* object,
135 InlineCacheHolderFlag holder);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000136
137 protected:
138 Address fp() const { return fp_; }
139 Address pc() const { return *pc_address_; }
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000140 Isolate* isolate() const { return isolate_; }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000141
ager@chromium.org65dad4b2009-04-23 08:48:43 +0000142#ifdef ENABLE_DEBUGGER_SUPPORT
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000143 // Computes the address in the original code when the code running is
144 // containing break points (calls to DebugBreakXXX builtins).
ricow@chromium.org64e3a4b2011-12-13 08:07:27 +0000145 Address OriginalCodeAddress() const;
ager@chromium.org65dad4b2009-04-23 08:48:43 +0000146#endif
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000147
148 // Set the call-site target.
149 void set_target(Code* code) { SetTargetAtAddress(address(), code); }
150
151#ifdef DEBUG
ricow@chromium.org64e3a4b2011-12-13 08:07:27 +0000152 char TransitionMarkFromState(IC::State state);
153
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000154 void TraceIC(const char* type,
155 Handle<Object> name,
156 State old_state,
lrn@chromium.org34e60782011-09-15 07:25:40 +0000157 Code* new_target);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000158#endif
159
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000160 Failure* TypeError(const char* type,
161 Handle<Object> object,
162 Handle<Object> key);
163 Failure* ReferenceError(const char* type, Handle<String> name);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000164
165 // Access the target code for the given IC address.
166 static inline Code* GetTargetAtAddress(Address address);
167 static inline void SetTargetAtAddress(Address address, Code* target);
jkummerow@chromium.orgf7a58842012-02-21 10:08:21 +0000168 static void PostPatching(Address address, Code* target, Code* old_target);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000169
170 private:
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000171 // Frame pointer for the frame that uses (calls) the IC.
172 Address fp_;
173
174 // All access to the program counter of an IC structure is indirect
175 // to make the code GC safe. This feature is crucial since
176 // GetProperty and SetProperty are called and they in turn might
177 // invoke the garbage collector.
178 Address* pc_address_;
mads.s.ager31e71382008-08-13 09:32:07 +0000179
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000180 Isolate* isolate_;
181
mads.s.ager31e71382008-08-13 09:32:07 +0000182 DISALLOW_IMPLICIT_CONSTRUCTORS(IC);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000183};
184
185
186// An IC_Utility encapsulates IC::UtilityId. It exists mainly because you
187// cannot make forward declarations to an enum.
188class IC_Utility {
189 public:
190 explicit IC_Utility(IC::UtilityId id)
191 : address_(IC::AddressFromUtilityId(id)), id_(id) {}
192
193 Address address() const { return address_; }
194
195 IC::UtilityId id() const { return id_; }
196 private:
197 Address address_;
198 IC::UtilityId id_;
199};
200
201
lrn@chromium.org1af7e1b2010-06-07 11:12:01 +0000202class CallICBase: public IC {
danno@chromium.org40cb8782011-05-25 07:58:50 +0000203 public:
204 class Contextual: public BitField<bool, 0, 1> {};
205 class StringStubState: public BitField<StringStubFeedback, 1, 1> {};
206
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000207 // Returns a JSFunction or a Failure.
lrn@chromium.org303ada72010-10-27 09:33:13 +0000208 MUST_USE_RESULT MaybeObject* LoadFunction(State state,
erik.corry@gmail.com0511e242011-01-19 11:11:08 +0000209 Code::ExtraICState extra_ic_state,
lrn@chromium.org303ada72010-10-27 09:33:13 +0000210 Handle<Object> object,
211 Handle<String> name);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000212
lrn@chromium.org1af7e1b2010-06-07 11:12:01 +0000213 protected:
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000214 CallICBase(Code::Kind kind, Isolate* isolate)
215 : IC(EXTRA_CALL_FRAME, isolate), kind_(kind) {}
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000216
erik.corry@gmail.com0511e242011-01-19 11:11:08 +0000217 bool TryUpdateExtraICState(LookupResult* lookup,
218 Handle<Object> object,
219 Code::ExtraICState* extra_ic_state);
220
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000221 // Compute a monomorphic stub if possible, otherwise return a null handle.
222 Handle<Code> ComputeMonomorphicStub(LookupResult* lookup,
223 State state,
224 Code::ExtraICState extra_state,
225 Handle<Object> object,
226 Handle<String> name);
erik.corry@gmail.com0511e242011-01-19 11:11:08 +0000227
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000228 // Update the inline cache and the global stub cache based on the lookup
229 // result.
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000230 void UpdateCaches(LookupResult* lookup,
231 State state,
erik.corry@gmail.com0511e242011-01-19 11:11:08 +0000232 Code::ExtraICState extra_ic_state,
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000233 Handle<Object> object,
234 Handle<String> name);
235
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000236 // Returns a JSFunction if the object can be called as a function, and
237 // patches the stack to be ready for the call. Otherwise, it returns the
238 // undefined value.
239 Handle<Object> TryCallAsFunction(Handle<Object> object);
ager@chromium.org9258b6b2008-09-11 09:11:10 +0000240
fschneider@chromium.org3a5fd782011-02-24 10:10:44 +0000241 void ReceiverToObjectIfRequired(Handle<Object> callee, Handle<Object> object);
sgjesse@chromium.orgb302e562010-02-03 11:26:59 +0000242
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000243 static void Clear(Address address, Code* target);
danno@chromium.org40cb8782011-05-25 07:58:50 +0000244
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000245 // Platform-specific code generation functions used by both call and
246 // keyed call.
247 static void GenerateMiss(MacroAssembler* masm,
248 int argc,
249 IC::UtilityId id,
250 Code::ExtraICState extra_state);
251
252 static void GenerateNormal(MacroAssembler* masm, int argc);
253
254 static void GenerateMonomorphicCacheProbe(MacroAssembler* masm,
255 int argc,
256 Code::Kind kind,
257 Code::ExtraICState extra_state);
258
259 Code::Kind kind_;
260
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000261 friend class IC;
262};
263
264
lrn@chromium.org1af7e1b2010-06-07 11:12:01 +0000265class CallIC: public CallICBase {
266 public:
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000267 explicit CallIC(Isolate* isolate) : CallICBase(Code::CALL_IC, isolate) {
268 ASSERT(target()->is_call_stub());
269 }
lrn@chromium.org1af7e1b2010-06-07 11:12:01 +0000270
271 // Code generator routines.
danno@chromium.org40cb8782011-05-25 07:58:50 +0000272 static void GenerateInitialize(MacroAssembler* masm,
273 int argc,
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000274 Code::ExtraICState extra_state) {
275 GenerateMiss(masm, argc, extra_state);
lrn@chromium.org1af7e1b2010-06-07 11:12:01 +0000276 }
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000277
danno@chromium.org40cb8782011-05-25 07:58:50 +0000278 static void GenerateMiss(MacroAssembler* masm,
279 int argc,
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000280 Code::ExtraICState extra_state) {
281 CallICBase::GenerateMiss(masm, argc, IC::kCallIC_Miss, extra_state);
282 }
283
danno@chromium.org40cb8782011-05-25 07:58:50 +0000284 static void GenerateMegamorphic(MacroAssembler* masm,
285 int argc,
286 Code::ExtraICState extra_ic_state);
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000287
288 static void GenerateNormal(MacroAssembler* masm, int argc) {
289 CallICBase::GenerateNormal(masm, argc);
290 GenerateMiss(masm, argc, Code::kNoExtraICState);
291 }
lrn@chromium.org1af7e1b2010-06-07 11:12:01 +0000292};
293
294
295class KeyedCallIC: public CallICBase {
296 public:
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000297 explicit KeyedCallIC(Isolate* isolate)
298 : CallICBase(Code::KEYED_CALL_IC, isolate) {
lrn@chromium.org1af7e1b2010-06-07 11:12:01 +0000299 ASSERT(target()->is_keyed_call_stub());
300 }
301
lrn@chromium.org303ada72010-10-27 09:33:13 +0000302 MUST_USE_RESULT MaybeObject* LoadFunction(State state,
303 Handle<Object> object,
304 Handle<Object> key);
lrn@chromium.org1af7e1b2010-06-07 11:12:01 +0000305
306 // Code generator routines.
307 static void GenerateInitialize(MacroAssembler* masm, int argc) {
308 GenerateMiss(masm, argc);
309 }
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000310
311 static void GenerateMiss(MacroAssembler* masm, int argc) {
312 CallICBase::GenerateMiss(masm, argc, IC::kKeyedCallIC_Miss,
313 Code::kNoExtraICState);
314 }
315
lrn@chromium.org1af7e1b2010-06-07 11:12:01 +0000316 static void GenerateMegamorphic(MacroAssembler* masm, int argc);
317 static void GenerateNormal(MacroAssembler* masm, int argc);
whesse@chromium.org7b260152011-06-20 15:33:18 +0000318 static void GenerateNonStrictArguments(MacroAssembler* masm, int argc);
lrn@chromium.org1af7e1b2010-06-07 11:12:01 +0000319};
320
321
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000322class LoadIC: public IC {
323 public:
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000324 explicit LoadIC(Isolate* isolate) : IC(NO_EXTRA_FRAME, isolate) {
325 ASSERT(target()->is_load_stub());
326 }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000327
lrn@chromium.org303ada72010-10-27 09:33:13 +0000328 MUST_USE_RESULT MaybeObject* Load(State state,
329 Handle<Object> object,
330 Handle<String> name);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000331
332 // Code generator routines.
ager@chromium.org5c838252010-02-19 08:53:10 +0000333 static void GenerateInitialize(MacroAssembler* masm) { GenerateMiss(masm); }
334 static void GeneratePreMonomorphic(MacroAssembler* masm) {
335 GenerateMiss(masm);
336 }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000337 static void GenerateMiss(MacroAssembler* masm);
338 static void GenerateMegamorphic(MacroAssembler* masm);
339 static void GenerateNormal(MacroAssembler* masm);
340
341 // Specialized code generator routines.
342 static void GenerateArrayLength(MacroAssembler* masm);
ager@chromium.org378b34e2011-01-28 08:04:38 +0000343 static void GenerateStringLength(MacroAssembler* masm,
344 bool support_wrappers);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000345 static void GenerateFunctionPrototype(MacroAssembler* masm);
346
347 private:
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000348 // Update the inline cache and the global stub cache based on the
349 // lookup result.
350 void UpdateCaches(LookupResult* lookup,
351 State state,
352 Handle<Object> object,
353 Handle<String> name);
354
355 // Stub accessors.
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000356 Handle<Code> megamorphic_stub() {
357 return isolate()->builtins()->LoadIC_Megamorphic();
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000358 }
359 static Code* initialize_stub() {
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000360 return Isolate::Current()->builtins()->builtin(
fschneider@chromium.org7979bbb2011-03-28 10:47:03 +0000361 Builtins::kLoadIC_Initialize);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000362 }
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000363 Handle<Code> pre_monomorphic_stub() {
364 return isolate()->builtins()->LoadIC_PreMonomorphic();
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000365 }
366
367 static void Clear(Address address, Code* target);
ager@chromium.org5ec48922009-05-05 07:25:34 +0000368
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000369 friend class IC;
370};
371
372
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000373class KeyedIC: public IC {
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000374 public:
rossberg@chromium.orgb4b2aa62011-10-13 09:49:59 +0000375 enum StubKind {
376 LOAD,
377 STORE_NO_TRANSITION,
378 STORE_TRANSITION_SMI_TO_OBJECT,
379 STORE_TRANSITION_SMI_TO_DOUBLE,
ulan@chromium.org65a89c22012-02-14 11:46:07 +0000380 STORE_TRANSITION_DOUBLE_TO_OBJECT,
svenpanne@chromium.org830d30c2012-05-29 13:20:14 +0000381 STORE_TRANSITION_HOLEY_SMI_TO_OBJECT,
382 STORE_TRANSITION_HOLEY_SMI_TO_DOUBLE,
383 STORE_TRANSITION_HOLEY_DOUBLE_TO_OBJECT,
ulan@chromium.org65a89c22012-02-14 11:46:07 +0000384 STORE_AND_GROW_NO_TRANSITION,
385 STORE_AND_GROW_TRANSITION_SMI_TO_OBJECT,
386 STORE_AND_GROW_TRANSITION_SMI_TO_DOUBLE,
svenpanne@chromium.org830d30c2012-05-29 13:20:14 +0000387 STORE_AND_GROW_TRANSITION_DOUBLE_TO_OBJECT,
388 STORE_AND_GROW_TRANSITION_HOLEY_SMI_TO_OBJECT,
389 STORE_AND_GROW_TRANSITION_HOLEY_SMI_TO_DOUBLE,
390 STORE_AND_GROW_TRANSITION_HOLEY_DOUBLE_TO_OBJECT
rossberg@chromium.orgb4b2aa62011-10-13 09:49:59 +0000391 };
ulan@chromium.org65a89c22012-02-14 11:46:07 +0000392
393 static const int kGrowICDelta = STORE_AND_GROW_NO_TRANSITION -
394 STORE_NO_TRANSITION;
395 STATIC_ASSERT(kGrowICDelta ==
396 STORE_AND_GROW_TRANSITION_SMI_TO_OBJECT -
397 STORE_TRANSITION_SMI_TO_OBJECT);
398 STATIC_ASSERT(kGrowICDelta ==
399 STORE_AND_GROW_TRANSITION_SMI_TO_DOUBLE -
400 STORE_TRANSITION_SMI_TO_DOUBLE);
401 STATIC_ASSERT(kGrowICDelta ==
402 STORE_AND_GROW_TRANSITION_DOUBLE_TO_OBJECT -
403 STORE_TRANSITION_DOUBLE_TO_OBJECT);
404
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000405 explicit KeyedIC(Isolate* isolate) : IC(NO_EXTRA_FRAME, isolate) {}
406 virtual ~KeyedIC() {}
407
ulan@chromium.org65a89c22012-02-14 11:46:07 +0000408 static inline KeyedAccessGrowMode GetGrowModeFromStubKind(
409 StubKind stub_kind) {
410 return (stub_kind >= STORE_AND_GROW_NO_TRANSITION)
411 ? ALLOW_JSARRAY_GROWTH
412 : DO_NOT_ALLOW_JSARRAY_GROWTH;
413 }
414
415 static inline StubKind GetGrowStubKind(StubKind stub_kind) {
416 ASSERT(stub_kind != LOAD);
417 if (stub_kind < STORE_AND_GROW_NO_TRANSITION) {
418 stub_kind = static_cast<StubKind>(static_cast<int>(stub_kind) +
419 kGrowICDelta);
420 }
421 return stub_kind;
422 }
423
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000424 virtual Handle<Code> GetElementStubWithoutMapCheck(
sgjesse@chromium.org6db88712011-07-11 11:41:22 +0000425 bool is_js_array,
ulan@chromium.org65a89c22012-02-14 11:46:07 +0000426 ElementsKind elements_kind,
427 KeyedAccessGrowMode grow_mode) = 0;
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000428
429 protected:
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000430 virtual Handle<Code> string_stub() {
431 return Handle<Code>::null();
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000432 }
433
434 virtual Code::Kind kind() const = 0;
435
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000436 Handle<Code> ComputeStub(Handle<JSObject> receiver,
rossberg@chromium.orgb4b2aa62011-10-13 09:49:59 +0000437 StubKind stub_kind,
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000438 StrictModeFlag strict_mode,
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000439 Handle<Code> default_stub);
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000440
ulan@chromium.org65a89c22012-02-14 11:46:07 +0000441 virtual Handle<Code> ComputePolymorphicStub(
442 MapHandleList* receiver_maps,
443 StrictModeFlag strict_mode,
444 KeyedAccessGrowMode grow_mode) = 0;
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000445
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000446 Handle<Code> ComputeMonomorphicStubWithoutMapCheck(
447 Handle<Map> receiver_map,
ulan@chromium.org65a89c22012-02-14 11:46:07 +0000448 StrictModeFlag strict_mode,
449 KeyedAccessGrowMode grow_mode);
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000450
rossberg@chromium.orgb4b2aa62011-10-13 09:49:59 +0000451 private:
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000452 void GetReceiverMapsForStub(Handle<Code> stub, MapHandleList* result);
rossberg@chromium.orgb4b2aa62011-10-13 09:49:59 +0000453
mmassi@chromium.org7028c052012-06-13 11:51:58 +0000454 Handle<Code> ComputeMonomorphicStub(Handle<Map> receiver_map,
rossberg@chromium.orgb4b2aa62011-10-13 09:49:59 +0000455 StubKind stub_kind,
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000456 StrictModeFlag strict_mode,
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000457 Handle<Code> default_stub);
rossberg@chromium.orgb4b2aa62011-10-13 09:49:59 +0000458
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000459 Handle<Map> ComputeTransitionedMap(Handle<JSObject> receiver,
460 StubKind stub_kind);
rossberg@chromium.orgb4b2aa62011-10-13 09:49:59 +0000461
462 static bool IsTransitionStubKind(StubKind stub_kind) {
ulan@chromium.org65a89c22012-02-14 11:46:07 +0000463 return stub_kind > STORE_NO_TRANSITION &&
464 stub_kind != STORE_AND_GROW_NO_TRANSITION;
465 }
466
467 static bool IsGrowStubKind(StubKind stub_kind) {
468 return stub_kind >= STORE_AND_GROW_NO_TRANSITION;
rossberg@chromium.orgb4b2aa62011-10-13 09:49:59 +0000469 }
mmassi@chromium.org7028c052012-06-13 11:51:58 +0000470
471 static StubKind GetNoTransitionStubKind(StubKind stub_kind) {
472 if (!IsTransitionStubKind(stub_kind)) return stub_kind;
473 if (IsGrowStubKind(stub_kind)) return STORE_AND_GROW_NO_TRANSITION;
474 return STORE_NO_TRANSITION;
475 }
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000476};
477
478
479class KeyedLoadIC: public KeyedIC {
480 public:
481 explicit KeyedLoadIC(Isolate* isolate) : KeyedIC(isolate) {
482 ASSERT(target()->is_keyed_load_stub());
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000483 }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000484
lrn@chromium.org303ada72010-10-27 09:33:13 +0000485 MUST_USE_RESULT MaybeObject* Load(State state,
486 Handle<Object> object,
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000487 Handle<Object> key,
488 bool force_generic_stub);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000489
490 // Code generator routines.
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000491 static void GenerateMiss(MacroAssembler* masm, bool force_generic);
ager@chromium.org5c838252010-02-19 08:53:10 +0000492 static void GenerateRuntimeGetProperty(MacroAssembler* masm);
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000493 static void GenerateInitialize(MacroAssembler* masm) {
494 GenerateMiss(masm, false);
495 }
ager@chromium.org5c838252010-02-19 08:53:10 +0000496 static void GeneratePreMonomorphic(MacroAssembler* masm) {
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000497 GenerateMiss(masm, false);
ager@chromium.org5c838252010-02-19 08:53:10 +0000498 }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000499 static void GenerateGeneric(MacroAssembler* masm);
fschneider@chromium.org0c20e672010-01-14 15:28:53 +0000500 static void GenerateString(MacroAssembler* masm);
ager@chromium.org5c838252010-02-19 08:53:10 +0000501 static void GenerateIndexedInterceptor(MacroAssembler* masm);
whesse@chromium.org7b260152011-06-20 15:33:18 +0000502 static void GenerateNonStrictArguments(MacroAssembler* masm);
ager@chromium.org3811b432009-10-28 14:53:37 +0000503
kasperl@chromium.orgeac059f2010-01-25 11:02:06 +0000504 // Bit mask to be tested against bit field for the cases when
505 // generic stub should go into slow case.
506 // Access check is necessary explicitly since generic stub does not perform
507 // map checks.
508 static const int kSlowCaseBitFieldMask =
509 (1 << Map::kIsAccessCheckNeeded) | (1 << Map::kHasIndexedInterceptor);
510
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000511 virtual Handle<Code> GetElementStubWithoutMapCheck(
sgjesse@chromium.org6db88712011-07-11 11:41:22 +0000512 bool is_js_array,
ulan@chromium.org65a89c22012-02-14 11:46:07 +0000513 ElementsKind elements_kind,
514 KeyedAccessGrowMode grow_mode);
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000515
ricow@chromium.org64e3a4b2011-12-13 08:07:27 +0000516 virtual bool IsGeneric() const {
517 return target() == *generic_stub();
518 }
519
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000520 protected:
521 virtual Code::Kind kind() const { return Code::KEYED_LOAD_IC; }
522
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000523 virtual Handle<Code> ComputePolymorphicStub(MapHandleList* receiver_maps,
ulan@chromium.org65a89c22012-02-14 11:46:07 +0000524 StrictModeFlag strict_mode,
525 KeyedAccessGrowMode grow_mode);
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000526
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000527 virtual Handle<Code> string_stub() {
528 return isolate()->builtins()->KeyedLoadIC_String();
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000529 }
530
kmillikin@chromium.org4111b802010-05-03 10:34:42 +0000531 private:
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000532 // Update the inline cache.
533 void UpdateCaches(LookupResult* lookup,
534 State state,
535 Handle<Object> object,
536 Handle<String> name);
537
538 // Stub accessors.
539 static Code* initialize_stub() {
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000540 return Isolate::Current()->builtins()->builtin(
fschneider@chromium.org7979bbb2011-03-28 10:47:03 +0000541 Builtins::kKeyedLoadIC_Initialize);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000542 }
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000543 Handle<Code> megamorphic_stub() {
544 return isolate()->builtins()->KeyedLoadIC_Generic();
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000545 }
ricow@chromium.org64e3a4b2011-12-13 08:07:27 +0000546 Handle<Code> generic_stub() const {
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000547 return isolate()->builtins()->KeyedLoadIC_Generic();
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000548 }
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000549 Handle<Code> pre_monomorphic_stub() {
550 return isolate()->builtins()->KeyedLoadIC_PreMonomorphic();
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000551 }
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000552 Handle<Code> indexed_interceptor_stub() {
553 return isolate()->builtins()->KeyedLoadIC_IndexedInterceptor();
ager@chromium.org5c838252010-02-19 08:53:10 +0000554 }
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000555 Handle<Code> non_strict_arguments_stub() {
556 return isolate()->builtins()->KeyedLoadIC_NonStrictArguments();
whesse@chromium.org7b260152011-06-20 15:33:18 +0000557 }
ager@chromium.org5c838252010-02-19 08:53:10 +0000558
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000559 static void Clear(Address address, Code* target);
christian.plesner.hansen@gmail.com37abdec2009-01-06 14:43:28 +0000560
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000561 friend class IC;
562};
563
564
565class StoreIC: public IC {
566 public:
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000567 explicit StoreIC(Isolate* isolate) : IC(NO_EXTRA_FRAME, isolate) {
568 ASSERT(target()->is_store_stub());
569 }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000570
lrn@chromium.org303ada72010-10-27 09:33:13 +0000571 MUST_USE_RESULT MaybeObject* Store(State state,
ager@chromium.org9ee27ae2011-03-02 13:43:26 +0000572 StrictModeFlag strict_mode,
lrn@chromium.org303ada72010-10-27 09:33:13 +0000573 Handle<Object> object,
574 Handle<String> name,
575 Handle<Object> value);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000576
577 // Code generators for stub routines. Only called once at startup.
sgjesse@chromium.orgb302e562010-02-03 11:26:59 +0000578 static void GenerateInitialize(MacroAssembler* masm) { GenerateMiss(masm); }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000579 static void GenerateMiss(MacroAssembler* masm);
sgjesse@chromium.org496c03a2011-02-14 12:05:43 +0000580 static void GenerateMegamorphic(MacroAssembler* masm,
ager@chromium.org9ee27ae2011-03-02 13:43:26 +0000581 StrictModeFlag strict_mode);
ager@chromium.org5c838252010-02-19 08:53:10 +0000582 static void GenerateArrayLength(MacroAssembler* masm);
kmillikin@chromium.org69ea3962010-07-05 11:01:40 +0000583 static void GenerateNormal(MacroAssembler* masm);
ager@chromium.org9ee27ae2011-03-02 13:43:26 +0000584 static void GenerateGlobalProxy(MacroAssembler* masm,
585 StrictModeFlag strict_mode);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000586
587 private:
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000588 // Update the inline cache and the global stub cache based on the
589 // lookup result.
590 void UpdateCaches(LookupResult* lookup,
sgjesse@chromium.org496c03a2011-02-14 12:05:43 +0000591 State state,
ager@chromium.org9ee27ae2011-03-02 13:43:26 +0000592 StrictModeFlag strict_mode,
sgjesse@chromium.org496c03a2011-02-14 12:05:43 +0000593 Handle<JSObject> receiver,
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000594 Handle<String> name,
595 Handle<Object> value);
596
ager@chromium.org9ee27ae2011-03-02 13:43:26 +0000597 void set_target(Code* code) {
598 // Strict mode must be preserved across IC patching.
ulan@chromium.org65a89c22012-02-14 11:46:07 +0000599 ASSERT(Code::GetStrictMode(code->extra_ic_state()) ==
600 Code::GetStrictMode(target()->extra_ic_state()));
ager@chromium.org9ee27ae2011-03-02 13:43:26 +0000601 IC::set_target(code);
602 }
603
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000604 // Stub accessors.
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000605 Code* megamorphic_stub() {
606 return isolate()->builtins()->builtin(
fschneider@chromium.org7979bbb2011-03-28 10:47:03 +0000607 Builtins::kStoreIC_Megamorphic);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000608 }
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000609 Code* megamorphic_stub_strict() {
610 return isolate()->builtins()->builtin(
fschneider@chromium.org7979bbb2011-03-28 10:47:03 +0000611 Builtins::kStoreIC_Megamorphic_Strict);
sgjesse@chromium.org496c03a2011-02-14 12:05:43 +0000612 }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000613 static Code* initialize_stub() {
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000614 return Isolate::Current()->builtins()->builtin(
fschneider@chromium.org7979bbb2011-03-28 10:47:03 +0000615 Builtins::kStoreIC_Initialize);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000616 }
sgjesse@chromium.org496c03a2011-02-14 12:05:43 +0000617 static Code* initialize_stub_strict() {
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000618 return Isolate::Current()->builtins()->builtin(
fschneider@chromium.org7979bbb2011-03-28 10:47:03 +0000619 Builtins::kStoreIC_Initialize_Strict);
sgjesse@chromium.org496c03a2011-02-14 12:05:43 +0000620 }
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000621 Handle<Code> global_proxy_stub() {
622 return isolate()->builtins()->StoreIC_GlobalProxy();
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000623 }
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000624 Handle<Code> global_proxy_stub_strict() {
625 return isolate()->builtins()->StoreIC_GlobalProxy_Strict();
sgjesse@chromium.org496c03a2011-02-14 12:05:43 +0000626 }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000627
628 static void Clear(Address address, Code* target);
fschneider@chromium.orged78ffd2010-07-21 11:05:19 +0000629
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000630 friend class IC;
631};
632
633
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000634class KeyedStoreIC: public KeyedIC {
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000635 public:
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000636 explicit KeyedStoreIC(Isolate* isolate) : KeyedIC(isolate) {
637 ASSERT(target()->is_keyed_store_stub());
638 }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000639
lrn@chromium.org303ada72010-10-27 09:33:13 +0000640 MUST_USE_RESULT MaybeObject* Store(State state,
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000641 StrictModeFlag strict_mode,
lrn@chromium.org303ada72010-10-27 09:33:13 +0000642 Handle<Object> object,
643 Handle<Object> name,
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000644 Handle<Object> value,
645 bool force_generic);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000646
647 // Code generators for stub routines. Only called once at startup.
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000648 static void GenerateInitialize(MacroAssembler* masm) {
649 GenerateMiss(masm, false);
650 }
651 static void GenerateMiss(MacroAssembler* masm, bool force_generic);
652 static void GenerateSlow(MacroAssembler* masm);
ager@chromium.org9ee27ae2011-03-02 13:43:26 +0000653 static void GenerateRuntimeSetProperty(MacroAssembler* masm,
654 StrictModeFlag strict_mode);
655 static void GenerateGeneric(MacroAssembler* masm, StrictModeFlag strict_mode);
whesse@chromium.org7b260152011-06-20 15:33:18 +0000656 static void GenerateNonStrictArguments(MacroAssembler* masm);
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000657 static void GenerateTransitionElementsSmiToDouble(MacroAssembler* masm);
658 static void GenerateTransitionElementsDoubleToObject(MacroAssembler* masm);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000659
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000660 virtual Handle<Code> GetElementStubWithoutMapCheck(
sgjesse@chromium.org6db88712011-07-11 11:41:22 +0000661 bool is_js_array,
ulan@chromium.org65a89c22012-02-14 11:46:07 +0000662 ElementsKind elements_kind,
663 KeyedAccessGrowMode grow_mode);
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000664
ricow@chromium.org64e3a4b2011-12-13 08:07:27 +0000665 virtual bool IsGeneric() const {
666 return target() == *generic_stub() ||
667 target() == *generic_stub_strict();
668 }
669
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000670 protected:
671 virtual Code::Kind kind() const { return Code::KEYED_STORE_IC; }
672
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000673 virtual Handle<Code> ComputePolymorphicStub(MapHandleList* receiver_maps,
ulan@chromium.org65a89c22012-02-14 11:46:07 +0000674 StrictModeFlag strict_mode,
675 KeyedAccessGrowMode grow_mode);
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000676
677 private:
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000678 // Update the inline cache.
679 void UpdateCaches(LookupResult* lookup,
680 State state,
ager@chromium.org9ee27ae2011-03-02 13:43:26 +0000681 StrictModeFlag strict_mode,
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000682 Handle<JSObject> receiver,
683 Handle<String> name,
684 Handle<Object> value);
685
ager@chromium.org9ee27ae2011-03-02 13:43:26 +0000686 void set_target(Code* code) {
687 // Strict mode must be preserved across IC patching.
ulan@chromium.org65a89c22012-02-14 11:46:07 +0000688 ASSERT(Code::GetStrictMode(code->extra_ic_state()) ==
689 Code::GetStrictMode(target()->extra_ic_state()));
ager@chromium.org9ee27ae2011-03-02 13:43:26 +0000690 IC::set_target(code);
691 }
692
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000693 // Stub accessors.
694 static Code* initialize_stub() {
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000695 return Isolate::Current()->builtins()->builtin(
fschneider@chromium.org7979bbb2011-03-28 10:47:03 +0000696 Builtins::kKeyedStoreIC_Initialize);
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000697 }
ager@chromium.org9ee27ae2011-03-02 13:43:26 +0000698 static Code* initialize_stub_strict() {
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000699 return Isolate::Current()->builtins()->builtin(
fschneider@chromium.org7979bbb2011-03-28 10:47:03 +0000700 Builtins::kKeyedStoreIC_Initialize_Strict);
ager@chromium.org9ee27ae2011-03-02 13:43:26 +0000701 }
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000702 Handle<Code> megamorphic_stub() {
703 return isolate()->builtins()->KeyedStoreIC_Generic();
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000704 }
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000705 Handle<Code> megamorphic_stub_strict() {
706 return isolate()->builtins()->KeyedStoreIC_Generic_Strict();
ager@chromium.org9ee27ae2011-03-02 13:43:26 +0000707 }
ricow@chromium.org64e3a4b2011-12-13 08:07:27 +0000708 Handle<Code> generic_stub() const {
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000709 return isolate()->builtins()->KeyedStoreIC_Generic();
ager@chromium.org9ee27ae2011-03-02 13:43:26 +0000710 }
ricow@chromium.org64e3a4b2011-12-13 08:07:27 +0000711 Handle<Code> generic_stub_strict() const {
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000712 return isolate()->builtins()->KeyedStoreIC_Generic_Strict();
713 }
714 Handle<Code> non_strict_arguments_stub() {
715 return isolate()->builtins()->KeyedStoreIC_NonStrictArguments();
whesse@chromium.org7b260152011-06-20 15:33:18 +0000716 }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000717
718 static void Clear(Address address, Code* target);
ager@chromium.orgeadaf222009-06-16 09:43:10 +0000719
ulan@chromium.org65a89c22012-02-14 11:46:07 +0000720 StubKind GetStubKind(Handle<JSObject> receiver,
721 Handle<Object> key,
722 Handle<Object> value);
723
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000724 friend class IC;
725};
726
727
danno@chromium.org40cb8782011-05-25 07:58:50 +0000728class UnaryOpIC: public IC {
sgjesse@chromium.org8e8294a2011-05-02 14:30:53 +0000729 public:
sgjesse@chromium.org8e8294a2011-05-02 14:30:53 +0000730 // sorted: increasingly more unspecific (ignoring UNINITIALIZED)
731 // TODO(svenpanne) Using enums+switch is an antipattern, use a class instead.
732 enum TypeInfo {
733 UNINITIALIZED,
734 SMI,
735 HEAP_NUMBER,
736 GENERIC
737 };
738
danno@chromium.org40cb8782011-05-25 07:58:50 +0000739 explicit UnaryOpIC(Isolate* isolate) : IC(NO_EXTRA_FRAME, isolate) { }
sgjesse@chromium.org8e8294a2011-05-02 14:30:53 +0000740
741 void patch(Code* code);
742
743 static const char* GetName(TypeInfo type_info);
744
745 static State ToState(TypeInfo type_info);
746
747 static TypeInfo GetTypeInfo(Handle<Object> operand);
748
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000749 static TypeInfo ComputeNewType(TypeInfo type, TypeInfo previous);
sgjesse@chromium.org8e8294a2011-05-02 14:30:53 +0000750};
751
752
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000753// Type Recording BinaryOpIC, that records the types of the inputs and outputs.
danno@chromium.org40cb8782011-05-25 07:58:50 +0000754class BinaryOpIC: public IC {
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000755 public:
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000756 enum TypeInfo {
757 UNINITIALIZED,
758 SMI,
759 INT32,
760 HEAP_NUMBER,
lrn@chromium.org7516f052011-03-30 08:52:27 +0000761 ODDBALL,
danno@chromium.org160a7b02011-04-18 15:51:38 +0000762 BOTH_STRING, // Only used for addition operation.
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000763 STRING, // Only used for addition operation. At least one string operand.
764 GENERIC
765 };
766
danno@chromium.org40cb8782011-05-25 07:58:50 +0000767 explicit BinaryOpIC(Isolate* isolate) : IC(NO_EXTRA_FRAME, isolate) { }
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000768
769 void patch(Code* code);
770
771 static const char* GetName(TypeInfo type_info);
772
773 static State ToState(TypeInfo type_info);
774
775 static TypeInfo GetTypeInfo(Handle<Object> left, Handle<Object> right);
776
777 static TypeInfo JoinTypes(TypeInfo x, TypeInfo y);
778};
779
780
781class CompareIC: public IC {
782 public:
783 enum State {
784 UNINITIALIZED,
785 SMIS,
786 HEAP_NUMBERS,
karlklose@chromium.org83a47282011-05-11 11:54:09 +0000787 SYMBOLS,
lrn@chromium.org1c092762011-05-09 09:42:16 +0000788 STRINGS,
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000789 OBJECTS,
ricow@chromium.org64e3a4b2011-12-13 08:07:27 +0000790 KNOWN_OBJECTS,
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000791 GENERIC
792 };
793
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000794 CompareIC(Isolate* isolate, Token::Value op)
795 : IC(EXTRA_CALL_FRAME, isolate), op_(op) { }
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000796
797 // Update the inline cache for the given operands.
798 void UpdateCaches(Handle<Object> x, Handle<Object> y);
799
800 // Factory method for getting an uninitialized compare stub.
801 static Handle<Code> GetUninitialized(Token::Value op);
802
803 // Helper function for computing the condition for a compare operation.
804 static Condition ComputeCondition(Token::Value op);
805
806 // Helper function for determining the state of a compare IC.
807 static State ComputeState(Code* target);
808
jkummerow@chromium.org212d9642012-05-11 15:02:09 +0000809 // Helper function for determining the operation a compare IC is for.
810 static Token::Value ComputeOperation(Code* target);
811
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000812 static const char* GetStateName(State state);
813
814 private:
ager@chromium.org5f0c45f2010-12-17 08:51:21 +0000815 State TargetState(State state, bool has_inlined_smi_code,
816 Handle<Object> x, Handle<Object> y);
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000817
818 bool strict() const { return op_ == Token::EQ_STRICT; }
819 Condition GetCondition() const { return ComputeCondition(op_); }
820 State GetState() { return ComputeState(target()); }
821
jkummerow@chromium.org212d9642012-05-11 15:02:09 +0000822 static Code* GetRawUninitialized(Token::Value op);
823
824 static void Clear(Address address, Code* target);
825
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000826 Token::Value op_;
jkummerow@chromium.org212d9642012-05-11 15:02:09 +0000827
828 friend class IC;
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000829};
830
ricow@chromium.org9fa09672011-07-25 11:05:35 +0000831
832class ToBooleanIC: public IC {
833 public:
834 explicit ToBooleanIC(Isolate* isolate) : IC(NO_EXTRA_FRAME, isolate) { }
835
836 void patch(Code* code);
837};
838
839
danno@chromium.org40cb8782011-05-25 07:58:50 +0000840// Helper for BinaryOpIC and CompareIC.
jkummerow@chromium.org212d9642012-05-11 15:02:09 +0000841enum InlinedSmiCheck { ENABLE_INLINED_SMI_CHECK, DISABLE_INLINED_SMI_CHECK };
842void PatchInlinedSmiCode(Address address, InlinedSmiCheck check);
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000843
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000844} } // namespace v8::internal
845
846#endif // V8_IC_H_