blob: cb316407c0563c340e143bc621c69d75642916dc [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
ager@chromium.orgce5e87b2010-03-10 10:24:18 +0000100 // Compute the current IC state based on the target stub, receiver and name.
101 static State StateFrom(Code* target, Object* receiver, Object* name);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000102
103 // Clear the inline cache to initial state.
104 static void Clear(Address address);
105
106 // Computes the reloc info for this IC. This is a fairly expensive
107 // operation as it has to search through the heap to find the code
108 // object that contains this IC site.
ager@chromium.org236ad962008-09-25 09:45:57 +0000109 RelocInfo::Mode ComputeMode();
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000110
111 // Returns if this IC is for contextual (no explicit receiver)
112 // access to properties.
mvstanton@chromium.orgc47dff52013-01-23 16:28:41 +0000113 bool IsUndeclaredGlobal(Handle<Object> receiver) {
kmillikin@chromium.org13bd2942009-12-16 15:36:05 +0000114 if (receiver->IsGlobalObject()) {
mvstanton@chromium.orgc47dff52013-01-23 16:28:41 +0000115 return SlowIsUndeclaredGlobal();
kmillikin@chromium.org13bd2942009-12-16 15:36:05 +0000116 } else {
mvstanton@chromium.orgc47dff52013-01-23 16:28:41 +0000117 ASSERT(!SlowIsUndeclaredGlobal());
kmillikin@chromium.org13bd2942009-12-16 15:36:05 +0000118 return false;
119 }
120 }
121
mvstanton@chromium.orgc47dff52013-01-23 16:28:41 +0000122 bool SlowIsUndeclaredGlobal() {
ager@chromium.org236ad962008-09-25 09:45:57 +0000123 return ComputeMode() == RelocInfo::CODE_TARGET_CONTEXT;
124 }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000125
kmillikin@chromium.org69ea3962010-07-05 11:01:40 +0000126 // Determines which map must be used for keeping the code stub.
127 // These methods should not be called with undefined or null.
128 static inline InlineCacheHolderFlag GetCodeCacheForObject(Object* object,
129 JSObject* holder);
130 static inline InlineCacheHolderFlag GetCodeCacheForObject(JSObject* object,
131 JSObject* holder);
ricow@chromium.org65fae842010-08-25 15:26:24 +0000132 static inline JSObject* GetCodeCacheHolder(Object* object,
133 InlineCacheHolderFlag holder);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000134
135 protected:
136 Address fp() const { return fp_; }
137 Address pc() const { return *pc_address_; }
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000138 Isolate* isolate() const { return isolate_; }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000139
ager@chromium.org65dad4b2009-04-23 08:48:43 +0000140#ifdef ENABLE_DEBUGGER_SUPPORT
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000141 // Computes the address in the original code when the code running is
142 // containing break points (calls to DebugBreakXXX builtins).
ricow@chromium.org64e3a4b2011-12-13 08:07:27 +0000143 Address OriginalCodeAddress() const;
ager@chromium.org65dad4b2009-04-23 08:48:43 +0000144#endif
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000145
146 // Set the call-site target.
147 void set_target(Code* code) { SetTargetAtAddress(address(), code); }
148
149#ifdef DEBUG
ricow@chromium.org64e3a4b2011-12-13 08:07:27 +0000150 char TransitionMarkFromState(IC::State state);
151
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000152 void TraceIC(const char* type,
153 Handle<Object> name,
154 State old_state,
lrn@chromium.org34e60782011-09-15 07:25:40 +0000155 Code* new_target);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000156#endif
157
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000158 Failure* TypeError(const char* type,
159 Handle<Object> object,
160 Handle<Object> key);
161 Failure* ReferenceError(const char* type, Handle<String> name);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000162
163 // Access the target code for the given IC address.
164 static inline Code* GetTargetAtAddress(Address address);
165 static inline void SetTargetAtAddress(Address address, Code* target);
jkummerow@chromium.orgf7a58842012-02-21 10:08:21 +0000166 static void PostPatching(Address address, Code* target, Code* old_target);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000167
mvstanton@chromium.orgd16d8532013-01-25 13:29:10 +0000168 void PatchCache(State state,
169 StrictModeFlag strict_mode,
170 Handle<JSObject> receiver,
171 Handle<String> name,
172 Handle<Code> code);
173 virtual void UpdateMegamorphicCache(Map* map, String* name, Code* code);
174 virtual Handle<Code> megamorphic_stub() {
175 UNREACHABLE();
176 return Handle<Code>::null();
177 }
178 virtual Handle<Code> megamorphic_stub_strict() {
179 UNREACHABLE();
180 return Handle<Code>::null();
181 }
mstarzinger@chromium.org068ea0a2013-01-30 09:39:44 +0000182 virtual Handle<Code> generic_stub() const {
183 UNREACHABLE();
184 return Handle<Code>::null();
185 }
186 virtual Handle<Code> generic_stub_strict() const {
187 UNREACHABLE();
188 return Handle<Code>::null();
189 }
mvstanton@chromium.orgd16d8532013-01-25 13:29:10 +0000190
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000191 private:
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000192 // Frame pointer for the frame that uses (calls) the IC.
193 Address fp_;
194
195 // All access to the program counter of an IC structure is indirect
196 // to make the code GC safe. This feature is crucial since
197 // GetProperty and SetProperty are called and they in turn might
198 // invoke the garbage collector.
199 Address* pc_address_;
mads.s.ager31e71382008-08-13 09:32:07 +0000200
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000201 Isolate* isolate_;
202
mads.s.ager31e71382008-08-13 09:32:07 +0000203 DISALLOW_IMPLICIT_CONSTRUCTORS(IC);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000204};
205
206
207// An IC_Utility encapsulates IC::UtilityId. It exists mainly because you
208// cannot make forward declarations to an enum.
209class IC_Utility {
210 public:
211 explicit IC_Utility(IC::UtilityId id)
212 : address_(IC::AddressFromUtilityId(id)), id_(id) {}
213
214 Address address() const { return address_; }
215
216 IC::UtilityId id() const { return id_; }
217 private:
218 Address address_;
219 IC::UtilityId id_;
220};
221
222
lrn@chromium.org1af7e1b2010-06-07 11:12:01 +0000223class CallICBase: public IC {
danno@chromium.org40cb8782011-05-25 07:58:50 +0000224 public:
225 class Contextual: public BitField<bool, 0, 1> {};
226 class StringStubState: public BitField<StringStubFeedback, 1, 1> {};
227
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000228 // Returns a JSFunction or a Failure.
lrn@chromium.org303ada72010-10-27 09:33:13 +0000229 MUST_USE_RESULT MaybeObject* LoadFunction(State state,
erik.corry@gmail.com0511e242011-01-19 11:11:08 +0000230 Code::ExtraICState extra_ic_state,
lrn@chromium.org303ada72010-10-27 09:33:13 +0000231 Handle<Object> object,
232 Handle<String> name);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000233
lrn@chromium.org1af7e1b2010-06-07 11:12:01 +0000234 protected:
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000235 CallICBase(Code::Kind kind, Isolate* isolate)
236 : IC(EXTRA_CALL_FRAME, isolate), kind_(kind) {}
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000237
erik.corry@gmail.com0511e242011-01-19 11:11:08 +0000238 bool TryUpdateExtraICState(LookupResult* lookup,
239 Handle<Object> object,
240 Code::ExtraICState* extra_ic_state);
241
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000242 // Compute a monomorphic stub if possible, otherwise return a null handle.
243 Handle<Code> ComputeMonomorphicStub(LookupResult* lookup,
244 State state,
245 Code::ExtraICState extra_state,
246 Handle<Object> object,
247 Handle<String> name);
erik.corry@gmail.com0511e242011-01-19 11:11:08 +0000248
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000249 // Update the inline cache and the global stub cache based on the lookup
250 // result.
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000251 void UpdateCaches(LookupResult* lookup,
252 State state,
erik.corry@gmail.com0511e242011-01-19 11:11:08 +0000253 Code::ExtraICState extra_ic_state,
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000254 Handle<Object> object,
255 Handle<String> name);
256
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000257 // Returns a JSFunction if the object can be called as a function, and
258 // patches the stack to be ready for the call. Otherwise, it returns the
259 // undefined value.
260 Handle<Object> TryCallAsFunction(Handle<Object> object);
ager@chromium.org9258b6b2008-09-11 09:11:10 +0000261
fschneider@chromium.org3a5fd782011-02-24 10:10:44 +0000262 void ReceiverToObjectIfRequired(Handle<Object> callee, Handle<Object> object);
sgjesse@chromium.orgb302e562010-02-03 11:26:59 +0000263
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000264 static void Clear(Address address, Code* target);
danno@chromium.org40cb8782011-05-25 07:58:50 +0000265
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000266 // Platform-specific code generation functions used by both call and
267 // keyed call.
268 static void GenerateMiss(MacroAssembler* masm,
269 int argc,
270 IC::UtilityId id,
271 Code::ExtraICState extra_state);
272
273 static void GenerateNormal(MacroAssembler* masm, int argc);
274
275 static void GenerateMonomorphicCacheProbe(MacroAssembler* masm,
276 int argc,
277 Code::Kind kind,
278 Code::ExtraICState extra_state);
279
280 Code::Kind kind_;
281
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000282 friend class IC;
283};
284
285
lrn@chromium.org1af7e1b2010-06-07 11:12:01 +0000286class CallIC: public CallICBase {
287 public:
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000288 explicit CallIC(Isolate* isolate) : CallICBase(Code::CALL_IC, isolate) {
289 ASSERT(target()->is_call_stub());
290 }
lrn@chromium.org1af7e1b2010-06-07 11:12:01 +0000291
292 // Code generator routines.
danno@chromium.org40cb8782011-05-25 07:58:50 +0000293 static void GenerateInitialize(MacroAssembler* masm,
294 int argc,
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000295 Code::ExtraICState extra_state) {
296 GenerateMiss(masm, argc, extra_state);
lrn@chromium.org1af7e1b2010-06-07 11:12:01 +0000297 }
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000298
danno@chromium.org40cb8782011-05-25 07:58:50 +0000299 static void GenerateMiss(MacroAssembler* masm,
300 int argc,
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000301 Code::ExtraICState extra_state) {
302 CallICBase::GenerateMiss(masm, argc, IC::kCallIC_Miss, extra_state);
303 }
304
danno@chromium.org40cb8782011-05-25 07:58:50 +0000305 static void GenerateMegamorphic(MacroAssembler* masm,
306 int argc,
307 Code::ExtraICState extra_ic_state);
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000308
309 static void GenerateNormal(MacroAssembler* masm, int argc) {
310 CallICBase::GenerateNormal(masm, argc);
311 GenerateMiss(masm, argc, Code::kNoExtraICState);
312 }
lrn@chromium.org1af7e1b2010-06-07 11:12:01 +0000313};
314
315
316class KeyedCallIC: public CallICBase {
317 public:
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000318 explicit KeyedCallIC(Isolate* isolate)
319 : CallICBase(Code::KEYED_CALL_IC, isolate) {
lrn@chromium.org1af7e1b2010-06-07 11:12:01 +0000320 ASSERT(target()->is_keyed_call_stub());
321 }
322
lrn@chromium.org303ada72010-10-27 09:33:13 +0000323 MUST_USE_RESULT MaybeObject* LoadFunction(State state,
324 Handle<Object> object,
325 Handle<Object> key);
lrn@chromium.org1af7e1b2010-06-07 11:12:01 +0000326
327 // Code generator routines.
328 static void GenerateInitialize(MacroAssembler* masm, int argc) {
329 GenerateMiss(masm, argc);
330 }
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000331
332 static void GenerateMiss(MacroAssembler* masm, int argc) {
333 CallICBase::GenerateMiss(masm, argc, IC::kKeyedCallIC_Miss,
334 Code::kNoExtraICState);
335 }
336
lrn@chromium.org1af7e1b2010-06-07 11:12:01 +0000337 static void GenerateMegamorphic(MacroAssembler* masm, int argc);
338 static void GenerateNormal(MacroAssembler* masm, int argc);
whesse@chromium.org7b260152011-06-20 15:33:18 +0000339 static void GenerateNonStrictArguments(MacroAssembler* masm, int argc);
lrn@chromium.org1af7e1b2010-06-07 11:12:01 +0000340};
341
342
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000343class LoadIC: public IC {
344 public:
mstarzinger@chromium.orge3b8d0f2013-02-01 09:06:41 +0000345 explicit LoadIC(FrameDepth depth, Isolate* isolate) : IC(depth, isolate) {
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000346 ASSERT(target()->is_load_stub() || target()->is_keyed_load_stub());
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000347 }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000348
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000349 // Code generator routines.
ager@chromium.org5c838252010-02-19 08:53:10 +0000350 static void GenerateInitialize(MacroAssembler* masm) { GenerateMiss(masm); }
351 static void GeneratePreMonomorphic(MacroAssembler* masm) {
352 GenerateMiss(masm);
353 }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000354 static void GenerateMiss(MacroAssembler* masm);
355 static void GenerateMegamorphic(MacroAssembler* masm);
356 static void GenerateNormal(MacroAssembler* masm);
357
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000358 MUST_USE_RESULT MaybeObject* Load(State state,
359 Handle<Object> object,
360 Handle<String> name);
361
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000362 protected:
363 virtual Code::Kind kind() const { return Code::LOAD_IC; }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000364
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000365 virtual Handle<Code> generic_stub() const {
366 UNREACHABLE();
367 return Handle<Code>::null();
368 }
369
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000370 virtual Handle<Code> megamorphic_stub() {
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000371 return isolate()->builtins()->LoadIC_Megamorphic();
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000372 }
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000373
374 // Update the inline cache and the global stub cache based on the
375 // lookup result.
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000376 void UpdateCaches(LookupResult* lookup,
377 State state,
378 Handle<Object> object,
379 Handle<String> name);
380 virtual Handle<Code> ComputeLoadMonomorphic(LookupResult* lookup,
381 Handle<JSObject> receiver,
382 Handle<String> name);
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000383
384 private:
385 // Stub accessors.
386 static Handle<Code> initialize_stub() {
387 return Isolate::Current()->builtins()->LoadIC_Initialize();
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000388 }
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000389 virtual Handle<Code> pre_monomorphic_stub() {
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000390 return isolate()->builtins()->LoadIC_PreMonomorphic();
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000391 }
392
393 static void Clear(Address address, Code* target);
ager@chromium.org5ec48922009-05-05 07:25:34 +0000394
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000395 friend class IC;
396};
397
398
jkummerow@chromium.org59297c72013-01-09 16:32:23 +0000399enum ICMissMode {
400 MISS_FORCE_GENERIC,
401 MISS
402};
403
404
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000405class KeyedLoadIC: public LoadIC {
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000406 public:
mstarzinger@chromium.orge3b8d0f2013-02-01 09:06:41 +0000407 explicit KeyedLoadIC(FrameDepth depth, Isolate* isolate)
408 : LoadIC(depth, isolate) {
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000409 ASSERT(target()->is_keyed_load_stub());
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000410 }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000411
lrn@chromium.org303ada72010-10-27 09:33:13 +0000412 MUST_USE_RESULT MaybeObject* Load(State state,
413 Handle<Object> object,
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000414 Handle<Object> key,
jkummerow@chromium.org59297c72013-01-09 16:32:23 +0000415 ICMissMode force_generic);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000416
417 // Code generator routines.
jkummerow@chromium.org59297c72013-01-09 16:32:23 +0000418 static void GenerateMiss(MacroAssembler* masm, ICMissMode force_generic);
ager@chromium.org5c838252010-02-19 08:53:10 +0000419 static void GenerateRuntimeGetProperty(MacroAssembler* masm);
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000420 static void GenerateInitialize(MacroAssembler* masm) {
jkummerow@chromium.org59297c72013-01-09 16:32:23 +0000421 GenerateMiss(masm, MISS);
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000422 }
ager@chromium.org5c838252010-02-19 08:53:10 +0000423 static void GeneratePreMonomorphic(MacroAssembler* masm) {
jkummerow@chromium.org59297c72013-01-09 16:32:23 +0000424 GenerateMiss(masm, MISS);
ager@chromium.org5c838252010-02-19 08:53:10 +0000425 }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000426 static void GenerateGeneric(MacroAssembler* masm);
fschneider@chromium.org0c20e672010-01-14 15:28:53 +0000427 static void GenerateString(MacroAssembler* masm);
ager@chromium.org5c838252010-02-19 08:53:10 +0000428 static void GenerateIndexedInterceptor(MacroAssembler* masm);
whesse@chromium.org7b260152011-06-20 15:33:18 +0000429 static void GenerateNonStrictArguments(MacroAssembler* masm);
ager@chromium.org3811b432009-10-28 14:53:37 +0000430
kasperl@chromium.orgeac059f2010-01-25 11:02:06 +0000431 // Bit mask to be tested against bit field for the cases when
432 // generic stub should go into slow case.
433 // Access check is necessary explicitly since generic stub does not perform
434 // map checks.
435 static const int kSlowCaseBitFieldMask =
436 (1 << Map::kIsAccessCheckNeeded) | (1 << Map::kHasIndexedInterceptor);
437
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000438 protected:
439 virtual Code::Kind kind() const { return Code::KEYED_LOAD_IC; }
440
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000441 Handle<Code> LoadElementStub(Handle<JSObject> receiver);
442
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000443 virtual Handle<Code> megamorphic_stub() {
444 return isolate()->builtins()->KeyedLoadIC_Generic();
445 }
446 virtual Handle<Code> generic_stub() const {
447 return isolate()->builtins()->KeyedLoadIC_Generic();
448 }
449
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000450 // Update the inline cache.
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000451 virtual Handle<Code> ComputeLoadMonomorphic(LookupResult* lookup,
452 Handle<JSObject> receiver,
453 Handle<String> name);
454 virtual void UpdateMegamorphicCache(Map* map, String* name, Code* code) { }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000455
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000456 private:
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000457 // Stub accessors.
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000458 static Handle<Code> initialize_stub() {
459 return Isolate::Current()->builtins()->KeyedLoadIC_Initialize();
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000460 }
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000461 virtual Handle<Code> pre_monomorphic_stub() {
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000462 return isolate()->builtins()->KeyedLoadIC_PreMonomorphic();
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000463 }
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000464 Handle<Code> indexed_interceptor_stub() {
465 return isolate()->builtins()->KeyedLoadIC_IndexedInterceptor();
ager@chromium.org5c838252010-02-19 08:53:10 +0000466 }
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000467 Handle<Code> non_strict_arguments_stub() {
468 return isolate()->builtins()->KeyedLoadIC_NonStrictArguments();
whesse@chromium.org7b260152011-06-20 15:33:18 +0000469 }
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000470 Handle<Code> string_stub() {
471 return isolate()->builtins()->KeyedLoadIC_String();
472 }
ager@chromium.org5c838252010-02-19 08:53:10 +0000473
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000474 static void Clear(Address address, Code* target);
christian.plesner.hansen@gmail.com37abdec2009-01-06 14:43:28 +0000475
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000476 friend class IC;
477};
478
479
480class StoreIC: public IC {
481 public:
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000482 explicit StoreIC(Isolate* isolate) : IC(NO_EXTRA_FRAME, isolate) {
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000483 ASSERT(target()->is_store_stub() || target()->is_keyed_store_stub());
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000484 }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000485
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000486 // Code generators for stub routines. Only called once at startup.
sgjesse@chromium.orgb302e562010-02-03 11:26:59 +0000487 static void GenerateInitialize(MacroAssembler* masm) { GenerateMiss(masm); }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000488 static void GenerateMiss(MacroAssembler* masm);
sgjesse@chromium.org496c03a2011-02-14 12:05:43 +0000489 static void GenerateMegamorphic(MacroAssembler* masm,
ager@chromium.org9ee27ae2011-03-02 13:43:26 +0000490 StrictModeFlag strict_mode);
kmillikin@chromium.org69ea3962010-07-05 11:01:40 +0000491 static void GenerateNormal(MacroAssembler* masm);
ager@chromium.org9ee27ae2011-03-02 13:43:26 +0000492 static void GenerateGlobalProxy(MacroAssembler* masm,
493 StrictModeFlag strict_mode);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000494
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000495 MUST_USE_RESULT MaybeObject* Store(
496 State state,
497 StrictModeFlag strict_mode,
498 Handle<Object> object,
499 Handle<String> name,
500 Handle<Object> value,
501 JSReceiver::StoreFromKeyed store_mode =
502 JSReceiver::CERTAINLY_NOT_STORE_FROM_KEYED);
503
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000504 protected:
505 virtual Code::Kind kind() const { return Code::STORE_IC; }
506 virtual Handle<Code> megamorphic_stub() {
507 return isolate()->builtins()->StoreIC_Megamorphic();
508 }
509 // Stub accessors.
510 virtual Handle<Code> megamorphic_stub_strict() {
511 return isolate()->builtins()->StoreIC_Megamorphic_Strict();
512 }
513 virtual Handle<Code> global_proxy_stub() {
514 return isolate()->builtins()->StoreIC_GlobalProxy();
515 }
516 virtual Handle<Code> global_proxy_stub_strict() {
517 return isolate()->builtins()->StoreIC_GlobalProxy_Strict();
518 }
519
520
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000521 // Update the inline cache and the global stub cache based on the
522 // lookup result.
mvstanton@chromium.orgd16d8532013-01-25 13:29:10 +0000523 void UpdateCaches(LookupResult* lookup,
524 State state,
525 StrictModeFlag strict_mode,
526 Handle<JSObject> receiver,
527 Handle<String> name,
528 Handle<Object> value);
529 // Compute the code stub for this store; used for rewriting to
530 // monomorphic state and making sure that the code stub is in the
531 // stub cache.
532 virtual Handle<Code> ComputeStoreMonomorphic(LookupResult* lookup,
533 StrictModeFlag strict_mode,
534 Handle<JSObject> receiver,
535 Handle<String> name);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000536
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000537 private:
ager@chromium.org9ee27ae2011-03-02 13:43:26 +0000538 void set_target(Code* code) {
539 // Strict mode must be preserved across IC patching.
ulan@chromium.org65a89c22012-02-14 11:46:07 +0000540 ASSERT(Code::GetStrictMode(code->extra_ic_state()) ==
541 Code::GetStrictMode(target()->extra_ic_state()));
ager@chromium.org9ee27ae2011-03-02 13:43:26 +0000542 IC::set_target(code);
543 }
544
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000545 static Handle<Code> initialize_stub() {
546 return Isolate::Current()->builtins()->StoreIC_Initialize();
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000547 }
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000548 static Handle<Code> initialize_stub_strict() {
549 return Isolate::Current()->builtins()->StoreIC_Initialize_Strict();
sgjesse@chromium.org496c03a2011-02-14 12:05:43 +0000550 }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000551 static void Clear(Address address, Code* target);
fschneider@chromium.orged78ffd2010-07-21 11:05:19 +0000552
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000553 friend class IC;
554};
555
556
verwaest@chromium.org33e09c82012-10-10 17:07:22 +0000557enum KeyedStoreCheckMap {
558 kDontCheckMap,
559 kCheckMap
560};
561
562
563enum KeyedStoreIncrementLength {
564 kDontIncrementLength,
565 kIncrementLength
566};
567
568
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000569class KeyedStoreIC: public StoreIC {
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000570 public:
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000571 enum StubKind {
572 STORE_NO_TRANSITION,
573 STORE_TRANSITION_SMI_TO_OBJECT,
574 STORE_TRANSITION_SMI_TO_DOUBLE,
575 STORE_TRANSITION_DOUBLE_TO_OBJECT,
576 STORE_TRANSITION_HOLEY_SMI_TO_OBJECT,
577 STORE_TRANSITION_HOLEY_SMI_TO_DOUBLE,
578 STORE_TRANSITION_HOLEY_DOUBLE_TO_OBJECT,
579 STORE_AND_GROW_NO_TRANSITION,
580 STORE_AND_GROW_TRANSITION_SMI_TO_OBJECT,
581 STORE_AND_GROW_TRANSITION_SMI_TO_DOUBLE,
582 STORE_AND_GROW_TRANSITION_DOUBLE_TO_OBJECT,
583 STORE_AND_GROW_TRANSITION_HOLEY_SMI_TO_OBJECT,
584 STORE_AND_GROW_TRANSITION_HOLEY_SMI_TO_DOUBLE,
585 STORE_AND_GROW_TRANSITION_HOLEY_DOUBLE_TO_OBJECT
586 };
587
588 static const int kGrowICDelta = STORE_AND_GROW_NO_TRANSITION -
589 STORE_NO_TRANSITION;
590 STATIC_ASSERT(kGrowICDelta ==
591 STORE_AND_GROW_TRANSITION_SMI_TO_OBJECT -
592 STORE_TRANSITION_SMI_TO_OBJECT);
593 STATIC_ASSERT(kGrowICDelta ==
594 STORE_AND_GROW_TRANSITION_SMI_TO_DOUBLE -
595 STORE_TRANSITION_SMI_TO_DOUBLE);
596 STATIC_ASSERT(kGrowICDelta ==
597 STORE_AND_GROW_TRANSITION_DOUBLE_TO_OBJECT -
598 STORE_TRANSITION_DOUBLE_TO_OBJECT);
599
600 static inline StubKind GetGrowStubKind(StubKind stub_kind) {
601 if (stub_kind < STORE_AND_GROW_NO_TRANSITION) {
602 stub_kind = static_cast<StubKind>(static_cast<int>(stub_kind) +
603 kGrowICDelta);
604 }
605 return stub_kind;
606 }
607
608 explicit KeyedStoreIC(Isolate* isolate) : StoreIC(isolate) {
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000609 ASSERT(target()->is_keyed_store_stub());
610 }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000611
lrn@chromium.org303ada72010-10-27 09:33:13 +0000612 MUST_USE_RESULT MaybeObject* Store(State state,
verwaest@chromium.org33e09c82012-10-10 17:07:22 +0000613 StrictModeFlag strict_mode,
lrn@chromium.org303ada72010-10-27 09:33:13 +0000614 Handle<Object> object,
615 Handle<Object> name,
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000616 Handle<Object> value,
jkummerow@chromium.org59297c72013-01-09 16:32:23 +0000617 ICMissMode force_generic);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000618
619 // Code generators for stub routines. Only called once at startup.
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000620 static void GenerateInitialize(MacroAssembler* masm) {
jkummerow@chromium.org59297c72013-01-09 16:32:23 +0000621 GenerateMiss(masm, MISS);
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000622 }
jkummerow@chromium.org59297c72013-01-09 16:32:23 +0000623 static void GenerateMiss(MacroAssembler* masm, ICMissMode force_generic);
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000624 static void GenerateSlow(MacroAssembler* masm);
ager@chromium.org9ee27ae2011-03-02 13:43:26 +0000625 static void GenerateRuntimeSetProperty(MacroAssembler* masm,
626 StrictModeFlag strict_mode);
627 static void GenerateGeneric(MacroAssembler* masm, StrictModeFlag strict_mode);
whesse@chromium.org7b260152011-06-20 15:33:18 +0000628 static void GenerateNonStrictArguments(MacroAssembler* masm);
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000629 static void GenerateTransitionElementsSmiToDouble(MacroAssembler* masm);
630 static void GenerateTransitionElementsDoubleToObject(MacroAssembler* masm);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000631
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000632 protected:
633 virtual Code::Kind kind() const { return Code::KEYED_STORE_IC; }
634
mvstanton@chromium.orgd16d8532013-01-25 13:29:10 +0000635 virtual Handle<Code> ComputeStoreMonomorphic(LookupResult* lookup,
636 StrictModeFlag strict_mode,
637 Handle<JSObject> receiver,
638 Handle<String> name);
639 virtual void UpdateMegamorphicCache(Map* map, String* name, Code* code) { }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000640
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000641 virtual Handle<Code> megamorphic_stub() {
642 return isolate()->builtins()->KeyedStoreIC_Generic();
643 }
644 virtual Handle<Code> megamorphic_stub_strict() {
645 return isolate()->builtins()->KeyedStoreIC_Generic_Strict();
646 }
647
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000648 Handle<Code> StoreElementStub(Handle<JSObject> receiver,
649 StubKind stub_kind,
650 StrictModeFlag strict_mode);
651
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000652 private:
ager@chromium.org9ee27ae2011-03-02 13:43:26 +0000653 void set_target(Code* code) {
654 // Strict mode must be preserved across IC patching.
ulan@chromium.org65a89c22012-02-14 11:46:07 +0000655 ASSERT(Code::GetStrictMode(code->extra_ic_state()) ==
656 Code::GetStrictMode(target()->extra_ic_state()));
ager@chromium.org9ee27ae2011-03-02 13:43:26 +0000657 IC::set_target(code);
658 }
659
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000660 // Stub accessors.
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000661 static Handle<Code> initialize_stub() {
662 return Isolate::Current()->builtins()->KeyedStoreIC_Initialize();
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000663 }
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000664 static Handle<Code> initialize_stub_strict() {
665 return Isolate::Current()->builtins()->KeyedStoreIC_Initialize_Strict();
ager@chromium.org9ee27ae2011-03-02 13:43:26 +0000666 }
ricow@chromium.org64e3a4b2011-12-13 08:07:27 +0000667 Handle<Code> generic_stub() const {
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000668 return isolate()->builtins()->KeyedStoreIC_Generic();
ager@chromium.org9ee27ae2011-03-02 13:43:26 +0000669 }
ricow@chromium.org64e3a4b2011-12-13 08:07:27 +0000670 Handle<Code> generic_stub_strict() const {
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000671 return isolate()->builtins()->KeyedStoreIC_Generic_Strict();
672 }
673 Handle<Code> non_strict_arguments_stub() {
674 return isolate()->builtins()->KeyedStoreIC_NonStrictArguments();
whesse@chromium.org7b260152011-06-20 15:33:18 +0000675 }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000676
677 static void Clear(Address address, Code* target);
ager@chromium.orgeadaf222009-06-16 09:43:10 +0000678
ulan@chromium.org65a89c22012-02-14 11:46:07 +0000679 StubKind GetStubKind(Handle<JSObject> receiver,
680 Handle<Object> key,
681 Handle<Object> value);
682
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000683 static bool IsTransitionStubKind(StubKind stub_kind) {
684 return stub_kind > STORE_NO_TRANSITION &&
685 stub_kind != STORE_AND_GROW_NO_TRANSITION;
686 }
687
688 static bool IsGrowStubKind(StubKind stub_kind) {
689 return stub_kind >= STORE_AND_GROW_NO_TRANSITION;
690 }
691
692 static StubKind GetNoTransitionStubKind(StubKind stub_kind) {
693 if (!IsTransitionStubKind(stub_kind)) return stub_kind;
694 if (IsGrowStubKind(stub_kind)) return STORE_AND_GROW_NO_TRANSITION;
695 return STORE_NO_TRANSITION;
696 }
697
698 Handle<Map> ComputeTransitionedMap(Handle<JSObject> receiver,
699 StubKind stub_kind);
700
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000701 friend class IC;
702};
703
704
danno@chromium.org40cb8782011-05-25 07:58:50 +0000705class UnaryOpIC: public IC {
sgjesse@chromium.org8e8294a2011-05-02 14:30:53 +0000706 public:
sgjesse@chromium.org8e8294a2011-05-02 14:30:53 +0000707 // sorted: increasingly more unspecific (ignoring UNINITIALIZED)
708 // TODO(svenpanne) Using enums+switch is an antipattern, use a class instead.
709 enum TypeInfo {
710 UNINITIALIZED,
711 SMI,
712 HEAP_NUMBER,
713 GENERIC
714 };
715
danno@chromium.org40cb8782011-05-25 07:58:50 +0000716 explicit UnaryOpIC(Isolate* isolate) : IC(NO_EXTRA_FRAME, isolate) { }
sgjesse@chromium.org8e8294a2011-05-02 14:30:53 +0000717
718 void patch(Code* code);
719
720 static const char* GetName(TypeInfo type_info);
721
722 static State ToState(TypeInfo type_info);
723
724 static TypeInfo GetTypeInfo(Handle<Object> operand);
725
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000726 static TypeInfo ComputeNewType(TypeInfo type, TypeInfo previous);
sgjesse@chromium.org8e8294a2011-05-02 14:30:53 +0000727};
728
729
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000730// Type Recording BinaryOpIC, that records the types of the inputs and outputs.
danno@chromium.org40cb8782011-05-25 07:58:50 +0000731class BinaryOpIC: public IC {
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000732 public:
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000733 enum TypeInfo {
734 UNINITIALIZED,
735 SMI,
736 INT32,
737 HEAP_NUMBER,
lrn@chromium.org7516f052011-03-30 08:52:27 +0000738 ODDBALL,
yangguo@chromium.orgfb377212012-11-16 14:43:43 +0000739 STRING, // Only used for addition operation.
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000740 GENERIC
741 };
742
danno@chromium.org40cb8782011-05-25 07:58:50 +0000743 explicit BinaryOpIC(Isolate* isolate) : IC(NO_EXTRA_FRAME, isolate) { }
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000744
745 void patch(Code* code);
746
747 static const char* GetName(TypeInfo type_info);
748
749 static State ToState(TypeInfo type_info);
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000750};
751
752
753class CompareIC: public IC {
754 public:
755 enum State {
756 UNINITIALIZED,
yangguo@chromium.orgfb377212012-11-16 14:43:43 +0000757 SMI,
758 HEAP_NUMBER,
759 SYMBOL,
760 STRING,
761 OBJECT,
ricow@chromium.org64e3a4b2011-12-13 08:07:27 +0000762 KNOWN_OBJECTS,
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000763 GENERIC
764 };
765
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000766 CompareIC(Isolate* isolate, Token::Value op)
767 : IC(EXTRA_CALL_FRAME, isolate), op_(op) { }
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000768
769 // Update the inline cache for the given operands.
770 void UpdateCaches(Handle<Object> x, Handle<Object> y);
771
yangguo@chromium.orgfb377212012-11-16 14:43:43 +0000772
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000773 // Factory method for getting an uninitialized compare stub.
774 static Handle<Code> GetUninitialized(Token::Value op);
775
776 // Helper function for computing the condition for a compare operation.
777 static Condition ComputeCondition(Token::Value op);
778
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000779 static const char* GetStateName(State state);
780
781 private:
yangguo@chromium.orgfb377212012-11-16 14:43:43 +0000782 static bool HasInlinedSmiCode(Address address);
783
784 State TargetState(State old_state,
785 State old_left,
786 State old_right,
787 bool has_inlined_smi_code,
788 Handle<Object> x,
789 Handle<Object> y);
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000790
791 bool strict() const { return op_ == Token::EQ_STRICT; }
792 Condition GetCondition() const { return ComputeCondition(op_); }
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000793
jkummerow@chromium.org212d9642012-05-11 15:02:09 +0000794 static Code* GetRawUninitialized(Token::Value op);
795
796 static void Clear(Address address, Code* target);
797
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000798 Token::Value op_;
jkummerow@chromium.org212d9642012-05-11 15:02:09 +0000799
800 friend class IC;
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000801};
802
ricow@chromium.org9fa09672011-07-25 11:05:35 +0000803
804class ToBooleanIC: public IC {
805 public:
806 explicit ToBooleanIC(Isolate* isolate) : IC(NO_EXTRA_FRAME, isolate) { }
807
808 void patch(Code* code);
809};
810
811
danno@chromium.org40cb8782011-05-25 07:58:50 +0000812// Helper for BinaryOpIC and CompareIC.
jkummerow@chromium.org212d9642012-05-11 15:02:09 +0000813enum InlinedSmiCheck { ENABLE_INLINED_SMI_CHECK, DISABLE_INLINED_SMI_CHECK };
814void PatchInlinedSmiCode(Address address, InlinedSmiCheck check);
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000815
jkummerow@chromium.org59297c72013-01-09 16:32:23 +0000816DECLARE_RUNTIME_FUNCTION(MaybeObject*, KeyedLoadIC_Miss);
mstarzinger@chromium.orge3b8d0f2013-02-01 09:06:41 +0000817DECLARE_RUNTIME_FUNCTION(MaybeObject*, KeyedLoadIC_MissFromStubFailure);
jkummerow@chromium.org59297c72013-01-09 16:32:23 +0000818
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000819} } // namespace v8::internal
820
821#endif // V8_IC_H_