blob: 78c364229ffcf67a0d78306698138b7c95e8e9f5 [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. */ \
sgjesse@chromium.org0b6db592009-07-30 14:48:31 +000053 ICU(StoreCallbackProperty) \
54 ICU(LoadPropertyWithInterceptorOnly) \
55 ICU(LoadPropertyWithInterceptorForLoad) \
56 ICU(LoadPropertyWithInterceptorForCall) \
ager@chromium.org5c838252010-02-19 08:53:10 +000057 ICU(KeyedLoadPropertyWithInterceptor) \
ager@chromium.orgce5e87b2010-03-10 10:24:18 +000058 ICU(StoreInterceptorProperty) \
danno@chromium.org40cb8782011-05-25 07:58:50 +000059 ICU(UnaryOp_Patch) \
60 ICU(BinaryOp_Patch) \
ricow@chromium.org9fa09672011-07-25 11:05:35 +000061 ICU(CompareIC_Miss) \
62 ICU(ToBoolean_Patch)
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +000063//
kasperl@chromium.orge959c182009-07-27 08:59:04 +000064// IC is the base class for LoadIC, StoreIC, CallIC, KeyedLoadIC,
65// and KeyedStoreIC.
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +000066//
67class IC {
68 public:
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +000069 // The ids for utility called from the generated code.
70 enum UtilityId {
71 #define CONST_NAME(name) k##name,
72 IC_UTIL_LIST(CONST_NAME)
73 #undef CONST_NAME
74 kUtilityCount
75 };
76
77 // Looks up the address of the named utility.
78 static Address AddressFromUtilityId(UtilityId id);
79
80 // Alias the inline cache state type to make the IC code more readable.
81 typedef InlineCacheState State;
82
83 // The IC code is either invoked with no extra frames on the stack
84 // or with a single extra frame for supporting calls.
85 enum FrameDepth {
86 NO_EXTRA_FRAME = 0,
87 EXTRA_CALL_FRAME = 1
88 };
89
90 // Construct the IC structure with the given number of extra
91 // JavaScript frames on the stack.
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +000092 IC(FrameDepth depth, Isolate* isolate);
ricow@chromium.org64e3a4b2011-12-13 08:07:27 +000093 virtual ~IC() {}
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +000094
95 // Get the call-site target; used for determining the state.
ricow@chromium.org64e3a4b2011-12-13 08:07:27 +000096 Code* target() const { return GetTargetAtAddress(address()); }
97 inline Address address() const;
98
ager@chromium.orgce5e87b2010-03-10 10:24:18 +000099 // Compute the current IC state based on the target stub, receiver and name.
100 static State StateFrom(Code* target, Object* receiver, Object* name);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000101
102 // Clear the inline cache to initial state.
103 static void Clear(Address address);
104
105 // Computes the reloc info for this IC. This is a fairly expensive
106 // operation as it has to search through the heap to find the code
107 // object that contains this IC site.
ager@chromium.org236ad962008-09-25 09:45:57 +0000108 RelocInfo::Mode ComputeMode();
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000109
110 // Returns if this IC is for contextual (no explicit receiver)
111 // access to properties.
mvstanton@chromium.orgc47dff52013-01-23 16:28:41 +0000112 bool IsUndeclaredGlobal(Handle<Object> receiver) {
kmillikin@chromium.org13bd2942009-12-16 15:36:05 +0000113 if (receiver->IsGlobalObject()) {
mvstanton@chromium.orgc47dff52013-01-23 16:28:41 +0000114 return SlowIsUndeclaredGlobal();
kmillikin@chromium.org13bd2942009-12-16 15:36:05 +0000115 } else {
mvstanton@chromium.orgc47dff52013-01-23 16:28:41 +0000116 ASSERT(!SlowIsUndeclaredGlobal());
kmillikin@chromium.org13bd2942009-12-16 15:36:05 +0000117 return false;
118 }
119 }
120
mvstanton@chromium.orgc47dff52013-01-23 16:28:41 +0000121 bool SlowIsUndeclaredGlobal() {
ager@chromium.org236ad962008-09-25 09:45:57 +0000122 return ComputeMode() == RelocInfo::CODE_TARGET_CONTEXT;
123 }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000124
kmillikin@chromium.org69ea3962010-07-05 11:01:40 +0000125 // Determines which map must be used for keeping the code stub.
126 // These methods should not be called with undefined or null.
127 static inline InlineCacheHolderFlag GetCodeCacheForObject(Object* object,
128 JSObject* holder);
129 static inline InlineCacheHolderFlag GetCodeCacheForObject(JSObject* object,
130 JSObject* holder);
hpayer@chromium.org8432c912013-02-28 15:55:26 +0000131 static inline JSObject* GetCodeCacheHolder(Isolate* isolate,
132 Object* object,
ricow@chromium.org65fae842010-08-25 15:26:24 +0000133 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
yangguo@chromium.org4a9f6552013-03-04 14:46:33 +0000168 virtual void UpdateMonomorphicIC(Handle<JSObject> receiver,
169 Handle<Code> handler,
170 Handle<String> name) {
171 set_target(*handler);
172 }
173 bool UpdatePolymorphicIC(State state,
174 StrictModeFlag strict_mode,
175 Handle<JSObject> receiver,
176 Handle<String> name,
177 Handle<Code> code);
mvstanton@chromium.orgd16d8532013-01-25 13:29:10 +0000178 void PatchCache(State state,
179 StrictModeFlag strict_mode,
180 Handle<JSObject> receiver,
181 Handle<String> name,
182 Handle<Code> code);
183 virtual void UpdateMegamorphicCache(Map* map, String* name, Code* code);
184 virtual Handle<Code> megamorphic_stub() {
185 UNREACHABLE();
186 return Handle<Code>::null();
187 }
188 virtual Handle<Code> megamorphic_stub_strict() {
189 UNREACHABLE();
190 return Handle<Code>::null();
191 }
mstarzinger@chromium.org068ea0a2013-01-30 09:39:44 +0000192 virtual Handle<Code> generic_stub() const {
193 UNREACHABLE();
194 return Handle<Code>::null();
195 }
196 virtual Handle<Code> generic_stub_strict() const {
197 UNREACHABLE();
198 return Handle<Code>::null();
199 }
mvstanton@chromium.orgd16d8532013-01-25 13:29:10 +0000200
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000201 private:
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000202 // Frame pointer for the frame that uses (calls) the IC.
203 Address fp_;
204
205 // All access to the program counter of an IC structure is indirect
206 // to make the code GC safe. This feature is crucial since
207 // GetProperty and SetProperty are called and they in turn might
208 // invoke the garbage collector.
209 Address* pc_address_;
mads.s.ager31e71382008-08-13 09:32:07 +0000210
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000211 Isolate* isolate_;
212
mads.s.ager31e71382008-08-13 09:32:07 +0000213 DISALLOW_IMPLICIT_CONSTRUCTORS(IC);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000214};
215
216
217// An IC_Utility encapsulates IC::UtilityId. It exists mainly because you
218// cannot make forward declarations to an enum.
219class IC_Utility {
220 public:
221 explicit IC_Utility(IC::UtilityId id)
222 : address_(IC::AddressFromUtilityId(id)), id_(id) {}
223
224 Address address() const { return address_; }
225
226 IC::UtilityId id() const { return id_; }
227 private:
228 Address address_;
229 IC::UtilityId id_;
230};
231
232
lrn@chromium.org1af7e1b2010-06-07 11:12:01 +0000233class CallICBase: public IC {
danno@chromium.org40cb8782011-05-25 07:58:50 +0000234 public:
235 class Contextual: public BitField<bool, 0, 1> {};
236 class StringStubState: public BitField<StringStubFeedback, 1, 1> {};
237
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000238 // Returns a JSFunction or a Failure.
lrn@chromium.org303ada72010-10-27 09:33:13 +0000239 MUST_USE_RESULT MaybeObject* LoadFunction(State state,
erik.corry@gmail.com0511e242011-01-19 11:11:08 +0000240 Code::ExtraICState extra_ic_state,
lrn@chromium.org303ada72010-10-27 09:33:13 +0000241 Handle<Object> object,
242 Handle<String> name);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000243
lrn@chromium.org1af7e1b2010-06-07 11:12:01 +0000244 protected:
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000245 CallICBase(Code::Kind kind, Isolate* isolate)
246 : IC(EXTRA_CALL_FRAME, isolate), kind_(kind) {}
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000247
erik.corry@gmail.com0511e242011-01-19 11:11:08 +0000248 bool TryUpdateExtraICState(LookupResult* lookup,
249 Handle<Object> object,
250 Code::ExtraICState* extra_ic_state);
251
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000252 // Compute a monomorphic stub if possible, otherwise return a null handle.
253 Handle<Code> ComputeMonomorphicStub(LookupResult* lookup,
254 State state,
255 Code::ExtraICState extra_state,
256 Handle<Object> object,
257 Handle<String> name);
erik.corry@gmail.com0511e242011-01-19 11:11:08 +0000258
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000259 // Update the inline cache and the global stub cache based on the lookup
260 // result.
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000261 void UpdateCaches(LookupResult* lookup,
262 State state,
erik.corry@gmail.com0511e242011-01-19 11:11:08 +0000263 Code::ExtraICState extra_ic_state,
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000264 Handle<Object> object,
265 Handle<String> name);
266
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000267 // Returns a JSFunction if the object can be called as a function, and
268 // patches the stack to be ready for the call. Otherwise, it returns the
269 // undefined value.
270 Handle<Object> TryCallAsFunction(Handle<Object> object);
ager@chromium.org9258b6b2008-09-11 09:11:10 +0000271
fschneider@chromium.org3a5fd782011-02-24 10:10:44 +0000272 void ReceiverToObjectIfRequired(Handle<Object> callee, Handle<Object> object);
sgjesse@chromium.orgb302e562010-02-03 11:26:59 +0000273
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000274 static void Clear(Address address, Code* target);
danno@chromium.org40cb8782011-05-25 07:58:50 +0000275
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000276 // Platform-specific code generation functions used by both call and
277 // keyed call.
278 static void GenerateMiss(MacroAssembler* masm,
279 int argc,
280 IC::UtilityId id,
281 Code::ExtraICState extra_state);
282
283 static void GenerateNormal(MacroAssembler* masm, int argc);
284
285 static void GenerateMonomorphicCacheProbe(MacroAssembler* masm,
286 int argc,
287 Code::Kind kind,
288 Code::ExtraICState extra_state);
289
290 Code::Kind kind_;
291
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000292 friend class IC;
293};
294
295
lrn@chromium.org1af7e1b2010-06-07 11:12:01 +0000296class CallIC: public CallICBase {
297 public:
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000298 explicit CallIC(Isolate* isolate) : CallICBase(Code::CALL_IC, isolate) {
299 ASSERT(target()->is_call_stub());
300 }
lrn@chromium.org1af7e1b2010-06-07 11:12:01 +0000301
302 // Code generator routines.
danno@chromium.org40cb8782011-05-25 07:58:50 +0000303 static void GenerateInitialize(MacroAssembler* masm,
304 int argc,
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000305 Code::ExtraICState extra_state) {
306 GenerateMiss(masm, argc, extra_state);
lrn@chromium.org1af7e1b2010-06-07 11:12:01 +0000307 }
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000308
danno@chromium.org40cb8782011-05-25 07:58:50 +0000309 static void GenerateMiss(MacroAssembler* masm,
310 int argc,
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000311 Code::ExtraICState extra_state) {
312 CallICBase::GenerateMiss(masm, argc, IC::kCallIC_Miss, extra_state);
313 }
314
danno@chromium.org40cb8782011-05-25 07:58:50 +0000315 static void GenerateMegamorphic(MacroAssembler* masm,
316 int argc,
317 Code::ExtraICState extra_ic_state);
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000318
319 static void GenerateNormal(MacroAssembler* masm, int argc) {
320 CallICBase::GenerateNormal(masm, argc);
321 GenerateMiss(masm, argc, Code::kNoExtraICState);
322 }
lrn@chromium.org1af7e1b2010-06-07 11:12:01 +0000323};
324
325
326class KeyedCallIC: public CallICBase {
327 public:
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000328 explicit KeyedCallIC(Isolate* isolate)
329 : CallICBase(Code::KEYED_CALL_IC, isolate) {
lrn@chromium.org1af7e1b2010-06-07 11:12:01 +0000330 ASSERT(target()->is_keyed_call_stub());
331 }
332
lrn@chromium.org303ada72010-10-27 09:33:13 +0000333 MUST_USE_RESULT MaybeObject* LoadFunction(State state,
334 Handle<Object> object,
335 Handle<Object> key);
lrn@chromium.org1af7e1b2010-06-07 11:12:01 +0000336
337 // Code generator routines.
338 static void GenerateInitialize(MacroAssembler* masm, int argc) {
339 GenerateMiss(masm, argc);
340 }
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000341
342 static void GenerateMiss(MacroAssembler* masm, int argc) {
343 CallICBase::GenerateMiss(masm, argc, IC::kKeyedCallIC_Miss,
344 Code::kNoExtraICState);
345 }
346
lrn@chromium.org1af7e1b2010-06-07 11:12:01 +0000347 static void GenerateMegamorphic(MacroAssembler* masm, int argc);
348 static void GenerateNormal(MacroAssembler* masm, int argc);
whesse@chromium.org7b260152011-06-20 15:33:18 +0000349 static void GenerateNonStrictArguments(MacroAssembler* masm, int argc);
lrn@chromium.org1af7e1b2010-06-07 11:12:01 +0000350};
351
352
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000353class LoadIC: public IC {
354 public:
mstarzinger@chromium.orge3b8d0f2013-02-01 09:06:41 +0000355 explicit LoadIC(FrameDepth depth, Isolate* isolate) : IC(depth, isolate) {
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000356 ASSERT(target()->is_load_stub() || target()->is_keyed_load_stub());
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000357 }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000358
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000359 // Code generator routines.
ager@chromium.org5c838252010-02-19 08:53:10 +0000360 static void GenerateInitialize(MacroAssembler* masm) { GenerateMiss(masm); }
361 static void GeneratePreMonomorphic(MacroAssembler* masm) {
362 GenerateMiss(masm);
363 }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000364 static void GenerateMiss(MacroAssembler* masm);
365 static void GenerateMegamorphic(MacroAssembler* masm);
366 static void GenerateNormal(MacroAssembler* masm);
367
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000368 MUST_USE_RESULT MaybeObject* Load(State state,
369 Handle<Object> object,
370 Handle<String> name);
371
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000372 protected:
373 virtual Code::Kind kind() const { return Code::LOAD_IC; }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000374
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000375 virtual Handle<Code> generic_stub() const {
376 UNREACHABLE();
377 return Handle<Code>::null();
378 }
379
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000380 virtual Handle<Code> megamorphic_stub() {
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000381 return isolate()->builtins()->LoadIC_Megamorphic();
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000382 }
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000383
384 // Update the inline cache and the global stub cache based on the
385 // lookup result.
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000386 void UpdateCaches(LookupResult* lookup,
387 State state,
388 Handle<Object> object,
389 Handle<String> name);
yangguo@chromium.org4a9f6552013-03-04 14:46:33 +0000390 virtual void UpdateMonomorphicIC(Handle<JSObject> receiver,
391 Handle<Code> handler,
392 Handle<String> name);
393 virtual Handle<Code> ComputeLoadHandler(LookupResult* lookup,
394 Handle<JSObject> receiver,
395 Handle<String> name);
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000396
397 private:
398 // Stub accessors.
399 static Handle<Code> initialize_stub() {
400 return Isolate::Current()->builtins()->LoadIC_Initialize();
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000401 }
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000402 virtual Handle<Code> pre_monomorphic_stub() {
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000403 return isolate()->builtins()->LoadIC_PreMonomorphic();
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000404 }
405
406 static void Clear(Address address, Code* target);
ager@chromium.org5ec48922009-05-05 07:25:34 +0000407
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000408 friend class IC;
409};
410
411
jkummerow@chromium.org59297c72013-01-09 16:32:23 +0000412enum ICMissMode {
413 MISS_FORCE_GENERIC,
414 MISS
415};
416
417
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000418class KeyedLoadIC: public LoadIC {
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000419 public:
mstarzinger@chromium.orge3b8d0f2013-02-01 09:06:41 +0000420 explicit KeyedLoadIC(FrameDepth depth, Isolate* isolate)
421 : LoadIC(depth, isolate) {
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000422 ASSERT(target()->is_keyed_load_stub());
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000423 }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000424
lrn@chromium.org303ada72010-10-27 09:33:13 +0000425 MUST_USE_RESULT MaybeObject* Load(State state,
426 Handle<Object> object,
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000427 Handle<Object> key,
jkummerow@chromium.org59297c72013-01-09 16:32:23 +0000428 ICMissMode force_generic);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000429
430 // Code generator routines.
jkummerow@chromium.org59297c72013-01-09 16:32:23 +0000431 static void GenerateMiss(MacroAssembler* masm, ICMissMode force_generic);
ager@chromium.org5c838252010-02-19 08:53:10 +0000432 static void GenerateRuntimeGetProperty(MacroAssembler* masm);
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000433 static void GenerateInitialize(MacroAssembler* masm) {
jkummerow@chromium.org59297c72013-01-09 16:32:23 +0000434 GenerateMiss(masm, MISS);
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000435 }
ager@chromium.org5c838252010-02-19 08:53:10 +0000436 static void GeneratePreMonomorphic(MacroAssembler* masm) {
jkummerow@chromium.org59297c72013-01-09 16:32:23 +0000437 GenerateMiss(masm, MISS);
ager@chromium.org5c838252010-02-19 08:53:10 +0000438 }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000439 static void GenerateGeneric(MacroAssembler* masm);
fschneider@chromium.org0c20e672010-01-14 15:28:53 +0000440 static void GenerateString(MacroAssembler* masm);
ager@chromium.org5c838252010-02-19 08:53:10 +0000441 static void GenerateIndexedInterceptor(MacroAssembler* masm);
whesse@chromium.org7b260152011-06-20 15:33:18 +0000442 static void GenerateNonStrictArguments(MacroAssembler* masm);
ager@chromium.org3811b432009-10-28 14:53:37 +0000443
kasperl@chromium.orgeac059f2010-01-25 11:02:06 +0000444 // Bit mask to be tested against bit field for the cases when
445 // generic stub should go into slow case.
446 // Access check is necessary explicitly since generic stub does not perform
447 // map checks.
448 static const int kSlowCaseBitFieldMask =
449 (1 << Map::kIsAccessCheckNeeded) | (1 << Map::kHasIndexedInterceptor);
450
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000451 protected:
452 virtual Code::Kind kind() const { return Code::KEYED_LOAD_IC; }
453
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000454 Handle<Code> LoadElementStub(Handle<JSObject> receiver);
455
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000456 virtual Handle<Code> megamorphic_stub() {
457 return isolate()->builtins()->KeyedLoadIC_Generic();
458 }
459 virtual Handle<Code> generic_stub() const {
460 return isolate()->builtins()->KeyedLoadIC_Generic();
461 }
462
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000463 // Update the inline cache.
yangguo@chromium.org4a9f6552013-03-04 14:46:33 +0000464 virtual void UpdateMonomorphicIC(Handle<JSObject> receiver,
465 Handle<Code> handler,
466 Handle<String> name);
467 virtual Handle<Code> ComputeLoadHandler(LookupResult* lookup,
468 Handle<JSObject> receiver,
469 Handle<String> name);
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000470 virtual void UpdateMegamorphicCache(Map* map, String* name, Code* code) { }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000471
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000472 private:
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000473 // Stub accessors.
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000474 static Handle<Code> initialize_stub() {
475 return Isolate::Current()->builtins()->KeyedLoadIC_Initialize();
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000476 }
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000477 virtual Handle<Code> pre_monomorphic_stub() {
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000478 return isolate()->builtins()->KeyedLoadIC_PreMonomorphic();
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000479 }
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000480 Handle<Code> indexed_interceptor_stub() {
481 return isolate()->builtins()->KeyedLoadIC_IndexedInterceptor();
ager@chromium.org5c838252010-02-19 08:53:10 +0000482 }
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000483 Handle<Code> non_strict_arguments_stub() {
484 return isolate()->builtins()->KeyedLoadIC_NonStrictArguments();
whesse@chromium.org7b260152011-06-20 15:33:18 +0000485 }
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000486 Handle<Code> string_stub() {
487 return isolate()->builtins()->KeyedLoadIC_String();
488 }
ager@chromium.org5c838252010-02-19 08:53:10 +0000489
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000490 static void Clear(Address address, Code* target);
christian.plesner.hansen@gmail.com37abdec2009-01-06 14:43:28 +0000491
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000492 friend class IC;
493};
494
495
496class StoreIC: public IC {
497 public:
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000498 explicit StoreIC(Isolate* isolate) : IC(NO_EXTRA_FRAME, isolate) {
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000499 ASSERT(target()->is_store_stub() || target()->is_keyed_store_stub());
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000500 }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000501
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000502 // Code generators for stub routines. Only called once at startup.
sgjesse@chromium.orgb302e562010-02-03 11:26:59 +0000503 static void GenerateInitialize(MacroAssembler* masm) { GenerateMiss(masm); }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000504 static void GenerateMiss(MacroAssembler* masm);
sgjesse@chromium.org496c03a2011-02-14 12:05:43 +0000505 static void GenerateMegamorphic(MacroAssembler* masm,
ager@chromium.org9ee27ae2011-03-02 13:43:26 +0000506 StrictModeFlag strict_mode);
kmillikin@chromium.org69ea3962010-07-05 11:01:40 +0000507 static void GenerateNormal(MacroAssembler* masm);
ager@chromium.org9ee27ae2011-03-02 13:43:26 +0000508 static void GenerateGlobalProxy(MacroAssembler* masm,
509 StrictModeFlag strict_mode);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000510
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000511 MUST_USE_RESULT MaybeObject* Store(
512 State state,
513 StrictModeFlag strict_mode,
514 Handle<Object> object,
515 Handle<String> name,
516 Handle<Object> value,
517 JSReceiver::StoreFromKeyed store_mode =
518 JSReceiver::CERTAINLY_NOT_STORE_FROM_KEYED);
519
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000520 protected:
521 virtual Code::Kind kind() const { return Code::STORE_IC; }
522 virtual Handle<Code> megamorphic_stub() {
523 return isolate()->builtins()->StoreIC_Megamorphic();
524 }
525 // Stub accessors.
526 virtual Handle<Code> megamorphic_stub_strict() {
527 return isolate()->builtins()->StoreIC_Megamorphic_Strict();
528 }
529 virtual Handle<Code> global_proxy_stub() {
530 return isolate()->builtins()->StoreIC_GlobalProxy();
531 }
532 virtual Handle<Code> global_proxy_stub_strict() {
533 return isolate()->builtins()->StoreIC_GlobalProxy_Strict();
534 }
535
536
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000537 // Update the inline cache and the global stub cache based on the
538 // lookup result.
mvstanton@chromium.orgd16d8532013-01-25 13:29:10 +0000539 void UpdateCaches(LookupResult* lookup,
540 State state,
541 StrictModeFlag strict_mode,
542 Handle<JSObject> receiver,
543 Handle<String> name,
544 Handle<Object> value);
545 // Compute the code stub for this store; used for rewriting to
546 // monomorphic state and making sure that the code stub is in the
547 // stub cache.
548 virtual Handle<Code> ComputeStoreMonomorphic(LookupResult* lookup,
549 StrictModeFlag strict_mode,
550 Handle<JSObject> receiver,
551 Handle<String> name);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000552
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000553 private:
ager@chromium.org9ee27ae2011-03-02 13:43:26 +0000554 void set_target(Code* code) {
555 // Strict mode must be preserved across IC patching.
ulan@chromium.org65a89c22012-02-14 11:46:07 +0000556 ASSERT(Code::GetStrictMode(code->extra_ic_state()) ==
557 Code::GetStrictMode(target()->extra_ic_state()));
ager@chromium.org9ee27ae2011-03-02 13:43:26 +0000558 IC::set_target(code);
559 }
560
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000561 static Handle<Code> initialize_stub() {
562 return Isolate::Current()->builtins()->StoreIC_Initialize();
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000563 }
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000564 static Handle<Code> initialize_stub_strict() {
565 return Isolate::Current()->builtins()->StoreIC_Initialize_Strict();
sgjesse@chromium.org496c03a2011-02-14 12:05:43 +0000566 }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000567 static void Clear(Address address, Code* target);
fschneider@chromium.orged78ffd2010-07-21 11:05:19 +0000568
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000569 friend class IC;
570};
571
572
verwaest@chromium.org33e09c82012-10-10 17:07:22 +0000573enum KeyedStoreCheckMap {
574 kDontCheckMap,
575 kCheckMap
576};
577
578
579enum KeyedStoreIncrementLength {
580 kDontIncrementLength,
581 kIncrementLength
582};
583
584
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000585class KeyedStoreIC: public StoreIC {
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000586 public:
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000587 enum StubKind {
588 STORE_NO_TRANSITION,
589 STORE_TRANSITION_SMI_TO_OBJECT,
590 STORE_TRANSITION_SMI_TO_DOUBLE,
591 STORE_TRANSITION_DOUBLE_TO_OBJECT,
592 STORE_TRANSITION_HOLEY_SMI_TO_OBJECT,
593 STORE_TRANSITION_HOLEY_SMI_TO_DOUBLE,
594 STORE_TRANSITION_HOLEY_DOUBLE_TO_OBJECT,
595 STORE_AND_GROW_NO_TRANSITION,
596 STORE_AND_GROW_TRANSITION_SMI_TO_OBJECT,
597 STORE_AND_GROW_TRANSITION_SMI_TO_DOUBLE,
598 STORE_AND_GROW_TRANSITION_DOUBLE_TO_OBJECT,
599 STORE_AND_GROW_TRANSITION_HOLEY_SMI_TO_OBJECT,
600 STORE_AND_GROW_TRANSITION_HOLEY_SMI_TO_DOUBLE,
601 STORE_AND_GROW_TRANSITION_HOLEY_DOUBLE_TO_OBJECT
602 };
603
604 static const int kGrowICDelta = STORE_AND_GROW_NO_TRANSITION -
605 STORE_NO_TRANSITION;
606 STATIC_ASSERT(kGrowICDelta ==
607 STORE_AND_GROW_TRANSITION_SMI_TO_OBJECT -
608 STORE_TRANSITION_SMI_TO_OBJECT);
609 STATIC_ASSERT(kGrowICDelta ==
610 STORE_AND_GROW_TRANSITION_SMI_TO_DOUBLE -
611 STORE_TRANSITION_SMI_TO_DOUBLE);
612 STATIC_ASSERT(kGrowICDelta ==
613 STORE_AND_GROW_TRANSITION_DOUBLE_TO_OBJECT -
614 STORE_TRANSITION_DOUBLE_TO_OBJECT);
615
616 static inline StubKind GetGrowStubKind(StubKind stub_kind) {
617 if (stub_kind < STORE_AND_GROW_NO_TRANSITION) {
618 stub_kind = static_cast<StubKind>(static_cast<int>(stub_kind) +
619 kGrowICDelta);
620 }
621 return stub_kind;
622 }
623
624 explicit KeyedStoreIC(Isolate* isolate) : StoreIC(isolate) {
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000625 ASSERT(target()->is_keyed_store_stub());
626 }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000627
lrn@chromium.org303ada72010-10-27 09:33:13 +0000628 MUST_USE_RESULT MaybeObject* Store(State state,
verwaest@chromium.org33e09c82012-10-10 17:07:22 +0000629 StrictModeFlag strict_mode,
lrn@chromium.org303ada72010-10-27 09:33:13 +0000630 Handle<Object> object,
631 Handle<Object> name,
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000632 Handle<Object> value,
jkummerow@chromium.org59297c72013-01-09 16:32:23 +0000633 ICMissMode force_generic);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000634
635 // Code generators for stub routines. Only called once at startup.
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000636 static void GenerateInitialize(MacroAssembler* masm) {
jkummerow@chromium.org59297c72013-01-09 16:32:23 +0000637 GenerateMiss(masm, MISS);
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000638 }
jkummerow@chromium.org59297c72013-01-09 16:32:23 +0000639 static void GenerateMiss(MacroAssembler* masm, ICMissMode force_generic);
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000640 static void GenerateSlow(MacroAssembler* masm);
ager@chromium.org9ee27ae2011-03-02 13:43:26 +0000641 static void GenerateRuntimeSetProperty(MacroAssembler* masm,
642 StrictModeFlag strict_mode);
643 static void GenerateGeneric(MacroAssembler* masm, StrictModeFlag strict_mode);
whesse@chromium.org7b260152011-06-20 15:33:18 +0000644 static void GenerateNonStrictArguments(MacroAssembler* masm);
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000645 static void GenerateTransitionElementsSmiToDouble(MacroAssembler* masm);
646 static void GenerateTransitionElementsDoubleToObject(MacroAssembler* masm);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000647
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000648 protected:
649 virtual Code::Kind kind() const { return Code::KEYED_STORE_IC; }
650
mvstanton@chromium.orgd16d8532013-01-25 13:29:10 +0000651 virtual Handle<Code> ComputeStoreMonomorphic(LookupResult* lookup,
652 StrictModeFlag strict_mode,
653 Handle<JSObject> receiver,
654 Handle<String> name);
655 virtual void UpdateMegamorphicCache(Map* map, String* name, Code* code) { }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000656
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000657 virtual Handle<Code> megamorphic_stub() {
658 return isolate()->builtins()->KeyedStoreIC_Generic();
659 }
660 virtual Handle<Code> megamorphic_stub_strict() {
661 return isolate()->builtins()->KeyedStoreIC_Generic_Strict();
662 }
663
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000664 Handle<Code> StoreElementStub(Handle<JSObject> receiver,
665 StubKind stub_kind,
666 StrictModeFlag strict_mode);
667
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000668 private:
ager@chromium.org9ee27ae2011-03-02 13:43:26 +0000669 void set_target(Code* code) {
670 // Strict mode must be preserved across IC patching.
ulan@chromium.org65a89c22012-02-14 11:46:07 +0000671 ASSERT(Code::GetStrictMode(code->extra_ic_state()) ==
672 Code::GetStrictMode(target()->extra_ic_state()));
ager@chromium.org9ee27ae2011-03-02 13:43:26 +0000673 IC::set_target(code);
674 }
675
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000676 // Stub accessors.
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000677 static Handle<Code> initialize_stub() {
678 return Isolate::Current()->builtins()->KeyedStoreIC_Initialize();
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000679 }
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000680 static Handle<Code> initialize_stub_strict() {
681 return Isolate::Current()->builtins()->KeyedStoreIC_Initialize_Strict();
ager@chromium.org9ee27ae2011-03-02 13:43:26 +0000682 }
ricow@chromium.org64e3a4b2011-12-13 08:07:27 +0000683 Handle<Code> generic_stub() const {
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000684 return isolate()->builtins()->KeyedStoreIC_Generic();
ager@chromium.org9ee27ae2011-03-02 13:43:26 +0000685 }
ricow@chromium.org64e3a4b2011-12-13 08:07:27 +0000686 Handle<Code> generic_stub_strict() const {
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000687 return isolate()->builtins()->KeyedStoreIC_Generic_Strict();
688 }
689 Handle<Code> non_strict_arguments_stub() {
690 return isolate()->builtins()->KeyedStoreIC_NonStrictArguments();
whesse@chromium.org7b260152011-06-20 15:33:18 +0000691 }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000692
693 static void Clear(Address address, Code* target);
ager@chromium.orgeadaf222009-06-16 09:43:10 +0000694
ulan@chromium.org65a89c22012-02-14 11:46:07 +0000695 StubKind GetStubKind(Handle<JSObject> receiver,
696 Handle<Object> key,
697 Handle<Object> value);
698
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000699 static bool IsTransitionStubKind(StubKind stub_kind) {
700 return stub_kind > STORE_NO_TRANSITION &&
701 stub_kind != STORE_AND_GROW_NO_TRANSITION;
702 }
703
704 static bool IsGrowStubKind(StubKind stub_kind) {
705 return stub_kind >= STORE_AND_GROW_NO_TRANSITION;
706 }
707
708 static StubKind GetNoTransitionStubKind(StubKind stub_kind) {
709 if (!IsTransitionStubKind(stub_kind)) return stub_kind;
710 if (IsGrowStubKind(stub_kind)) return STORE_AND_GROW_NO_TRANSITION;
711 return STORE_NO_TRANSITION;
712 }
713
714 Handle<Map> ComputeTransitionedMap(Handle<JSObject> receiver,
715 StubKind stub_kind);
716
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000717 friend class IC;
718};
719
720
danno@chromium.org40cb8782011-05-25 07:58:50 +0000721class UnaryOpIC: public IC {
sgjesse@chromium.org8e8294a2011-05-02 14:30:53 +0000722 public:
sgjesse@chromium.org8e8294a2011-05-02 14:30:53 +0000723 // sorted: increasingly more unspecific (ignoring UNINITIALIZED)
724 // TODO(svenpanne) Using enums+switch is an antipattern, use a class instead.
725 enum TypeInfo {
726 UNINITIALIZED,
727 SMI,
hpayer@chromium.org8432c912013-02-28 15:55:26 +0000728 NUMBER,
sgjesse@chromium.org8e8294a2011-05-02 14:30:53 +0000729 GENERIC
730 };
731
danno@chromium.org40cb8782011-05-25 07:58:50 +0000732 explicit UnaryOpIC(Isolate* isolate) : IC(NO_EXTRA_FRAME, isolate) { }
sgjesse@chromium.org8e8294a2011-05-02 14:30:53 +0000733
734 void patch(Code* code);
735
736 static const char* GetName(TypeInfo type_info);
737
738 static State ToState(TypeInfo type_info);
739
740 static TypeInfo GetTypeInfo(Handle<Object> operand);
741
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000742 static TypeInfo ComputeNewType(TypeInfo type, TypeInfo previous);
sgjesse@chromium.org8e8294a2011-05-02 14:30:53 +0000743};
744
745
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000746// Type Recording BinaryOpIC, that records the types of the inputs and outputs.
danno@chromium.org40cb8782011-05-25 07:58:50 +0000747class BinaryOpIC: public IC {
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000748 public:
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000749 enum TypeInfo {
750 UNINITIALIZED,
751 SMI,
752 INT32,
hpayer@chromium.org8432c912013-02-28 15:55:26 +0000753 NUMBER,
lrn@chromium.org7516f052011-03-30 08:52:27 +0000754 ODDBALL,
yangguo@chromium.orgfb377212012-11-16 14:43:43 +0000755 STRING, // Only used for addition operation.
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000756 GENERIC
757 };
758
danno@chromium.org40cb8782011-05-25 07:58:50 +0000759 explicit BinaryOpIC(Isolate* isolate) : IC(NO_EXTRA_FRAME, isolate) { }
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000760
761 void patch(Code* code);
762
763 static const char* GetName(TypeInfo type_info);
764
765 static State ToState(TypeInfo type_info);
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000766};
767
768
769class CompareIC: public IC {
770 public:
hpayer@chromium.org8432c912013-02-28 15:55:26 +0000771 // The type/state lattice is defined by the following inequations:
772 // UNINITIALIZED < ...
773 // ... < GENERIC
774 // SMI < NUMBER
yangguo@chromium.org4a9f6552013-03-04 14:46:33 +0000775 // INTERNALIZED_STRING < STRING
hpayer@chromium.org8432c912013-02-28 15:55:26 +0000776 // KNOWN_OBJECT < OBJECT
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000777 enum State {
778 UNINITIALIZED,
yangguo@chromium.orgfb377212012-11-16 14:43:43 +0000779 SMI,
hpayer@chromium.org8432c912013-02-28 15:55:26 +0000780 NUMBER,
yangguo@chromium.orgfb377212012-11-16 14:43:43 +0000781 STRING,
yangguo@chromium.org4a9f6552013-03-04 14:46:33 +0000782 INTERNALIZED_STRING,
783 UNIQUE_NAME, // Symbol or InternalizedString
hpayer@chromium.org8432c912013-02-28 15:55:26 +0000784 OBJECT, // JSObject
785 KNOWN_OBJECT, // JSObject with specific map (faster check)
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000786 GENERIC
787 };
788
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000789 CompareIC(Isolate* isolate, Token::Value op)
790 : IC(EXTRA_CALL_FRAME, isolate), op_(op) { }
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000791
792 // Update the inline cache for the given operands.
793 void UpdateCaches(Handle<Object> x, Handle<Object> y);
794
yangguo@chromium.orgfb377212012-11-16 14:43:43 +0000795
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000796 // Factory method for getting an uninitialized compare stub.
hpayer@chromium.org8432c912013-02-28 15:55:26 +0000797 static Handle<Code> GetUninitialized(Isolate* isolate, Token::Value op);
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000798
799 // Helper function for computing the condition for a compare operation.
800 static Condition ComputeCondition(Token::Value op);
801
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000802 static const char* GetStateName(State state);
803
804 private:
yangguo@chromium.orgfb377212012-11-16 14:43:43 +0000805 static bool HasInlinedSmiCode(Address address);
806
807 State TargetState(State old_state,
808 State old_left,
809 State old_right,
810 bool has_inlined_smi_code,
811 Handle<Object> x,
812 Handle<Object> y);
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000813
814 bool strict() const { return op_ == Token::EQ_STRICT; }
815 Condition GetCondition() const { return ComputeCondition(op_); }
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000816
jkummerow@chromium.org212d9642012-05-11 15:02:09 +0000817 static Code* GetRawUninitialized(Token::Value op);
818
819 static void Clear(Address address, Code* target);
820
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000821 Token::Value op_;
jkummerow@chromium.org212d9642012-05-11 15:02:09 +0000822
823 friend class IC;
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000824};
825
ricow@chromium.org9fa09672011-07-25 11:05:35 +0000826
827class ToBooleanIC: public IC {
828 public:
829 explicit ToBooleanIC(Isolate* isolate) : IC(NO_EXTRA_FRAME, isolate) { }
830
831 void patch(Code* code);
832};
833
834
danno@chromium.org40cb8782011-05-25 07:58:50 +0000835// Helper for BinaryOpIC and CompareIC.
jkummerow@chromium.org212d9642012-05-11 15:02:09 +0000836enum InlinedSmiCheck { ENABLE_INLINED_SMI_CHECK, DISABLE_INLINED_SMI_CHECK };
837void PatchInlinedSmiCode(Address address, InlinedSmiCheck check);
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000838
jkummerow@chromium.org59297c72013-01-09 16:32:23 +0000839DECLARE_RUNTIME_FUNCTION(MaybeObject*, KeyedLoadIC_Miss);
mstarzinger@chromium.orge3b8d0f2013-02-01 09:06:41 +0000840DECLARE_RUNTIME_FUNCTION(MaybeObject*, KeyedLoadIC_MissFromStubFailure);
jkummerow@chromium.org59297c72013-01-09 16:32:23 +0000841
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000842} } // namespace v8::internal
843
844#endif // V8_IC_H_