blob: 7113b0b5121918a6b3d04694b422e67c0a63fe79 [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) \
43 ICU(CallIC_Miss) \
lrn@chromium.org1af7e1b2010-06-07 11:12:01 +000044 ICU(KeyedCallIC_Miss) \
sgjesse@chromium.org0b6db592009-07-30 14:48:31 +000045 ICU(StoreIC_Miss) \
ager@chromium.org5c838252010-02-19 08:53:10 +000046 ICU(StoreIC_ArrayLength) \
ulan@chromium.org57ff8812013-05-10 08:16:55 +000047 ICU(StoreIC_Slow) \
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_Slow) \
sgjesse@chromium.org0b6db592009-07-30 14:48:31 +000051 /* Utilities for IC stubs. */ \
sgjesse@chromium.org0b6db592009-07-30 14:48:31 +000052 ICU(StoreCallbackProperty) \
53 ICU(LoadPropertyWithInterceptorOnly) \
54 ICU(LoadPropertyWithInterceptorForLoad) \
55 ICU(LoadPropertyWithInterceptorForCall) \
ager@chromium.org5c838252010-02-19 08:53:10 +000056 ICU(KeyedLoadPropertyWithInterceptor) \
ager@chromium.orgce5e87b2010-03-10 10:24:18 +000057 ICU(StoreInterceptorProperty) \
ricow@chromium.org9fa09672011-07-25 11:05:35 +000058 ICU(CompareIC_Miss) \
jkummerow@chromium.org25b0e212013-10-04 15:38:52 +000059 ICU(BinaryOpIC_Miss) \
danno@chromium.orgca29dd82013-04-26 11:59:48 +000060 ICU(CompareNilIC_Miss) \
ulan@chromium.org77ca49a2013-04-22 09:43:56 +000061 ICU(Unreachable) \
rossberg@chromium.orgb99c7542013-05-31 11:40:45 +000062 ICU(ToBooleanIC_Miss)
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
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +000095 State state() const { return state_; }
ricow@chromium.org64e3a4b2011-12-13 08:07:27 +000096 inline Address address() const;
97
ager@chromium.orgce5e87b2010-03-10 10:24:18 +000098 // Compute the current IC state based on the target stub, receiver and name.
jkummerow@chromium.orgd8a3a142013-10-03 12:15:05 +000099 void UpdateState(Handle<Object> receiver, Handle<Object> name);
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000100 void MarkMonomorphicPrototypeFailure() {
101 state_ = MONOMORPHIC_PROTOTYPE_FAILURE;
102 }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000103
104 // Clear the inline cache to initial state.
jkummerow@chromium.org3d00d0a2013-09-04 13:57:32 +0000105 static void Clear(Isolate* isolate, Address address);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000106
107 // Computes the reloc info for this IC. This is a fairly expensive
108 // operation as it has to search through the heap to find the code
109 // object that contains this IC site.
ager@chromium.org236ad962008-09-25 09:45:57 +0000110 RelocInfo::Mode ComputeMode();
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000111
112 // Returns if this IC is for contextual (no explicit receiver)
113 // access to properties.
mvstanton@chromium.orgc47dff52013-01-23 16:28:41 +0000114 bool IsUndeclaredGlobal(Handle<Object> receiver) {
kmillikin@chromium.org13bd2942009-12-16 15:36:05 +0000115 if (receiver->IsGlobalObject()) {
mvstanton@chromium.orgc47dff52013-01-23 16:28:41 +0000116 return SlowIsUndeclaredGlobal();
kmillikin@chromium.org13bd2942009-12-16 15:36:05 +0000117 } else {
mvstanton@chromium.orgc47dff52013-01-23 16:28:41 +0000118 ASSERT(!SlowIsUndeclaredGlobal());
kmillikin@chromium.org13bd2942009-12-16 15:36:05 +0000119 return false;
120 }
121 }
122
mvstanton@chromium.orgc47dff52013-01-23 16:28:41 +0000123 bool SlowIsUndeclaredGlobal() {
ager@chromium.org236ad962008-09-25 09:45:57 +0000124 return ComputeMode() == RelocInfo::CODE_TARGET_CONTEXT;
125 }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000126
machenbach@chromium.orgb5be0a92013-11-15 10:32:41 +0000127#ifdef DEBUG
128 bool IsLoadStub() {
129 return target()->is_load_stub() || target()->is_keyed_load_stub();
130 }
131
132 bool IsStoreStub() {
133 return target()->is_store_stub() || target()->is_keyed_store_stub();
134 }
135
136 bool IsCallStub() {
137 return target()->is_call_stub() || target()->is_keyed_call_stub();
138 }
139#endif
140
kmillikin@chromium.org69ea3962010-07-05 11:01:40 +0000141 // Determines which map must be used for keeping the code stub.
142 // These methods should not be called with undefined or null.
machenbach@chromium.orgb5be0a92013-11-15 10:32:41 +0000143 static inline InlineCacheHolderFlag GetCodeCacheForObject(Object* object);
144 // TODO(verwaest): This currently returns a HeapObject rather than JSObject*
145 // since loading the IC for loading the length from strings are stored on
146 // the string map directly, rather than on the JSObject-typed prototype.
147 static inline HeapObject* GetCodeCacheHolder(Isolate* isolate,
148 Object* object,
149 InlineCacheHolderFlag holder);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000150
machenbach@chromium.orgaf9cfcb2013-11-19 11:05:18 +0000151 static inline InlineCacheHolderFlag GetCodeCacheFlag(Type* type);
152 static inline Handle<Map> GetCodeCacheHolder(InlineCacheHolderFlag flag,
153 Type* type,
154 Isolate* isolate);
155
machenbach@chromium.org8e36b5b2013-09-26 07:36:30 +0000156 static bool IsCleared(Code* code) {
157 InlineCacheState state = code->ic_state();
158 return state == UNINITIALIZED || state == PREMONOMORPHIC;
159 }
160
machenbach@chromium.orgaf9cfcb2013-11-19 11:05:18 +0000161 // Utility functions to convert maps to types and back. There are two special
162 // cases:
163 // - The heap_number_map is used as a marker which includes heap numbers as
164 // well as smis.
165 // - The oddball map is only used for booleans.
166 static Handle<Map> TypeToMap(Type* type, Isolate* isolate);
167 static Type* MapToType(Handle<Map> type);
168
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000169 protected:
machenbach@chromium.orgb5be0a92013-11-15 10:32:41 +0000170 // Get the call-site target; used for determining the state.
171 Handle<Code> target() const { return target_; }
172
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000173 Address fp() const { return fp_; }
174 Address pc() const { return *pc_address_; }
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000175 Isolate* isolate() const { return isolate_; }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000176
ager@chromium.org65dad4b2009-04-23 08:48:43 +0000177#ifdef ENABLE_DEBUGGER_SUPPORT
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000178 // Computes the address in the original code when the code running is
179 // containing break points (calls to DebugBreakXXX builtins).
ricow@chromium.org64e3a4b2011-12-13 08:07:27 +0000180 Address OriginalCodeAddress() const;
ager@chromium.org65dad4b2009-04-23 08:48:43 +0000181#endif
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000182
183 // Set the call-site target.
bmeurer@chromium.orgc9913f02013-10-24 06:31:36 +0000184 void set_target(Code* code) {
185 SetTargetAtAddress(address(), code);
186 target_set_ = true;
187 }
188
189 bool is_target_set() { return target_set_; }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000190
191#ifdef DEBUG
ricow@chromium.org64e3a4b2011-12-13 08:07:27 +0000192 char TransitionMarkFromState(IC::State state);
193
jkummerow@chromium.orgd8a3a142013-10-03 12:15:05 +0000194 void TraceIC(const char* type, Handle<Object> name);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000195#endif
196
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000197 Failure* TypeError(const char* type,
198 Handle<Object> object,
199 Handle<Object> key);
200 Failure* ReferenceError(const char* type, Handle<String> name);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000201
202 // Access the target code for the given IC address.
203 static inline Code* GetTargetAtAddress(Address address);
204 static inline void SetTargetAtAddress(Address address, Code* target);
jkummerow@chromium.orgf7a58842012-02-21 10:08:21 +0000205 static void PostPatching(Address address, Code* target, Code* old_target);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000206
mstarzinger@chromium.org2efc3e42013-10-14 08:45:38 +0000207 // Compute the handler either by compiling or by retrieving a cached version.
208 Handle<Code> ComputeHandler(LookupResult* lookup,
machenbach@chromium.orgb5be0a92013-11-15 10:32:41 +0000209 Handle<Object> object,
mstarzinger@chromium.org2efc3e42013-10-14 08:45:38 +0000210 Handle<String> name,
211 Handle<Object> value = Handle<Code>::null());
212 virtual Handle<Code> CompileHandler(LookupResult* lookup,
machenbach@chromium.orgb5be0a92013-11-15 10:32:41 +0000213 Handle<Object> object,
mstarzinger@chromium.org2efc3e42013-10-14 08:45:38 +0000214 Handle<String> name,
machenbach@chromium.orgb5be0a92013-11-15 10:32:41 +0000215 Handle<Object> value,
216 InlineCacheHolderFlag cache_holder) {
mstarzinger@chromium.org2efc3e42013-10-14 08:45:38 +0000217 UNREACHABLE();
218 return Handle<Code>::null();
219 }
machenbach@chromium.orgaf9cfcb2013-11-19 11:05:18 +0000220
221 void UpdateMonomorphicIC(Handle<Type> type,
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000222 Handle<Code> handler,
223 Handle<String> name);
danno@chromium.orgbee51992013-07-10 14:57:15 +0000224
machenbach@chromium.orgaf9cfcb2013-11-19 11:05:18 +0000225 bool UpdatePolymorphicIC(Handle<Type> type,
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000226 Handle<String> name,
227 Handle<Code> code);
danno@chromium.orgbee51992013-07-10 14:57:15 +0000228
machenbach@chromium.orgaf9cfcb2013-11-19 11:05:18 +0000229 virtual void UpdateMegamorphicCache(Type* type, Name* name, Code* code);
230
ulan@chromium.org6e196bf2013-03-13 09:38:22 +0000231 void CopyICToMegamorphicCache(Handle<String> name);
machenbach@chromium.orgaf9cfcb2013-11-19 11:05:18 +0000232 bool IsTransitionOfMonomorphicTarget(Type* type);
233 void PatchCache(Handle<Type> type,
mvstanton@chromium.orgd16d8532013-01-25 13:29:10 +0000234 Handle<String> name,
235 Handle<Code> code);
mstarzinger@chromium.org2efc3e42013-10-14 08:45:38 +0000236 virtual Code::Kind kind() const {
237 UNREACHABLE();
238 return Code::STUB;
239 }
240 virtual Handle<Code> slow_stub() const {
241 UNREACHABLE();
242 return Handle<Code>::null();
243 }
mvstanton@chromium.orgd16d8532013-01-25 13:29:10 +0000244 virtual Handle<Code> megamorphic_stub() {
245 UNREACHABLE();
246 return Handle<Code>::null();
247 }
mstarzinger@chromium.org068ea0a2013-01-30 09:39:44 +0000248 virtual Handle<Code> generic_stub() const {
249 UNREACHABLE();
250 return Handle<Code>::null();
251 }
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000252 virtual StrictModeFlag strict_mode() const { return kNonStrictMode; }
jkummerow@chromium.orgd8a3a142013-10-03 12:15:05 +0000253 bool TryRemoveInvalidPrototypeDependentStub(Handle<Object> receiver,
254 Handle<String> name);
mstarzinger@chromium.org2efc3e42013-10-14 08:45:38 +0000255 void TryRemoveInvalidHandlers(Handle<Map> map, Handle<String> name);
mvstanton@chromium.orgd16d8532013-01-25 13:29:10 +0000256
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000257 private:
machenbach@chromium.orgb5be0a92013-11-15 10:32:41 +0000258 Code* raw_target() const { return GetTargetAtAddress(address()); }
259
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000260 // Frame pointer for the frame that uses (calls) the IC.
261 Address fp_;
262
263 // All access to the program counter of an IC structure is indirect
264 // to make the code GC safe. This feature is crucial since
265 // GetProperty and SetProperty are called and they in turn might
266 // invoke the garbage collector.
267 Address* pc_address_;
mads.s.ager31e71382008-08-13 09:32:07 +0000268
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000269 Isolate* isolate_;
270
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000271 // The original code target that missed.
272 Handle<Code> target_;
273 State state_;
bmeurer@chromium.orgc9913f02013-10-24 06:31:36 +0000274 bool target_set_;
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000275
mads.s.ager31e71382008-08-13 09:32:07 +0000276 DISALLOW_IMPLICIT_CONSTRUCTORS(IC);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000277};
278
279
280// An IC_Utility encapsulates IC::UtilityId. It exists mainly because you
281// cannot make forward declarations to an enum.
282class IC_Utility {
283 public:
284 explicit IC_Utility(IC::UtilityId id)
285 : address_(IC::AddressFromUtilityId(id)), id_(id) {}
286
287 Address address() const { return address_; }
288
289 IC::UtilityId id() const { return id_; }
290 private:
291 Address address_;
292 IC::UtilityId id_;
293};
294
295
lrn@chromium.org1af7e1b2010-06-07 11:12:01 +0000296class CallICBase: public IC {
danno@chromium.org40cb8782011-05-25 07:58:50 +0000297 public:
298 class Contextual: public BitField<bool, 0, 1> {};
299 class StringStubState: public BitField<StringStubFeedback, 1, 1> {};
300
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000301 // Returns a JSFunction or a Failure.
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000302 MUST_USE_RESULT MaybeObject* LoadFunction(Handle<Object> object,
lrn@chromium.org303ada72010-10-27 09:33:13 +0000303 Handle<String> name);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000304
lrn@chromium.org1af7e1b2010-06-07 11:12:01 +0000305 protected:
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000306 CallICBase(Code::Kind kind, Isolate* isolate)
307 : IC(EXTRA_CALL_FRAME, isolate), kind_(kind) {}
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000308
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000309 virtual Code::ExtraICState extra_ic_state() { return Code::kNoExtraICState; }
erik.corry@gmail.com0511e242011-01-19 11:11:08 +0000310
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000311 // Compute a monomorphic stub if possible, otherwise return a null handle.
312 Handle<Code> ComputeMonomorphicStub(LookupResult* lookup,
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000313 Handle<Object> object,
314 Handle<String> name);
erik.corry@gmail.com0511e242011-01-19 11:11:08 +0000315
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000316 // Update the inline cache and the global stub cache based on the lookup
317 // result.
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000318 void UpdateCaches(LookupResult* lookup,
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000319 Handle<Object> object,
320 Handle<String> name);
321
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000322 // Returns a JSFunction if the object can be called as a function, and
323 // patches the stack to be ready for the call. Otherwise, it returns the
324 // undefined value.
325 Handle<Object> TryCallAsFunction(Handle<Object> object);
ager@chromium.org9258b6b2008-09-11 09:11:10 +0000326
fschneider@chromium.org3a5fd782011-02-24 10:10:44 +0000327 void ReceiverToObjectIfRequired(Handle<Object> callee, Handle<Object> object);
sgjesse@chromium.orgb302e562010-02-03 11:26:59 +0000328
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000329 static void Clear(Address address, Code* target);
danno@chromium.org40cb8782011-05-25 07:58:50 +0000330
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000331 // Platform-specific code generation functions used by both call and
332 // keyed call.
333 static void GenerateMiss(MacroAssembler* masm,
334 int argc,
335 IC::UtilityId id,
336 Code::ExtraICState extra_state);
337
338 static void GenerateNormal(MacroAssembler* masm, int argc);
339
340 static void GenerateMonomorphicCacheProbe(MacroAssembler* masm,
341 int argc,
342 Code::Kind kind,
343 Code::ExtraICState extra_state);
344
jkummerow@chromium.orgd8a3a142013-10-03 12:15:05 +0000345 virtual Handle<Code> megamorphic_stub();
346 virtual Handle<Code> pre_monomorphic_stub();
347
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000348 Code::Kind kind_;
349
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000350 friend class IC;
351};
352
353
lrn@chromium.org1af7e1b2010-06-07 11:12:01 +0000354class CallIC: public CallICBase {
355 public:
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000356 explicit CallIC(Isolate* isolate)
357 : CallICBase(Code::CALL_IC, isolate),
358 extra_ic_state_(target()->extra_ic_state()) {
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000359 ASSERT(target()->is_call_stub());
360 }
lrn@chromium.org1af7e1b2010-06-07 11:12:01 +0000361
362 // Code generator routines.
danno@chromium.org40cb8782011-05-25 07:58:50 +0000363 static void GenerateInitialize(MacroAssembler* masm,
364 int argc,
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000365 Code::ExtraICState extra_state) {
366 GenerateMiss(masm, argc, extra_state);
lrn@chromium.org1af7e1b2010-06-07 11:12:01 +0000367 }
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000368
danno@chromium.org40cb8782011-05-25 07:58:50 +0000369 static void GenerateMiss(MacroAssembler* masm,
370 int argc,
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000371 Code::ExtraICState extra_state) {
372 CallICBase::GenerateMiss(masm, argc, IC::kCallIC_Miss, extra_state);
373 }
374
danno@chromium.org40cb8782011-05-25 07:58:50 +0000375 static void GenerateMegamorphic(MacroAssembler* masm,
376 int argc,
377 Code::ExtraICState extra_ic_state);
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000378
379 static void GenerateNormal(MacroAssembler* masm, int argc) {
380 CallICBase::GenerateNormal(masm, argc);
381 GenerateMiss(masm, argc, Code::kNoExtraICState);
382 }
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000383 bool TryUpdateExtraICState(LookupResult* lookup, Handle<Object> object);
384
385 protected:
386 virtual Code::ExtraICState extra_ic_state() { return extra_ic_state_; }
387
388 private:
389 Code::ExtraICState extra_ic_state_;
lrn@chromium.org1af7e1b2010-06-07 11:12:01 +0000390};
391
392
393class KeyedCallIC: public CallICBase {
394 public:
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000395 explicit KeyedCallIC(Isolate* isolate)
396 : CallICBase(Code::KEYED_CALL_IC, isolate) {
lrn@chromium.org1af7e1b2010-06-07 11:12:01 +0000397 ASSERT(target()->is_keyed_call_stub());
398 }
399
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000400 MUST_USE_RESULT MaybeObject* LoadFunction(Handle<Object> object,
lrn@chromium.org303ada72010-10-27 09:33:13 +0000401 Handle<Object> key);
lrn@chromium.org1af7e1b2010-06-07 11:12:01 +0000402
403 // Code generator routines.
404 static void GenerateInitialize(MacroAssembler* masm, int argc) {
405 GenerateMiss(masm, argc);
406 }
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000407
408 static void GenerateMiss(MacroAssembler* masm, int argc) {
409 CallICBase::GenerateMiss(masm, argc, IC::kKeyedCallIC_Miss,
410 Code::kNoExtraICState);
411 }
412
lrn@chromium.org1af7e1b2010-06-07 11:12:01 +0000413 static void GenerateMegamorphic(MacroAssembler* masm, int argc);
414 static void GenerateNormal(MacroAssembler* masm, int argc);
whesse@chromium.org7b260152011-06-20 15:33:18 +0000415 static void GenerateNonStrictArguments(MacroAssembler* masm, int argc);
lrn@chromium.org1af7e1b2010-06-07 11:12:01 +0000416};
417
418
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000419class LoadIC: public IC {
420 public:
mstarzinger@chromium.orge3b8d0f2013-02-01 09:06:41 +0000421 explicit LoadIC(FrameDepth depth, Isolate* isolate) : IC(depth, isolate) {
machenbach@chromium.orgb5be0a92013-11-15 10:32:41 +0000422 ASSERT(IsLoadStub());
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000423 }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000424
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000425 // Code generator routines.
ager@chromium.org5c838252010-02-19 08:53:10 +0000426 static void GenerateInitialize(MacroAssembler* masm) { GenerateMiss(masm); }
427 static void GeneratePreMonomorphic(MacroAssembler* masm) {
428 GenerateMiss(masm);
429 }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000430 static void GenerateMiss(MacroAssembler* masm);
431 static void GenerateMegamorphic(MacroAssembler* masm);
432 static void GenerateNormal(MacroAssembler* masm);
verwaest@chromium.orgd4be0f02013-06-05 13:39:03 +0000433 static void GenerateRuntimeGetProperty(MacroAssembler* masm);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000434
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000435 MUST_USE_RESULT MaybeObject* Load(Handle<Object> object,
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000436 Handle<String> name);
437
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000438 protected:
439 virtual Code::Kind kind() const { return Code::LOAD_IC; }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000440
dslomov@chromium.org4a35c5a2013-09-13 07:28:52 +0000441 virtual Handle<Code> slow_stub() const {
verwaest@chromium.orgd4be0f02013-06-05 13:39:03 +0000442 return isolate()->builtins()->LoadIC_Slow();
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000443 }
444
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000445 virtual Handle<Code> megamorphic_stub() {
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000446 return isolate()->builtins()->LoadIC_Megamorphic();
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000447 }
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000448
449 // Update the inline cache and the global stub cache based on the
450 // lookup result.
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000451 void UpdateCaches(LookupResult* lookup,
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000452 Handle<Object> object,
453 Handle<String> name);
danno@chromium.orgbee51992013-07-10 14:57:15 +0000454
mstarzinger@chromium.org2efc3e42013-10-14 08:45:38 +0000455 virtual Handle<Code> CompileHandler(LookupResult* lookup,
machenbach@chromium.orgb5be0a92013-11-15 10:32:41 +0000456 Handle<Object> object,
mstarzinger@chromium.org2efc3e42013-10-14 08:45:38 +0000457 Handle<String> name,
machenbach@chromium.orgb5be0a92013-11-15 10:32:41 +0000458 Handle<Object> unused,
459 InlineCacheHolderFlag cache_holder);
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000460
461 private:
462 // Stub accessors.
jkummerow@chromium.org3d00d0a2013-09-04 13:57:32 +0000463 static Handle<Code> initialize_stub(Isolate* isolate) {
464 return isolate->builtins()->LoadIC_Initialize();
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000465 }
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000466
machenbach@chromium.org8e36b5b2013-09-26 07:36:30 +0000467 static Handle<Code> pre_monomorphic_stub(Isolate* isolate) {
468 return isolate->builtins()->LoadIC_PreMonomorphic();
469 }
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000470
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000471 virtual Handle<Code> pre_monomorphic_stub() {
machenbach@chromium.org8e36b5b2013-09-26 07:36:30 +0000472 return pre_monomorphic_stub(isolate());
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000473 }
474
mstarzinger@chromium.org2efc3e42013-10-14 08:45:38 +0000475 Handle<Code> SimpleFieldLoad(int offset,
476 bool inobject = true,
477 Representation representation =
478 Representation::Tagged());
479
jkummerow@chromium.org3d00d0a2013-09-04 13:57:32 +0000480 static void Clear(Isolate* isolate, Address address, Code* target);
ager@chromium.org5ec48922009-05-05 07:25:34 +0000481
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000482 friend class IC;
483};
484
485
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000486class KeyedLoadIC: public LoadIC {
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000487 public:
mstarzinger@chromium.orge3b8d0f2013-02-01 09:06:41 +0000488 explicit KeyedLoadIC(FrameDepth depth, Isolate* isolate)
489 : LoadIC(depth, isolate) {
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000490 ASSERT(target()->is_keyed_load_stub());
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000491 }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000492
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000493 MUST_USE_RESULT MaybeObject* Load(Handle<Object> object,
machenbach@chromium.orgb5be0a92013-11-15 10:32:41 +0000494 Handle<Object> key);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000495
496 // Code generator routines.
machenbach@chromium.orgaf9cfcb2013-11-19 11:05:18 +0000497 static void GenerateMiss(MacroAssembler* masm);
ager@chromium.org5c838252010-02-19 08:53:10 +0000498 static void GenerateRuntimeGetProperty(MacroAssembler* masm);
machenbach@chromium.orgaf9cfcb2013-11-19 11:05:18 +0000499 static void GenerateInitialize(MacroAssembler* masm) { GenerateMiss(masm); }
ager@chromium.org5c838252010-02-19 08:53:10 +0000500 static void GeneratePreMonomorphic(MacroAssembler* masm) {
machenbach@chromium.orgaf9cfcb2013-11-19 11:05:18 +0000501 GenerateMiss(masm);
ager@chromium.org5c838252010-02-19 08:53:10 +0000502 }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000503 static void GenerateGeneric(MacroAssembler* masm);
fschneider@chromium.org0c20e672010-01-14 15:28:53 +0000504 static void GenerateString(MacroAssembler* masm);
ager@chromium.org5c838252010-02-19 08:53:10 +0000505 static void GenerateIndexedInterceptor(MacroAssembler* masm);
whesse@chromium.org7b260152011-06-20 15:33:18 +0000506 static void GenerateNonStrictArguments(MacroAssembler* masm);
ager@chromium.org3811b432009-10-28 14:53:37 +0000507
kasperl@chromium.orgeac059f2010-01-25 11:02:06 +0000508 // Bit mask to be tested against bit field for the cases when
509 // generic stub should go into slow case.
510 // Access check is necessary explicitly since generic stub does not perform
511 // map checks.
512 static const int kSlowCaseBitFieldMask =
513 (1 << Map::kIsAccessCheckNeeded) | (1 << Map::kHasIndexedInterceptor);
514
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000515 protected:
516 virtual Code::Kind kind() const { return Code::KEYED_LOAD_IC; }
517
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000518 Handle<Code> LoadElementStub(Handle<JSObject> receiver);
519
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000520 virtual Handle<Code> megamorphic_stub() {
521 return isolate()->builtins()->KeyedLoadIC_Generic();
522 }
523 virtual Handle<Code> generic_stub() const {
524 return isolate()->builtins()->KeyedLoadIC_Generic();
525 }
dslomov@chromium.org4a35c5a2013-09-13 07:28:52 +0000526 virtual Handle<Code> slow_stub() const {
527 return isolate()->builtins()->KeyedLoadIC_Slow();
528 }
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000529
machenbach@chromium.orgaf9cfcb2013-11-19 11:05:18 +0000530 virtual void UpdateMegamorphicCache(Type* type, Name* name, Code* code) { }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000531
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000532 private:
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000533 // Stub accessors.
jkummerow@chromium.org3d00d0a2013-09-04 13:57:32 +0000534 static Handle<Code> initialize_stub(Isolate* isolate) {
535 return isolate->builtins()->KeyedLoadIC_Initialize();
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000536 }
machenbach@chromium.org8e36b5b2013-09-26 07:36:30 +0000537 static Handle<Code> pre_monomorphic_stub(Isolate* isolate) {
538 return isolate->builtins()->KeyedLoadIC_PreMonomorphic();
539 }
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000540 virtual Handle<Code> pre_monomorphic_stub() {
machenbach@chromium.org8e36b5b2013-09-26 07:36:30 +0000541 return pre_monomorphic_stub(isolate());
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000542 }
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000543 Handle<Code> indexed_interceptor_stub() {
544 return isolate()->builtins()->KeyedLoadIC_IndexedInterceptor();
ager@chromium.org5c838252010-02-19 08:53:10 +0000545 }
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000546 Handle<Code> non_strict_arguments_stub() {
547 return isolate()->builtins()->KeyedLoadIC_NonStrictArguments();
whesse@chromium.org7b260152011-06-20 15:33:18 +0000548 }
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000549 Handle<Code> string_stub() {
550 return isolate()->builtins()->KeyedLoadIC_String();
551 }
ager@chromium.org5c838252010-02-19 08:53:10 +0000552
jkummerow@chromium.org3d00d0a2013-09-04 13:57:32 +0000553 static void Clear(Isolate* isolate, Address address, Code* target);
christian.plesner.hansen@gmail.com37abdec2009-01-06 14:43:28 +0000554
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000555 friend class IC;
556};
557
558
559class StoreIC: public IC {
560 public:
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000561 StoreIC(FrameDepth depth, Isolate* isolate)
562 : IC(depth, isolate),
563 strict_mode_(Code::GetStrictMode(target()->extra_ic_state())) {
machenbach@chromium.orgb5be0a92013-11-15 10:32:41 +0000564 ASSERT(IsStoreStub());
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000565 }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000566
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000567 virtual StrictModeFlag strict_mode() const { return strict_mode_; }
568
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000569 // Code generators for stub routines. Only called once at startup.
ulan@chromium.org57ff8812013-05-10 08:16:55 +0000570 static void GenerateSlow(MacroAssembler* masm);
sgjesse@chromium.orgb302e562010-02-03 11:26:59 +0000571 static void GenerateInitialize(MacroAssembler* masm) { GenerateMiss(masm); }
jkummerow@chromium.org3d00d0a2013-09-04 13:57:32 +0000572 static void GeneratePreMonomorphic(MacroAssembler* masm) {
573 GenerateMiss(masm);
574 }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000575 static void GenerateMiss(MacroAssembler* masm);
sgjesse@chromium.org496c03a2011-02-14 12:05:43 +0000576 static void GenerateMegamorphic(MacroAssembler* masm,
ager@chromium.org9ee27ae2011-03-02 13:43:26 +0000577 StrictModeFlag strict_mode);
kmillikin@chromium.org69ea3962010-07-05 11:01:40 +0000578 static void GenerateNormal(MacroAssembler* masm);
dslomov@chromium.orgb752d402013-06-18 11:54:54 +0000579 static void GenerateRuntimeSetProperty(MacroAssembler* masm,
580 StrictModeFlag strict_mode);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000581
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000582 MUST_USE_RESULT MaybeObject* Store(
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000583 Handle<Object> object,
584 Handle<String> name,
585 Handle<Object> value,
586 JSReceiver::StoreFromKeyed store_mode =
587 JSReceiver::CERTAINLY_NOT_STORE_FROM_KEYED);
588
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000589 protected:
590 virtual Code::Kind kind() const { return Code::STORE_IC; }
591 virtual Handle<Code> megamorphic_stub() {
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000592 if (strict_mode() == kStrictMode) {
593 return isolate()->builtins()->StoreIC_Megamorphic_Strict();
594 } else {
595 return isolate()->builtins()->StoreIC_Megamorphic();
596 }
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000597 }
598 // Stub accessors.
dslomov@chromium.orgb752d402013-06-18 11:54:54 +0000599 virtual Handle<Code> generic_stub() const {
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000600 if (strict_mode() == kStrictMode) {
601 return isolate()->builtins()->StoreIC_Generic_Strict();
602 } else {
603 return isolate()->builtins()->StoreIC_Generic();
604 }
dslomov@chromium.orgb752d402013-06-18 11:54:54 +0000605 }
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000606
mstarzinger@chromium.org2efc3e42013-10-14 08:45:38 +0000607 virtual Handle<Code> slow_stub() const {
608 if (strict_mode() == kStrictMode) {
609 return isolate()->builtins()->StoreIC_Slow_Strict();
610 } else {
611 return isolate()->builtins()->StoreIC_Slow();
612 }
613 }
614
machenbach@chromium.org8e36b5b2013-09-26 07:36:30 +0000615 virtual Handle<Code> pre_monomorphic_stub() {
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000616 return pre_monomorphic_stub(isolate(), strict_mode());
jkummerow@chromium.org3d00d0a2013-09-04 13:57:32 +0000617 }
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000618
619 static Handle<Code> pre_monomorphic_stub(Isolate* isolate,
620 StrictModeFlag strict_mode) {
621 if (strict_mode == kStrictMode) {
622 return isolate->builtins()->StoreIC_PreMonomorphic_Strict();
623 } else {
624 return isolate->builtins()->StoreIC_PreMonomorphic();
625 }
machenbach@chromium.org8e36b5b2013-09-26 07:36:30 +0000626 }
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000627
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000628 virtual Handle<Code> global_proxy_stub() {
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000629 if (strict_mode() == kStrictMode) {
630 return isolate()->builtins()->StoreIC_GlobalProxy_Strict();
631 } else {
632 return isolate()->builtins()->StoreIC_GlobalProxy();
633 }
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000634 }
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000635
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000636 // Update the inline cache and the global stub cache based on the
637 // lookup result.
mvstanton@chromium.orgd16d8532013-01-25 13:29:10 +0000638 void UpdateCaches(LookupResult* lookup,
mvstanton@chromium.orgd16d8532013-01-25 13:29:10 +0000639 Handle<JSObject> receiver,
640 Handle<String> name,
641 Handle<Object> value);
mstarzinger@chromium.org2efc3e42013-10-14 08:45:38 +0000642 virtual Handle<Code> CompileHandler(LookupResult* lookup,
machenbach@chromium.orgb5be0a92013-11-15 10:32:41 +0000643 Handle<Object> object,
mstarzinger@chromium.org2efc3e42013-10-14 08:45:38 +0000644 Handle<String> name,
machenbach@chromium.orgb5be0a92013-11-15 10:32:41 +0000645 Handle<Object> value,
646 InlineCacheHolderFlag cache_holder);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000647
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000648 private:
ager@chromium.org9ee27ae2011-03-02 13:43:26 +0000649 void set_target(Code* code) {
650 // Strict mode must be preserved across IC patching.
ulan@chromium.org65a89c22012-02-14 11:46:07 +0000651 ASSERT(Code::GetStrictMode(code->extra_ic_state()) ==
652 Code::GetStrictMode(target()->extra_ic_state()));
ager@chromium.org9ee27ae2011-03-02 13:43:26 +0000653 IC::set_target(code);
654 }
655
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000656 static Handle<Code> initialize_stub(Isolate* isolate,
657 StrictModeFlag strict_mode) {
658 if (strict_mode == kStrictMode) {
659 return isolate->builtins()->StoreIC_Initialize_Strict();
660 } else {
661 return isolate->builtins()->StoreIC_Initialize();
662 }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000663 }
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000664
jkummerow@chromium.org3d00d0a2013-09-04 13:57:32 +0000665 static void Clear(Isolate* isolate, Address address, Code* target);
fschneider@chromium.orged78ffd2010-07-21 11:05:19 +0000666
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000667 StrictModeFlag strict_mode_;
668
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000669 friend class IC;
670};
671
672
verwaest@chromium.org33e09c82012-10-10 17:07:22 +0000673enum KeyedStoreCheckMap {
674 kDontCheckMap,
675 kCheckMap
676};
677
678
679enum KeyedStoreIncrementLength {
680 kDontIncrementLength,
681 kIncrementLength
682};
683
684
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000685class KeyedStoreIC: public StoreIC {
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000686 public:
jkummerow@chromium.org7bd87f02013-03-20 18:06:29 +0000687 KeyedStoreIC(FrameDepth depth, Isolate* isolate)
688 : StoreIC(depth, isolate) {
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000689 ASSERT(target()->is_keyed_store_stub());
690 }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000691
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000692 MUST_USE_RESULT MaybeObject* Store(Handle<Object> object,
lrn@chromium.org303ada72010-10-27 09:33:13 +0000693 Handle<Object> name,
machenbach@chromium.orgb5be0a92013-11-15 10:32:41 +0000694 Handle<Object> value);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000695
696 // Code generators for stub routines. Only called once at startup.
machenbach@chromium.orgaf9cfcb2013-11-19 11:05:18 +0000697 static void GenerateInitialize(MacroAssembler* masm) { GenerateMiss(masm); }
jkummerow@chromium.org3d00d0a2013-09-04 13:57:32 +0000698 static void GeneratePreMonomorphic(MacroAssembler* masm) {
machenbach@chromium.orgaf9cfcb2013-11-19 11:05:18 +0000699 GenerateMiss(masm);
jkummerow@chromium.org3d00d0a2013-09-04 13:57:32 +0000700 }
machenbach@chromium.orgaf9cfcb2013-11-19 11:05:18 +0000701 static void GenerateMiss(MacroAssembler* masm);
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000702 static void GenerateSlow(MacroAssembler* masm);
ager@chromium.org9ee27ae2011-03-02 13:43:26 +0000703 static void GenerateRuntimeSetProperty(MacroAssembler* masm,
704 StrictModeFlag strict_mode);
705 static void GenerateGeneric(MacroAssembler* masm, StrictModeFlag strict_mode);
whesse@chromium.org7b260152011-06-20 15:33:18 +0000706 static void GenerateNonStrictArguments(MacroAssembler* masm);
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000707
ager@chromium.orgea91cc52011-05-23 06:06:11 +0000708 protected:
709 virtual Code::Kind kind() const { return Code::KEYED_STORE_IC; }
710
machenbach@chromium.orgaf9cfcb2013-11-19 11:05:18 +0000711 virtual void UpdateMegamorphicCache(Type* type, Name* name, Code* code) { }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000712
machenbach@chromium.org8e36b5b2013-09-26 07:36:30 +0000713 virtual Handle<Code> pre_monomorphic_stub() {
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000714 return pre_monomorphic_stub(isolate(), strict_mode());
jkummerow@chromium.org3d00d0a2013-09-04 13:57:32 +0000715 }
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000716 static Handle<Code> pre_monomorphic_stub(Isolate* isolate,
717 StrictModeFlag strict_mode) {
718 if (strict_mode == kStrictMode) {
719 return isolate->builtins()->KeyedStoreIC_PreMonomorphic_Strict();
720 } else {
721 return isolate->builtins()->KeyedStoreIC_PreMonomorphic();
722 }
jkummerow@chromium.org3d00d0a2013-09-04 13:57:32 +0000723 }
mstarzinger@chromium.org2efc3e42013-10-14 08:45:38 +0000724 virtual Handle<Code> slow_stub() const {
725 if (strict_mode() == kStrictMode) {
726 return isolate()->builtins()->KeyedStoreIC_Slow_Strict();
727 } else {
728 return isolate()->builtins()->KeyedStoreIC_Slow();
729 }
730 }
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000731 virtual Handle<Code> megamorphic_stub() {
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000732 if (strict_mode() == kStrictMode) {
733 return isolate()->builtins()->KeyedStoreIC_Generic_Strict();
734 } else {
735 return isolate()->builtins()->KeyedStoreIC_Generic();
736 }
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000737 }
738
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000739 Handle<Code> StoreElementStub(Handle<JSObject> receiver,
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000740 KeyedAccessStoreMode store_mode);
danno@chromium.orgbee51992013-07-10 14:57:15 +0000741
mvstanton@chromium.org6bec0092013-01-23 13:46:53 +0000742 private:
ager@chromium.org9ee27ae2011-03-02 13:43:26 +0000743 void set_target(Code* code) {
744 // Strict mode must be preserved across IC patching.
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000745 ASSERT(Code::GetStrictMode(code->extra_ic_state()) == strict_mode());
ager@chromium.org9ee27ae2011-03-02 13:43:26 +0000746 IC::set_target(code);
747 }
748
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000749 // Stub accessors.
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000750 static Handle<Code> initialize_stub(Isolate* isolate,
751 StrictModeFlag strict_mode) {
752 if (strict_mode == kStrictMode) {
753 return isolate->builtins()->KeyedStoreIC_Initialize_Strict();
754 } else {
755 return isolate->builtins()->KeyedStoreIC_Initialize();
756 }
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000757 }
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000758
759 virtual Handle<Code> generic_stub() const {
760 if (strict_mode() == kStrictMode) {
761 return isolate()->builtins()->KeyedStoreIC_Generic_Strict();
762 } else {
763 return isolate()->builtins()->KeyedStoreIC_Generic();
764 }
ager@chromium.org9ee27ae2011-03-02 13:43:26 +0000765 }
jkummerow@chromium.orgfb7a7c42013-10-02 11:41:02 +0000766
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000767 Handle<Code> non_strict_arguments_stub() {
768 return isolate()->builtins()->KeyedStoreIC_NonStrictArguments();
whesse@chromium.org7b260152011-06-20 15:33:18 +0000769 }
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000770
jkummerow@chromium.org3d00d0a2013-09-04 13:57:32 +0000771 static void Clear(Isolate* isolate, Address address, Code* target);
ager@chromium.orgeadaf222009-06-16 09:43:10 +0000772
ulan@chromium.org750145a2013-03-07 15:14:13 +0000773 KeyedAccessStoreMode GetStoreMode(Handle<JSObject> receiver,
774 Handle<Object> key,
775 Handle<Object> value);
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000776
777 Handle<Map> ComputeTransitionedMap(Handle<JSObject> receiver,
ulan@chromium.org750145a2013-03-07 15:14:13 +0000778 KeyedAccessStoreMode store_mode);
yangguo@chromium.org003650e2013-01-24 16:31:08 +0000779
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000780 friend class IC;
781};
782
783
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000784// Type Recording BinaryOpIC, that records the types of the inputs and outputs.
danno@chromium.org40cb8782011-05-25 07:58:50 +0000785class BinaryOpIC: public IC {
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000786 public:
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000787 enum TypeInfo {
788 UNINITIALIZED,
789 SMI,
790 INT32,
hpayer@chromium.org8432c912013-02-28 15:55:26 +0000791 NUMBER,
lrn@chromium.org7516f052011-03-30 08:52:27 +0000792 ODDBALL,
yangguo@chromium.orgfb377212012-11-16 14:43:43 +0000793 STRING, // Only used for addition operation.
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000794 GENERIC
795 };
796
jkummerow@chromium.org25b0e212013-10-04 15:38:52 +0000797 explicit BinaryOpIC(Isolate* isolate) : IC(EXTRA_CALL_FRAME, isolate) { }
dslomov@chromium.orgb752d402013-06-18 11:54:54 +0000798
jkummerow@chromium.org25b0e212013-10-04 15:38:52 +0000799 static Builtins::JavaScript TokenToJSBuiltin(Token::Value op);
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000800
801 static const char* GetName(TypeInfo type_info);
802
jkummerow@chromium.org25b0e212013-10-04 15:38:52 +0000803 MUST_USE_RESULT MaybeObject* Transition(Handle<Object> left,
804 Handle<Object> right);
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000805};
806
807
808class CompareIC: public IC {
809 public:
hpayer@chromium.org8432c912013-02-28 15:55:26 +0000810 // The type/state lattice is defined by the following inequations:
811 // UNINITIALIZED < ...
812 // ... < GENERIC
813 // SMI < NUMBER
yangguo@chromium.org4a9f6552013-03-04 14:46:33 +0000814 // INTERNALIZED_STRING < STRING
hpayer@chromium.org8432c912013-02-28 15:55:26 +0000815 // KNOWN_OBJECT < OBJECT
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000816 enum State {
817 UNINITIALIZED,
yangguo@chromium.orgfb377212012-11-16 14:43:43 +0000818 SMI,
hpayer@chromium.org8432c912013-02-28 15:55:26 +0000819 NUMBER,
yangguo@chromium.orgfb377212012-11-16 14:43:43 +0000820 STRING,
yangguo@chromium.org4a9f6552013-03-04 14:46:33 +0000821 INTERNALIZED_STRING,
822 UNIQUE_NAME, // Symbol or InternalizedString
hpayer@chromium.org8432c912013-02-28 15:55:26 +0000823 OBJECT, // JSObject
824 KNOWN_OBJECT, // JSObject with specific map (faster check)
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000825 GENERIC
826 };
827
dslomov@chromium.orgb752d402013-06-18 11:54:54 +0000828 static State NewInputState(State old_state, Handle<Object> value);
829
830 static Handle<Type> StateToType(Isolate* isolate,
831 State state,
832 Handle<Map> map = Handle<Map>());
833
834 static void StubInfoToType(int stub_minor_key,
835 Handle<Type>* left_type,
836 Handle<Type>* right_type,
837 Handle<Type>* overall_type,
838 Handle<Map> map,
839 Isolate* isolate);
danno@chromium.org41728482013-06-12 22:31:22 +0000840
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +0000841 CompareIC(Isolate* isolate, Token::Value op)
842 : IC(EXTRA_CALL_FRAME, isolate), op_(op) { }
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000843
844 // Update the inline cache for the given operands.
machenbach@chromium.orgb5be0a92013-11-15 10:32:41 +0000845 Code* UpdateCaches(Handle<Object> x, Handle<Object> y);
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000846
yangguo@chromium.orgfb377212012-11-16 14:43:43 +0000847
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000848 // Factory method for getting an uninitialized compare stub.
hpayer@chromium.org8432c912013-02-28 15:55:26 +0000849 static Handle<Code> GetUninitialized(Isolate* isolate, Token::Value op);
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000850
851 // Helper function for computing the condition for a compare operation.
852 static Condition ComputeCondition(Token::Value op);
853
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000854 static const char* GetStateName(State state);
855
856 private:
yangguo@chromium.orgfb377212012-11-16 14:43:43 +0000857 static bool HasInlinedSmiCode(Address address);
858
859 State TargetState(State old_state,
860 State old_left,
861 State old_right,
862 bool has_inlined_smi_code,
863 Handle<Object> x,
864 Handle<Object> y);
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000865
866 bool strict() const { return op_ == Token::EQ_STRICT; }
867 Condition GetCondition() const { return ComputeCondition(op_); }
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000868
jkummerow@chromium.org3d00d0a2013-09-04 13:57:32 +0000869 static Code* GetRawUninitialized(Isolate* isolate, Token::Value op);
jkummerow@chromium.org212d9642012-05-11 15:02:09 +0000870
jkummerow@chromium.org3d00d0a2013-09-04 13:57:32 +0000871 static void Clear(Isolate* isolate, Address address, Code* target);
jkummerow@chromium.org212d9642012-05-11 15:02:09 +0000872
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000873 Token::Value op_;
jkummerow@chromium.org212d9642012-05-11 15:02:09 +0000874
875 friend class IC;
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000876};
877
ricow@chromium.org9fa09672011-07-25 11:05:35 +0000878
danno@chromium.orgca29dd82013-04-26 11:59:48 +0000879class CompareNilIC: public IC {
880 public:
881 explicit CompareNilIC(Isolate* isolate) : IC(EXTRA_CALL_FRAME, isolate) {}
882
883 MUST_USE_RESULT MaybeObject* CompareNil(Handle<Object> object);
884
885 static Handle<Code> GetUninitialized();
886
danno@chromium.orgca29dd82013-04-26 11:59:48 +0000887 static void Clear(Address address, Code* target);
888
ulan@chromium.org837a67e2013-06-11 15:39:48 +0000889 static MUST_USE_RESULT MaybeObject* DoCompareNilSlow(NilValue nil,
danno@chromium.orgca29dd82013-04-26 11:59:48 +0000890 Handle<Object> object);
891};
892
893
ricow@chromium.org9fa09672011-07-25 11:05:35 +0000894class ToBooleanIC: public IC {
895 public:
rossberg@chromium.orgb99c7542013-05-31 11:40:45 +0000896 explicit ToBooleanIC(Isolate* isolate) : IC(EXTRA_CALL_FRAME, isolate) { }
ricow@chromium.org9fa09672011-07-25 11:05:35 +0000897
machenbach@chromium.orgb5be0a92013-11-15 10:32:41 +0000898 MaybeObject* ToBoolean(Handle<Object> object);
ricow@chromium.org9fa09672011-07-25 11:05:35 +0000899};
900
901
danno@chromium.org40cb8782011-05-25 07:58:50 +0000902// Helper for BinaryOpIC and CompareIC.
jkummerow@chromium.org212d9642012-05-11 15:02:09 +0000903enum InlinedSmiCheck { ENABLE_INLINED_SMI_CHECK, DISABLE_INLINED_SMI_CHECK };
904void PatchInlinedSmiCode(Address address, InlinedSmiCheck check);
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000905
mstarzinger@chromium.orge3b8d0f2013-02-01 09:06:41 +0000906DECLARE_RUNTIME_FUNCTION(MaybeObject*, KeyedLoadIC_MissFromStubFailure);
jkummerow@chromium.org7bd87f02013-03-20 18:06:29 +0000907DECLARE_RUNTIME_FUNCTION(MaybeObject*, KeyedStoreIC_MissFromStubFailure);
mstarzinger@chromium.orge0e1b0d2013-07-08 08:38:06 +0000908DECLARE_RUNTIME_FUNCTION(MaybeObject*, UnaryOpIC_Miss);
909DECLARE_RUNTIME_FUNCTION(MaybeObject*, StoreIC_MissFromStubFailure);
machenbach@chromium.orgea468882013-11-18 08:53:19 +0000910DECLARE_RUNTIME_FUNCTION(MaybeObject*, KeyedCallIC_MissFromStubFailure);
jkummerow@chromium.orgba72ec82013-07-22 09:21:20 +0000911DECLARE_RUNTIME_FUNCTION(MaybeObject*, ElementsTransitionAndStoreIC_Miss);
jkummerow@chromium.org25b0e212013-10-04 15:38:52 +0000912DECLARE_RUNTIME_FUNCTION(MaybeObject*, BinaryOpIC_Miss);
danno@chromium.orgca29dd82013-04-26 11:59:48 +0000913DECLARE_RUNTIME_FUNCTION(MaybeObject*, CompareNilIC_Miss);
rossberg@chromium.orgb99c7542013-05-31 11:40:45 +0000914DECLARE_RUNTIME_FUNCTION(MaybeObject*, ToBooleanIC_Miss);
danno@chromium.orgca29dd82013-04-26 11:59:48 +0000915
jkummerow@chromium.org59297c72013-01-09 16:32:23 +0000916
christian.plesner.hansen43d26ec2008-07-03 15:10:15 +0000917} } // namespace v8::internal
918
919#endif // V8_IC_H_