blob: 3f0c5c611fdd92a43651040ff56fbf03cbb4a3e4 [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) \
ulan@chromium.org57ff8812013-05-10 08:16:55 +000048 ICU(StoreIC_Slow) \
sgjesse@chromium.org0b6db592009-07-30 14:48:31 +000049 ICU(SharedStoreIC_ExtendStorage) \
50 ICU(KeyedStoreIC_Miss) \
ager@chromium.orgea91cc52011-05-23 06:06:11 +000051 ICU(KeyedStoreIC_MissForceGeneric) \
52 ICU(KeyedStoreIC_Slow) \
sgjesse@chromium.org0b6db592009-07-30 14:48:31 +000053 /* Utilities for IC stubs. */ \
sgjesse@chromium.org0b6db592009-07-30 14:48:31 +000054 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) \
ricow@chromium.org9fa09672011-07-25 11:05:35 +000060 ICU(CompareIC_Miss) \
jkummerow@chromium.org25b0e212013-10-04 15:38:52 +000061 ICU(BinaryOpIC_Miss) \
danno@chromium.orgca29dd82013-04-26 11:59:48 +000062 ICU(CompareNilIC_Miss) \
ulan@chromium.org77ca49a2013-04-22 09:43:56 +000063 ICU(Unreachable) \
rossberg@chromium.orgb99c7542013-05-31 11:40:45 +000064 ICU(ToBooleanIC_Miss)
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +000065//
kasperl@chromium.orge959c182009-07-27 08:59:04 +000066// IC is the base class for LoadIC, StoreIC, CallIC, KeyedLoadIC,
67// and KeyedStoreIC.
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +000068//
69class IC {
70 public:
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +000071 // The ids for utility called from the generated code.
72 enum UtilityId {
73 #define CONST_NAME(name) k##name,
74 IC_UTIL_LIST(CONST_NAME)
75 #undef CONST_NAME
76 kUtilityCount
77 };
78
79 // Looks up the address of the named utility.
80 static Address AddressFromUtilityId(UtilityId id);
81
82 // Alias the inline cache state type to make the IC code more readable.
83 typedef InlineCacheState State;
84
85 // The IC code is either invoked with no extra frames on the stack
86 // or with a single extra frame for supporting calls.
87 enum FrameDepth {
88 NO_EXTRA_FRAME = 0,
89 EXTRA_CALL_FRAME = 1
90 };
91
92 // Construct the IC structure with the given number of extra
93 // JavaScript frames on the stack.
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +000094 IC(FrameDepth depth, Isolate* isolate);
ricow@chromium.org64e3a4b2011-12-13 08:07:27 +000095 virtual ~IC() {}
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +000096
97 // Get the call-site target; used for determining the state.
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +000098 Handle<Code> target() const { return target_; }
99 Code* raw_target() const { return GetTargetAtAddress(address()); }
100
101 State state() const { return state_; }
ricow@chromium.org64e3a4b2011-12-13 08:07:27 +0000102 inline Address address() const;
103
ager@chromium.orgce5e87b2010-03-10 10:24:18 +0000104 // Compute the current IC state based on the target stub, receiver and name.
jkummerow@chromium.orgd8a3a142013-10-03 12:15:05 +0000105 void UpdateState(Handle<Object> receiver, Handle<Object> name);
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000106 void MarkMonomorphicPrototypeFailure() {
107 state_ = MONOMORPHIC_PROTOTYPE_FAILURE;
108 }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000109
110 // Clear the inline cache to initial state.
jkummerow@chromium.org3d00d0a2013-09-04 13:57:32 +0000111 static void Clear(Isolate* isolate, Address address);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000112
113 // Computes the reloc info for this IC. This is a fairly expensive
114 // operation as it has to search through the heap to find the code
115 // object that contains this IC site.
ager@chromium.org236ad962008-09-25 09:45:57 +0000116 RelocInfo::Mode ComputeMode();
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000117
118 // Returns if this IC is for contextual (no explicit receiver)
119 // access to properties.
mvstanton@chromium.orgc47dff52013-01-23 16:28:41 +0000120 bool IsUndeclaredGlobal(Handle<Object> receiver) {
kmillikin@chromium.org13bd2942009-12-16 15:36:05 +0000121 if (receiver->IsGlobalObject()) {
mvstanton@chromium.orgc47dff52013-01-23 16:28:41 +0000122 return SlowIsUndeclaredGlobal();
kmillikin@chromium.org13bd2942009-12-16 15:36:05 +0000123 } else {
mvstanton@chromium.orgc47dff52013-01-23 16:28:41 +0000124 ASSERT(!SlowIsUndeclaredGlobal());
kmillikin@chromium.org13bd2942009-12-16 15:36:05 +0000125 return false;
126 }
127 }
128
mvstanton@chromium.orgc47dff52013-01-23 16:28:41 +0000129 bool SlowIsUndeclaredGlobal() {
ager@chromium.org236ad962008-09-25 09:45:57 +0000130 return ComputeMode() == RelocInfo::CODE_TARGET_CONTEXT;
131 }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000132
kmillikin@chromium.org69ea3962010-07-05 11:01:40 +0000133 // Determines which map must be used for keeping the code stub.
134 // These methods should not be called with undefined or null.
135 static inline InlineCacheHolderFlag GetCodeCacheForObject(Object* object,
136 JSObject* holder);
hpayer@chromium.org8432c912013-02-28 15:55:26 +0000137 static inline JSObject* GetCodeCacheHolder(Isolate* isolate,
138 Object* object,
ricow@chromium.org65fae842010-08-25 15:26:24 +0000139 InlineCacheHolderFlag holder);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000140
machenbach@chromium.org8e36b5b2013-09-26 07:36:30 +0000141 static bool IsCleared(Code* code) {
142 InlineCacheState state = code->ic_state();
143 return state == UNINITIALIZED || state == PREMONOMORPHIC;
144 }
145
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000146 protected:
147 Address fp() const { return fp_; }
148 Address pc() const { return *pc_address_; }
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000149 Isolate* isolate() const { return isolate_; }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000150
ager@chromium.org65dad4b2009-04-23 08:48:43 +0000151#ifdef ENABLE_DEBUGGER_SUPPORT
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000152 // Computes the address in the original code when the code running is
153 // containing break points (calls to DebugBreakXXX builtins).
ricow@chromium.org64e3a4b2011-12-13 08:07:27 +0000154 Address OriginalCodeAddress() const;
ager@chromium.org65dad4b2009-04-23 08:48:43 +0000155#endif
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000156
157 // Set the call-site target.
158 void set_target(Code* code) { SetTargetAtAddress(address(), code); }
159
160#ifdef DEBUG
ricow@chromium.org64e3a4b2011-12-13 08:07:27 +0000161 char TransitionMarkFromState(IC::State state);
162
jkummerow@chromium.orgd8a3a142013-10-03 12:15:05 +0000163 void TraceIC(const char* type, Handle<Object> name);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000164#endif
165
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000166 Failure* TypeError(const char* type,
167 Handle<Object> object,
168 Handle<Object> key);
169 Failure* ReferenceError(const char* type, Handle<String> name);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000170
171 // Access the target code for the given IC address.
172 static inline Code* GetTargetAtAddress(Address address);
173 static inline void SetTargetAtAddress(Address address, Code* target);
jkummerow@chromium.orgf7a58842012-02-21 10:08:21 +0000174 static void PostPatching(Address address, Code* target, Code* old_target);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000175
mstarzinger@chromium.org2efc3e42013-10-14 08:45:38 +0000176 // Compute the handler either by compiling or by retrieving a cached version.
177 Handle<Code> ComputeHandler(LookupResult* lookup,
178 Handle<JSObject> receiver,
179 Handle<String> name,
180 Handle<Object> value = Handle<Code>::null());
181 virtual Handle<Code> CompileHandler(LookupResult* lookup,
182 Handle<JSObject> receiver,
183 Handle<String> name,
184 Handle<Object> value) {
185 UNREACHABLE();
186 return Handle<Code>::null();
187 }
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000188 void UpdateMonomorphicIC(Handle<HeapObject> receiver,
189 Handle<Code> handler,
190 Handle<String> name);
danno@chromium.orgbee51992013-07-10 14:57:15 +0000191
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000192 bool UpdatePolymorphicIC(Handle<HeapObject> receiver,
193 Handle<String> name,
194 Handle<Code> code);
danno@chromium.orgbee51992013-07-10 14:57:15 +0000195
ulan@chromium.org6e196bf2013-03-13 09:38:22 +0000196 void CopyICToMegamorphicCache(Handle<String> name);
mstarzinger@chromium.orgf705b502013-04-04 11:38:09 +0000197 bool IsTransitionedMapOfMonomorphicTarget(Map* receiver_map);
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000198 void PatchCache(Handle<HeapObject> receiver,
mvstanton@chromium.orgd16d8532013-01-25 13:29:10 +0000199 Handle<String> name,
200 Handle<Code> code);
ulan@chromium.org57ff8812013-05-10 08:16:55 +0000201 virtual void UpdateMegamorphicCache(Map* map, Name* name, Code* code);
mstarzinger@chromium.org2efc3e42013-10-14 08:45:38 +0000202 virtual Code::Kind kind() const {
203 UNREACHABLE();
204 return Code::STUB;
205 }
206 virtual Handle<Code> slow_stub() const {
207 UNREACHABLE();
208 return Handle<Code>::null();
209 }
mvstanton@chromium.orgd16d8532013-01-25 13:29:10 +0000210 virtual Handle<Code> megamorphic_stub() {
211 UNREACHABLE();
212 return Handle<Code>::null();
213 }
mstarzinger@chromium.org068ea0a2013-01-30 09:39:44 +0000214 virtual Handle<Code> generic_stub() const {
215 UNREACHABLE();
216 return Handle<Code>::null();
217 }
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000218 virtual StrictModeFlag strict_mode() const { return kNonStrictMode; }
jkummerow@chromium.orgd8a3a142013-10-03 12:15:05 +0000219 bool TryRemoveInvalidPrototypeDependentStub(Handle<Object> receiver,
220 Handle<String> name);
mstarzinger@chromium.org2efc3e42013-10-14 08:45:38 +0000221 void TryRemoveInvalidHandlers(Handle<Map> map, Handle<String> name);
mvstanton@chromium.orgd16d8532013-01-25 13:29:10 +0000222
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000223 private:
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000224 // Frame pointer for the frame that uses (calls) the IC.
225 Address fp_;
226
227 // All access to the program counter of an IC structure is indirect
228 // to make the code GC safe. This feature is crucial since
229 // GetProperty and SetProperty are called and they in turn might
230 // invoke the garbage collector.
231 Address* pc_address_;
mads.s.ager31e71382008-08-13 09:32:07 +0000232
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000233 Isolate* isolate_;
234
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000235 // The original code target that missed.
236 Handle<Code> target_;
237 State state_;
238
mads.s.ager31e71382008-08-13 09:32:07 +0000239 DISALLOW_IMPLICIT_CONSTRUCTORS(IC);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000240};
241
242
243// An IC_Utility encapsulates IC::UtilityId. It exists mainly because you
244// cannot make forward declarations to an enum.
245class IC_Utility {
246 public:
247 explicit IC_Utility(IC::UtilityId id)
248 : address_(IC::AddressFromUtilityId(id)), id_(id) {}
249
250 Address address() const { return address_; }
251
252 IC::UtilityId id() const { return id_; }
253 private:
254 Address address_;
255 IC::UtilityId id_;
256};
257
258
lrn@chromium.org1af7e1b2010-06-07 11:12:01 +0000259class CallICBase: public IC {
danno@chromium.org40cb8782011-05-25 07:58:50 +0000260 public:
261 class Contextual: public BitField<bool, 0, 1> {};
262 class StringStubState: public BitField<StringStubFeedback, 1, 1> {};
263
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000264 // Returns a JSFunction or a Failure.
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000265 MUST_USE_RESULT MaybeObject* LoadFunction(Handle<Object> object,
lrn@chromium.org303ada72010-10-27 09:33:13 +0000266 Handle<String> name);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000267
lrn@chromium.org1af7e1b2010-06-07 11:12:01 +0000268 protected:
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000269 CallICBase(Code::Kind kind, Isolate* isolate)
270 : IC(EXTRA_CALL_FRAME, isolate), kind_(kind) {}
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000271
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000272 virtual Code::ExtraICState extra_ic_state() { return Code::kNoExtraICState; }
erik.corry@gmail.com0511e242011-01-19 11:11:08 +0000273
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000274 // Compute a monomorphic stub if possible, otherwise return a null handle.
275 Handle<Code> ComputeMonomorphicStub(LookupResult* lookup,
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000276 Handle<Object> object,
277 Handle<String> name);
erik.corry@gmail.com0511e242011-01-19 11:11:08 +0000278
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000279 // Update the inline cache and the global stub cache based on the lookup
280 // result.
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000281 void UpdateCaches(LookupResult* lookup,
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000282 Handle<Object> object,
283 Handle<String> name);
284
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000285 // Returns a JSFunction if the object can be called as a function, and
286 // patches the stack to be ready for the call. Otherwise, it returns the
287 // undefined value.
288 Handle<Object> TryCallAsFunction(Handle<Object> object);
ager@chromium.org9258b6b2008-09-11 09:11:10 +0000289
fschneider@chromium.org3a5fd782011-02-24 10:10:44 +0000290 void ReceiverToObjectIfRequired(Handle<Object> callee, Handle<Object> object);
sgjesse@chromium.orgb302e562010-02-03 11:26:59 +0000291
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000292 static void Clear(Address address, Code* target);
danno@chromium.org40cb8782011-05-25 07:58:50 +0000293
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000294 // Platform-specific code generation functions used by both call and
295 // keyed call.
296 static void GenerateMiss(MacroAssembler* masm,
297 int argc,
298 IC::UtilityId id,
299 Code::ExtraICState extra_state);
300
301 static void GenerateNormal(MacroAssembler* masm, int argc);
302
303 static void GenerateMonomorphicCacheProbe(MacroAssembler* masm,
304 int argc,
305 Code::Kind kind,
306 Code::ExtraICState extra_state);
307
jkummerow@chromium.orgd8a3a142013-10-03 12:15:05 +0000308 virtual Handle<Code> megamorphic_stub();
309 virtual Handle<Code> pre_monomorphic_stub();
310
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000311 Code::Kind kind_;
312
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000313 friend class IC;
314};
315
316
lrn@chromium.org1af7e1b2010-06-07 11:12:01 +0000317class CallIC: public CallICBase {
318 public:
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000319 explicit CallIC(Isolate* isolate)
320 : CallICBase(Code::CALL_IC, isolate),
321 extra_ic_state_(target()->extra_ic_state()) {
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000322 ASSERT(target()->is_call_stub());
323 }
lrn@chromium.org1af7e1b2010-06-07 11:12:01 +0000324
325 // Code generator routines.
danno@chromium.org40cb8782011-05-25 07:58:50 +0000326 static void GenerateInitialize(MacroAssembler* masm,
327 int argc,
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000328 Code::ExtraICState extra_state) {
329 GenerateMiss(masm, argc, extra_state);
lrn@chromium.org1af7e1b2010-06-07 11:12:01 +0000330 }
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000331
danno@chromium.org40cb8782011-05-25 07:58:50 +0000332 static void GenerateMiss(MacroAssembler* masm,
333 int argc,
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000334 Code::ExtraICState extra_state) {
335 CallICBase::GenerateMiss(masm, argc, IC::kCallIC_Miss, extra_state);
336 }
337
danno@chromium.org40cb8782011-05-25 07:58:50 +0000338 static void GenerateMegamorphic(MacroAssembler* masm,
339 int argc,
340 Code::ExtraICState extra_ic_state);
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000341
342 static void GenerateNormal(MacroAssembler* masm, int argc) {
343 CallICBase::GenerateNormal(masm, argc);
344 GenerateMiss(masm, argc, Code::kNoExtraICState);
345 }
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000346 bool TryUpdateExtraICState(LookupResult* lookup, Handle<Object> object);
347
348 protected:
349 virtual Code::ExtraICState extra_ic_state() { return extra_ic_state_; }
350
351 private:
352 Code::ExtraICState extra_ic_state_;
lrn@chromium.org1af7e1b2010-06-07 11:12:01 +0000353};
354
355
356class KeyedCallIC: public CallICBase {
357 public:
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000358 explicit KeyedCallIC(Isolate* isolate)
359 : CallICBase(Code::KEYED_CALL_IC, isolate) {
lrn@chromium.org1af7e1b2010-06-07 11:12:01 +0000360 ASSERT(target()->is_keyed_call_stub());
361 }
362
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000363 MUST_USE_RESULT MaybeObject* LoadFunction(Handle<Object> object,
lrn@chromium.org303ada72010-10-27 09:33:13 +0000364 Handle<Object> key);
lrn@chromium.org1af7e1b2010-06-07 11:12:01 +0000365
366 // Code generator routines.
367 static void GenerateInitialize(MacroAssembler* masm, int argc) {
368 GenerateMiss(masm, argc);
369 }
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000370
371 static void GenerateMiss(MacroAssembler* masm, int argc) {
372 CallICBase::GenerateMiss(masm, argc, IC::kKeyedCallIC_Miss,
373 Code::kNoExtraICState);
374 }
375
lrn@chromium.org1af7e1b2010-06-07 11:12:01 +0000376 static void GenerateMegamorphic(MacroAssembler* masm, int argc);
377 static void GenerateNormal(MacroAssembler* masm, int argc);
whesse@chromium.org7b260152011-06-20 15:33:18 +0000378 static void GenerateNonStrictArguments(MacroAssembler* masm, int argc);
lrn@chromium.org1af7e1b2010-06-07 11:12:01 +0000379};
380
381
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000382class LoadIC: public IC {
383 public:
mstarzinger@chromium.orge3b8d0f2013-02-01 09:06:41 +0000384 explicit LoadIC(FrameDepth depth, Isolate* isolate) : IC(depth, isolate) {
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000385 ASSERT(target()->is_load_stub() || target()->is_keyed_load_stub());
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000386 }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000387
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000388 // Code generator routines.
ager@chromium.org5c838252010-02-19 08:53:10 +0000389 static void GenerateInitialize(MacroAssembler* masm) { GenerateMiss(masm); }
390 static void GeneratePreMonomorphic(MacroAssembler* masm) {
391 GenerateMiss(masm);
392 }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000393 static void GenerateMiss(MacroAssembler* masm);
394 static void GenerateMegamorphic(MacroAssembler* masm);
395 static void GenerateNormal(MacroAssembler* masm);
verwaest@chromium.orgd4be0f02013-06-05 13:39:03 +0000396 static void GenerateRuntimeGetProperty(MacroAssembler* masm);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000397
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000398 MUST_USE_RESULT MaybeObject* Load(Handle<Object> object,
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000399 Handle<String> name);
400
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000401 protected:
402 virtual Code::Kind kind() const { return Code::LOAD_IC; }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000403
dslomov@chromium.org4a35c5a2013-09-13 07:28:52 +0000404 virtual Handle<Code> slow_stub() const {
verwaest@chromium.orgd4be0f02013-06-05 13:39:03 +0000405 return isolate()->builtins()->LoadIC_Slow();
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000406 }
407
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000408 virtual Handle<Code> megamorphic_stub() {
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000409 return isolate()->builtins()->LoadIC_Megamorphic();
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000410 }
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000411
412 // Update the inline cache and the global stub cache based on the
413 // lookup result.
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000414 void UpdateCaches(LookupResult* lookup,
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000415 Handle<Object> object,
416 Handle<String> name);
danno@chromium.orgbee51992013-07-10 14:57:15 +0000417
mstarzinger@chromium.org2efc3e42013-10-14 08:45:38 +0000418 virtual Handle<Code> CompileHandler(LookupResult* lookup,
419 Handle<JSObject> receiver,
420 Handle<String> name,
421 Handle<Object> unused);
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000422
423 private:
424 // Stub accessors.
jkummerow@chromium.org3d00d0a2013-09-04 13:57:32 +0000425 static Handle<Code> initialize_stub(Isolate* isolate) {
426 return isolate->builtins()->LoadIC_Initialize();
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000427 }
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000428
machenbach@chromium.org8e36b5b2013-09-26 07:36:30 +0000429 static Handle<Code> pre_monomorphic_stub(Isolate* isolate) {
430 return isolate->builtins()->LoadIC_PreMonomorphic();
431 }
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000432
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000433 virtual Handle<Code> pre_monomorphic_stub() {
machenbach@chromium.org8e36b5b2013-09-26 07:36:30 +0000434 return pre_monomorphic_stub(isolate());
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000435 }
436
mstarzinger@chromium.org2efc3e42013-10-14 08:45:38 +0000437 Handle<Code> SimpleFieldLoad(int offset,
438 bool inobject = true,
439 Representation representation =
440 Representation::Tagged());
441
jkummerow@chromium.org3d00d0a2013-09-04 13:57:32 +0000442 static void Clear(Isolate* isolate, Address address, Code* target);
ager@chromium.org5ec48922009-05-05 07:25:34 +0000443
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000444 friend class IC;
445};
446
447
jkummerow@chromium.org59297c72013-01-09 16:32:23 +0000448enum ICMissMode {
449 MISS_FORCE_GENERIC,
450 MISS
451};
452
453
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000454class KeyedLoadIC: public LoadIC {
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000455 public:
mstarzinger@chromium.orge3b8d0f2013-02-01 09:06:41 +0000456 explicit KeyedLoadIC(FrameDepth depth, Isolate* isolate)
457 : LoadIC(depth, isolate) {
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000458 ASSERT(target()->is_keyed_load_stub());
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000459 }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000460
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000461 MUST_USE_RESULT MaybeObject* Load(Handle<Object> object,
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000462 Handle<Object> key,
jkummerow@chromium.org59297c72013-01-09 16:32:23 +0000463 ICMissMode force_generic);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000464
465 // Code generator routines.
jkummerow@chromium.org59297c72013-01-09 16:32:23 +0000466 static void GenerateMiss(MacroAssembler* masm, ICMissMode force_generic);
ager@chromium.org5c838252010-02-19 08:53:10 +0000467 static void GenerateRuntimeGetProperty(MacroAssembler* masm);
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000468 static void GenerateInitialize(MacroAssembler* masm) {
jkummerow@chromium.org59297c72013-01-09 16:32:23 +0000469 GenerateMiss(masm, MISS);
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000470 }
ager@chromium.org5c838252010-02-19 08:53:10 +0000471 static void GeneratePreMonomorphic(MacroAssembler* masm) {
jkummerow@chromium.org59297c72013-01-09 16:32:23 +0000472 GenerateMiss(masm, MISS);
ager@chromium.org5c838252010-02-19 08:53:10 +0000473 }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000474 static void GenerateGeneric(MacroAssembler* masm);
fschneider@chromium.org0c20e672010-01-14 15:28:53 +0000475 static void GenerateString(MacroAssembler* masm);
ager@chromium.org5c838252010-02-19 08:53:10 +0000476 static void GenerateIndexedInterceptor(MacroAssembler* masm);
whesse@chromium.org7b260152011-06-20 15:33:18 +0000477 static void GenerateNonStrictArguments(MacroAssembler* masm);
ager@chromium.org3811b432009-10-28 14:53:37 +0000478
kasperl@chromium.orgeac059f2010-01-25 11:02:06 +0000479 // Bit mask to be tested against bit field for the cases when
480 // generic stub should go into slow case.
481 // Access check is necessary explicitly since generic stub does not perform
482 // map checks.
483 static const int kSlowCaseBitFieldMask =
484 (1 << Map::kIsAccessCheckNeeded) | (1 << Map::kHasIndexedInterceptor);
485
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000486 protected:
487 virtual Code::Kind kind() const { return Code::KEYED_LOAD_IC; }
488
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000489 Handle<Code> LoadElementStub(Handle<JSObject> receiver);
490
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000491 virtual Handle<Code> megamorphic_stub() {
492 return isolate()->builtins()->KeyedLoadIC_Generic();
493 }
494 virtual Handle<Code> generic_stub() const {
495 return isolate()->builtins()->KeyedLoadIC_Generic();
496 }
dslomov@chromium.org4a35c5a2013-09-13 07:28:52 +0000497 virtual Handle<Code> slow_stub() const {
498 return isolate()->builtins()->KeyedLoadIC_Slow();
499 }
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000500
ulan@chromium.org57ff8812013-05-10 08:16:55 +0000501 virtual void UpdateMegamorphicCache(Map* map, Name* name, Code* code) { }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000502
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000503 private:
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000504 // Stub accessors.
jkummerow@chromium.org3d00d0a2013-09-04 13:57:32 +0000505 static Handle<Code> initialize_stub(Isolate* isolate) {
506 return isolate->builtins()->KeyedLoadIC_Initialize();
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000507 }
machenbach@chromium.org8e36b5b2013-09-26 07:36:30 +0000508 static Handle<Code> pre_monomorphic_stub(Isolate* isolate) {
509 return isolate->builtins()->KeyedLoadIC_PreMonomorphic();
510 }
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000511 virtual Handle<Code> pre_monomorphic_stub() {
machenbach@chromium.org8e36b5b2013-09-26 07:36:30 +0000512 return pre_monomorphic_stub(isolate());
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000513 }
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000514 Handle<Code> indexed_interceptor_stub() {
515 return isolate()->builtins()->KeyedLoadIC_IndexedInterceptor();
ager@chromium.org5c838252010-02-19 08:53:10 +0000516 }
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000517 Handle<Code> non_strict_arguments_stub() {
518 return isolate()->builtins()->KeyedLoadIC_NonStrictArguments();
whesse@chromium.org7b260152011-06-20 15:33:18 +0000519 }
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000520 Handle<Code> string_stub() {
521 return isolate()->builtins()->KeyedLoadIC_String();
522 }
ager@chromium.org5c838252010-02-19 08:53:10 +0000523
jkummerow@chromium.org3d00d0a2013-09-04 13:57:32 +0000524 static void Clear(Isolate* isolate, Address address, Code* target);
christian.plesner.hansen@gmail.com37abdec2009-01-06 14:43:28 +0000525
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000526 friend class IC;
527};
528
529
530class StoreIC: public IC {
531 public:
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000532 StoreIC(FrameDepth depth, Isolate* isolate)
533 : IC(depth, isolate),
534 strict_mode_(Code::GetStrictMode(target()->extra_ic_state())) {
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000535 ASSERT(target()->is_store_stub() || target()->is_keyed_store_stub());
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000536 }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000537
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000538 virtual StrictModeFlag strict_mode() const { return strict_mode_; }
539
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000540 // Code generators for stub routines. Only called once at startup.
ulan@chromium.org57ff8812013-05-10 08:16:55 +0000541 static void GenerateSlow(MacroAssembler* masm);
sgjesse@chromium.orgb302e562010-02-03 11:26:59 +0000542 static void GenerateInitialize(MacroAssembler* masm) { GenerateMiss(masm); }
jkummerow@chromium.org3d00d0a2013-09-04 13:57:32 +0000543 static void GeneratePreMonomorphic(MacroAssembler* masm) {
544 GenerateMiss(masm);
545 }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000546 static void GenerateMiss(MacroAssembler* masm);
sgjesse@chromium.org496c03a2011-02-14 12:05:43 +0000547 static void GenerateMegamorphic(MacroAssembler* masm,
ager@chromium.org9ee27ae2011-03-02 13:43:26 +0000548 StrictModeFlag strict_mode);
kmillikin@chromium.org69ea3962010-07-05 11:01:40 +0000549 static void GenerateNormal(MacroAssembler* masm);
dslomov@chromium.orgb752d402013-06-18 11:54:54 +0000550 static void GenerateRuntimeSetProperty(MacroAssembler* masm,
551 StrictModeFlag strict_mode);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000552
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000553 MUST_USE_RESULT MaybeObject* Store(
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000554 Handle<Object> object,
555 Handle<String> name,
556 Handle<Object> value,
557 JSReceiver::StoreFromKeyed store_mode =
558 JSReceiver::CERTAINLY_NOT_STORE_FROM_KEYED);
559
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000560 protected:
561 virtual Code::Kind kind() const { return Code::STORE_IC; }
562 virtual Handle<Code> megamorphic_stub() {
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000563 if (strict_mode() == kStrictMode) {
564 return isolate()->builtins()->StoreIC_Megamorphic_Strict();
565 } else {
566 return isolate()->builtins()->StoreIC_Megamorphic();
567 }
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000568 }
569 // Stub accessors.
dslomov@chromium.orgb752d402013-06-18 11:54:54 +0000570 virtual Handle<Code> generic_stub() const {
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000571 if (strict_mode() == kStrictMode) {
572 return isolate()->builtins()->StoreIC_Generic_Strict();
573 } else {
574 return isolate()->builtins()->StoreIC_Generic();
575 }
dslomov@chromium.orgb752d402013-06-18 11:54:54 +0000576 }
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000577
mstarzinger@chromium.org2efc3e42013-10-14 08:45:38 +0000578 virtual Handle<Code> slow_stub() const {
579 if (strict_mode() == kStrictMode) {
580 return isolate()->builtins()->StoreIC_Slow_Strict();
581 } else {
582 return isolate()->builtins()->StoreIC_Slow();
583 }
584 }
585
machenbach@chromium.org8e36b5b2013-09-26 07:36:30 +0000586 virtual Handle<Code> pre_monomorphic_stub() {
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000587 return pre_monomorphic_stub(isolate(), strict_mode());
jkummerow@chromium.org3d00d0a2013-09-04 13:57:32 +0000588 }
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000589
590 static Handle<Code> pre_monomorphic_stub(Isolate* isolate,
591 StrictModeFlag strict_mode) {
592 if (strict_mode == kStrictMode) {
593 return isolate->builtins()->StoreIC_PreMonomorphic_Strict();
594 } else {
595 return isolate->builtins()->StoreIC_PreMonomorphic();
596 }
machenbach@chromium.org8e36b5b2013-09-26 07:36:30 +0000597 }
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000598
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000599 virtual Handle<Code> global_proxy_stub() {
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000600 if (strict_mode() == kStrictMode) {
601 return isolate()->builtins()->StoreIC_GlobalProxy_Strict();
602 } else {
603 return isolate()->builtins()->StoreIC_GlobalProxy();
604 }
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000605 }
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000606
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000607 // Update the inline cache and the global stub cache based on the
608 // lookup result.
mvstanton@chromium.orgd16d8532013-01-25 13:29:10 +0000609 void UpdateCaches(LookupResult* lookup,
mvstanton@chromium.orgd16d8532013-01-25 13:29:10 +0000610 Handle<JSObject> receiver,
611 Handle<String> name,
612 Handle<Object> value);
mstarzinger@chromium.org2efc3e42013-10-14 08:45:38 +0000613 virtual Handle<Code> CompileHandler(LookupResult* lookup,
614 Handle<JSObject> receiver,
615 Handle<String> name,
616 Handle<Object> value);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000617
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000618 private:
ager@chromium.org9ee27ae2011-03-02 13:43:26 +0000619 void set_target(Code* code) {
620 // Strict mode must be preserved across IC patching.
ulan@chromium.org65a89c22012-02-14 11:46:07 +0000621 ASSERT(Code::GetStrictMode(code->extra_ic_state()) ==
622 Code::GetStrictMode(target()->extra_ic_state()));
ager@chromium.org9ee27ae2011-03-02 13:43:26 +0000623 IC::set_target(code);
624 }
625
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000626 static Handle<Code> initialize_stub(Isolate* isolate,
627 StrictModeFlag strict_mode) {
628 if (strict_mode == kStrictMode) {
629 return isolate->builtins()->StoreIC_Initialize_Strict();
630 } else {
631 return isolate->builtins()->StoreIC_Initialize();
632 }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000633 }
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000634
jkummerow@chromium.org3d00d0a2013-09-04 13:57:32 +0000635 static void Clear(Isolate* isolate, Address address, Code* target);
fschneider@chromium.orged78ffd2010-07-21 11:05:19 +0000636
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000637 StrictModeFlag strict_mode_;
638
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000639 friend class IC;
640};
641
642
verwaest@chromium.org33e09c82012-10-10 17:07:22 +0000643enum KeyedStoreCheckMap {
644 kDontCheckMap,
645 kCheckMap
646};
647
648
649enum KeyedStoreIncrementLength {
650 kDontIncrementLength,
651 kIncrementLength
652};
653
654
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000655class KeyedStoreIC: public StoreIC {
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000656 public:
jkummerow@chromium.org7bd87f02013-03-20 18:06:29 +0000657 KeyedStoreIC(FrameDepth depth, Isolate* isolate)
658 : StoreIC(depth, isolate) {
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000659 ASSERT(target()->is_keyed_store_stub());
660 }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000661
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000662 MUST_USE_RESULT MaybeObject* Store(Handle<Object> object,
lrn@chromium.org303ada72010-10-27 09:33:13 +0000663 Handle<Object> name,
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000664 Handle<Object> value,
jkummerow@chromium.org59297c72013-01-09 16:32:23 +0000665 ICMissMode force_generic);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000666
667 // Code generators for stub routines. Only called once at startup.
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000668 static void GenerateInitialize(MacroAssembler* masm) {
jkummerow@chromium.org59297c72013-01-09 16:32:23 +0000669 GenerateMiss(masm, MISS);
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000670 }
jkummerow@chromium.org3d00d0a2013-09-04 13:57:32 +0000671 static void GeneratePreMonomorphic(MacroAssembler* masm) {
672 GenerateMiss(masm, MISS);
673 }
jkummerow@chromium.org59297c72013-01-09 16:32:23 +0000674 static void GenerateMiss(MacroAssembler* masm, ICMissMode force_generic);
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000675 static void GenerateSlow(MacroAssembler* masm);
ager@chromium.org9ee27ae2011-03-02 13:43:26 +0000676 static void GenerateRuntimeSetProperty(MacroAssembler* masm,
677 StrictModeFlag strict_mode);
678 static void GenerateGeneric(MacroAssembler* masm, StrictModeFlag strict_mode);
whesse@chromium.org7b260152011-06-20 15:33:18 +0000679 static void GenerateNonStrictArguments(MacroAssembler* masm);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000680
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000681 protected:
682 virtual Code::Kind kind() const { return Code::KEYED_STORE_IC; }
683
ulan@chromium.org57ff8812013-05-10 08:16:55 +0000684 virtual void UpdateMegamorphicCache(Map* map, Name* name, Code* code) { }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000685
machenbach@chromium.org8e36b5b2013-09-26 07:36:30 +0000686 virtual Handle<Code> pre_monomorphic_stub() {
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000687 return pre_monomorphic_stub(isolate(), strict_mode());
jkummerow@chromium.org3d00d0a2013-09-04 13:57:32 +0000688 }
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000689 static Handle<Code> pre_monomorphic_stub(Isolate* isolate,
690 StrictModeFlag strict_mode) {
691 if (strict_mode == kStrictMode) {
692 return isolate->builtins()->KeyedStoreIC_PreMonomorphic_Strict();
693 } else {
694 return isolate->builtins()->KeyedStoreIC_PreMonomorphic();
695 }
jkummerow@chromium.org3d00d0a2013-09-04 13:57:32 +0000696 }
mstarzinger@chromium.org2efc3e42013-10-14 08:45:38 +0000697 virtual Handle<Code> slow_stub() const {
698 if (strict_mode() == kStrictMode) {
699 return isolate()->builtins()->KeyedStoreIC_Slow_Strict();
700 } else {
701 return isolate()->builtins()->KeyedStoreIC_Slow();
702 }
703 }
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000704 virtual Handle<Code> megamorphic_stub() {
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000705 if (strict_mode() == kStrictMode) {
706 return isolate()->builtins()->KeyedStoreIC_Generic_Strict();
707 } else {
708 return isolate()->builtins()->KeyedStoreIC_Generic();
709 }
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000710 }
711
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000712 Handle<Code> StoreElementStub(Handle<JSObject> receiver,
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000713 KeyedAccessStoreMode store_mode);
danno@chromium.orgbee51992013-07-10 14:57:15 +0000714
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000715 private:
ager@chromium.org9ee27ae2011-03-02 13:43:26 +0000716 void set_target(Code* code) {
717 // Strict mode must be preserved across IC patching.
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000718 ASSERT(Code::GetStrictMode(code->extra_ic_state()) == strict_mode());
ager@chromium.org9ee27ae2011-03-02 13:43:26 +0000719 IC::set_target(code);
720 }
721
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000722 // Stub accessors.
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000723 static Handle<Code> initialize_stub(Isolate* isolate,
724 StrictModeFlag strict_mode) {
725 if (strict_mode == kStrictMode) {
726 return isolate->builtins()->KeyedStoreIC_Initialize_Strict();
727 } else {
728 return isolate->builtins()->KeyedStoreIC_Initialize();
729 }
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000730 }
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000731
732 virtual Handle<Code> generic_stub() const {
733 if (strict_mode() == kStrictMode) {
734 return isolate()->builtins()->KeyedStoreIC_Generic_Strict();
735 } else {
736 return isolate()->builtins()->KeyedStoreIC_Generic();
737 }
ager@chromium.org9ee27ae2011-03-02 13:43:26 +0000738 }
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000739
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000740 Handle<Code> non_strict_arguments_stub() {
741 return isolate()->builtins()->KeyedStoreIC_NonStrictArguments();
whesse@chromium.org7b260152011-06-20 15:33:18 +0000742 }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000743
jkummerow@chromium.org3d00d0a2013-09-04 13:57:32 +0000744 static void Clear(Isolate* isolate, Address address, Code* target);
ager@chromium.orgeadaf222009-06-16 09:43:10 +0000745
ulan@chromium.org750145a2013-03-07 15:14:13 +0000746 KeyedAccessStoreMode GetStoreMode(Handle<JSObject> receiver,
747 Handle<Object> key,
748 Handle<Object> value);
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000749
750 Handle<Map> ComputeTransitionedMap(Handle<JSObject> receiver,
ulan@chromium.org750145a2013-03-07 15:14:13 +0000751 KeyedAccessStoreMode store_mode);
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000752
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000753 friend class IC;
754};
755
756
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000757// Type Recording BinaryOpIC, that records the types of the inputs and outputs.
danno@chromium.org40cb8782011-05-25 07:58:50 +0000758class BinaryOpIC: public IC {
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000759 public:
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000760 enum TypeInfo {
761 UNINITIALIZED,
762 SMI,
763 INT32,
hpayer@chromium.org8432c912013-02-28 15:55:26 +0000764 NUMBER,
lrn@chromium.org7516f052011-03-30 08:52:27 +0000765 ODDBALL,
yangguo@chromium.orgfb377212012-11-16 14:43:43 +0000766 STRING, // Only used for addition operation.
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000767 GENERIC
768 };
769
jkummerow@chromium.org25b0e212013-10-04 15:38:52 +0000770 explicit BinaryOpIC(Isolate* isolate) : IC(EXTRA_CALL_FRAME, isolate) { }
dslomov@chromium.orgb752d402013-06-18 11:54:54 +0000771
jkummerow@chromium.org25b0e212013-10-04 15:38:52 +0000772 static Builtins::JavaScript TokenToJSBuiltin(Token::Value op);
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000773
774 static const char* GetName(TypeInfo type_info);
775
jkummerow@chromium.org25b0e212013-10-04 15:38:52 +0000776 MUST_USE_RESULT MaybeObject* Transition(Handle<Object> left,
777 Handle<Object> right);
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000778};
779
780
781class CompareIC: public IC {
782 public:
hpayer@chromium.org8432c912013-02-28 15:55:26 +0000783 // The type/state lattice is defined by the following inequations:
784 // UNINITIALIZED < ...
785 // ... < GENERIC
786 // SMI < NUMBER
yangguo@chromium.org4a9f6552013-03-04 14:46:33 +0000787 // INTERNALIZED_STRING < STRING
hpayer@chromium.org8432c912013-02-28 15:55:26 +0000788 // KNOWN_OBJECT < OBJECT
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000789 enum State {
790 UNINITIALIZED,
yangguo@chromium.orgfb377212012-11-16 14:43:43 +0000791 SMI,
hpayer@chromium.org8432c912013-02-28 15:55:26 +0000792 NUMBER,
yangguo@chromium.orgfb377212012-11-16 14:43:43 +0000793 STRING,
yangguo@chromium.org4a9f6552013-03-04 14:46:33 +0000794 INTERNALIZED_STRING,
795 UNIQUE_NAME, // Symbol or InternalizedString
hpayer@chromium.org8432c912013-02-28 15:55:26 +0000796 OBJECT, // JSObject
797 KNOWN_OBJECT, // JSObject with specific map (faster check)
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000798 GENERIC
799 };
800
dslomov@chromium.orgb752d402013-06-18 11:54:54 +0000801 static State NewInputState(State old_state, Handle<Object> value);
802
803 static Handle<Type> StateToType(Isolate* isolate,
804 State state,
805 Handle<Map> map = Handle<Map>());
806
807 static void StubInfoToType(int stub_minor_key,
808 Handle<Type>* left_type,
809 Handle<Type>* right_type,
810 Handle<Type>* overall_type,
811 Handle<Map> map,
812 Isolate* isolate);
danno@chromium.org41728482013-06-12 22:31:22 +0000813
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000814 CompareIC(Isolate* isolate, Token::Value op)
815 : IC(EXTRA_CALL_FRAME, isolate), op_(op) { }
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000816
817 // Update the inline cache for the given operands.
818 void UpdateCaches(Handle<Object> x, Handle<Object> y);
819
yangguo@chromium.orgfb377212012-11-16 14:43:43 +0000820
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000821 // Factory method for getting an uninitialized compare stub.
hpayer@chromium.org8432c912013-02-28 15:55:26 +0000822 static Handle<Code> GetUninitialized(Isolate* isolate, Token::Value op);
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000823
824 // Helper function for computing the condition for a compare operation.
825 static Condition ComputeCondition(Token::Value op);
826
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000827 static const char* GetStateName(State state);
828
829 private:
yangguo@chromium.orgfb377212012-11-16 14:43:43 +0000830 static bool HasInlinedSmiCode(Address address);
831
832 State TargetState(State old_state,
833 State old_left,
834 State old_right,
835 bool has_inlined_smi_code,
836 Handle<Object> x,
837 Handle<Object> y);
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000838
839 bool strict() const { return op_ == Token::EQ_STRICT; }
840 Condition GetCondition() const { return ComputeCondition(op_); }
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000841
jkummerow@chromium.org3d00d0a2013-09-04 13:57:32 +0000842 static Code* GetRawUninitialized(Isolate* isolate, Token::Value op);
jkummerow@chromium.org212d9642012-05-11 15:02:09 +0000843
jkummerow@chromium.org3d00d0a2013-09-04 13:57:32 +0000844 static void Clear(Isolate* isolate, Address address, Code* target);
jkummerow@chromium.org212d9642012-05-11 15:02:09 +0000845
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000846 Token::Value op_;
jkummerow@chromium.org212d9642012-05-11 15:02:09 +0000847
848 friend class IC;
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000849};
850
ricow@chromium.org9fa09672011-07-25 11:05:35 +0000851
danno@chromium.orgca29dd82013-04-26 11:59:48 +0000852class CompareNilIC: public IC {
853 public:
854 explicit CompareNilIC(Isolate* isolate) : IC(EXTRA_CALL_FRAME, isolate) {}
855
856 MUST_USE_RESULT MaybeObject* CompareNil(Handle<Object> object);
857
858 static Handle<Code> GetUninitialized();
859
danno@chromium.orgca29dd82013-04-26 11:59:48 +0000860 static void Clear(Address address, Code* target);
861
ulan@chromium.org837a67e2013-06-11 15:39:48 +0000862 static MUST_USE_RESULT MaybeObject* DoCompareNilSlow(NilValue nil,
danno@chromium.orgca29dd82013-04-26 11:59:48 +0000863 Handle<Object> object);
864};
865
866
ricow@chromium.org9fa09672011-07-25 11:05:35 +0000867class ToBooleanIC: public IC {
868 public:
rossberg@chromium.orgb99c7542013-05-31 11:40:45 +0000869 explicit ToBooleanIC(Isolate* isolate) : IC(EXTRA_CALL_FRAME, isolate) { }
ricow@chromium.org9fa09672011-07-25 11:05:35 +0000870
rossberg@chromium.orgb99c7542013-05-31 11:40:45 +0000871 MaybeObject* ToBoolean(Handle<Object> object, Code::ExtraICState state);
ricow@chromium.org9fa09672011-07-25 11:05:35 +0000872};
873
874
danno@chromium.org40cb8782011-05-25 07:58:50 +0000875// Helper for BinaryOpIC and CompareIC.
jkummerow@chromium.org212d9642012-05-11 15:02:09 +0000876enum InlinedSmiCheck { ENABLE_INLINED_SMI_CHECK, DISABLE_INLINED_SMI_CHECK };
877void PatchInlinedSmiCode(Address address, InlinedSmiCheck check);
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000878
mstarzinger@chromium.orge3b8d0f2013-02-01 09:06:41 +0000879DECLARE_RUNTIME_FUNCTION(MaybeObject*, KeyedLoadIC_MissFromStubFailure);
jkummerow@chromium.org7bd87f02013-03-20 18:06:29 +0000880DECLARE_RUNTIME_FUNCTION(MaybeObject*, KeyedStoreIC_MissFromStubFailure);
mstarzinger@chromium.orge0e1b0d2013-07-08 08:38:06 +0000881DECLARE_RUNTIME_FUNCTION(MaybeObject*, UnaryOpIC_Miss);
882DECLARE_RUNTIME_FUNCTION(MaybeObject*, StoreIC_MissFromStubFailure);
jkummerow@chromium.orgba72ec82013-07-22 09:21:20 +0000883DECLARE_RUNTIME_FUNCTION(MaybeObject*, ElementsTransitionAndStoreIC_Miss);
jkummerow@chromium.org25b0e212013-10-04 15:38:52 +0000884DECLARE_RUNTIME_FUNCTION(MaybeObject*, BinaryOpIC_Miss);
danno@chromium.orgca29dd82013-04-26 11:59:48 +0000885DECLARE_RUNTIME_FUNCTION(MaybeObject*, CompareNilIC_Miss);
rossberg@chromium.orgb99c7542013-05-31 11:40:45 +0000886DECLARE_RUNTIME_FUNCTION(MaybeObject*, ToBooleanIC_Miss);
danno@chromium.orgca29dd82013-04-26 11:59:48 +0000887
jkummerow@chromium.org59297c72013-01-09 16:32:23 +0000888
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000889} } // namespace v8::internal
890
891#endif // V8_IC_H_