Index: src/ic/arm/handler-compiler-arm.cc |
diff --git a/src/ic/arm/ic-compiler-arm.cc b/src/ic/arm/handler-compiler-arm.cc |
similarity index 89% |
copy from src/ic/arm/ic-compiler-arm.cc |
copy to src/ic/arm/handler-compiler-arm.cc |
index c5b0903b9e76fa7a9f1e4803d4465f50f1ee6f10..bdb701cf179494d9a14f3b5bcc1bbc4c2adca76b 100644 |
--- a/src/ic/arm/ic-compiler-arm.cc |
+++ b/src/ic/arm/handler-compiler-arm.cc |
@@ -7,7 +7,7 @@ |
#if V8_TARGET_ARCH_ARM |
#include "src/ic/call-optimization.h" |
-#include "src/ic/ic-compiler.h" |
+#include "src/ic/handler-compiler.h" |
namespace v8 { |
namespace internal { |
@@ -15,6 +15,82 @@ namespace internal { |
#define __ ACCESS_MASM(masm) |
+void NamedLoadHandlerCompiler::GenerateLoadViaGetter( |
+ MacroAssembler* masm, Handle<HeapType> type, Register receiver, |
+ Handle<JSFunction> getter) { |
+ // ----------- S t a t e ------------- |
+ // -- r0 : receiver |
+ // -- r2 : name |
+ // -- lr : return address |
+ // ----------------------------------- |
+ { |
+ FrameAndConstantPoolScope scope(masm, StackFrame::INTERNAL); |
+ |
+ if (!getter.is_null()) { |
+ // Call the JavaScript getter with the receiver on the stack. |
+ if (IC::TypeToMap(*type, masm->isolate())->IsJSGlobalObjectMap()) { |
+ // Swap in the global receiver. |
+ __ ldr(receiver, |
+ FieldMemOperand(receiver, JSGlobalObject::kGlobalProxyOffset)); |
+ } |
+ __ push(receiver); |
+ ParameterCount actual(0); |
+ ParameterCount expected(getter); |
+ __ InvokeFunction(getter, expected, actual, CALL_FUNCTION, |
+ NullCallWrapper()); |
+ } else { |
+ // If we generate a global code snippet for deoptimization only, remember |
+ // the place to continue after deoptimization. |
+ masm->isolate()->heap()->SetGetterStubDeoptPCOffset(masm->pc_offset()); |
+ } |
+ |
+ // Restore context register. |
+ __ ldr(cp, MemOperand(fp, StandardFrameConstants::kContextOffset)); |
+ } |
+ __ Ret(); |
+} |
+ |
+ |
+void NamedStoreHandlerCompiler::GenerateStoreViaSetter( |
+ MacroAssembler* masm, Handle<HeapType> type, Register receiver, |
+ Handle<JSFunction> setter) { |
+ // ----------- S t a t e ------------- |
+ // -- lr : return address |
+ // ----------------------------------- |
+ { |
+ FrameAndConstantPoolScope scope(masm, StackFrame::INTERNAL); |
+ |
+ // Save value register, so we can restore it later. |
+ __ push(value()); |
+ |
+ if (!setter.is_null()) { |
+ // Call the JavaScript setter with receiver and value on the stack. |
+ if (IC::TypeToMap(*type, masm->isolate())->IsJSGlobalObjectMap()) { |
+ // Swap in the global receiver. |
+ __ ldr(receiver, |
+ FieldMemOperand(receiver, JSGlobalObject::kGlobalProxyOffset)); |
+ } |
+ __ Push(receiver, value()); |
+ ParameterCount actual(1); |
+ ParameterCount expected(setter); |
+ __ InvokeFunction(setter, expected, actual, CALL_FUNCTION, |
+ NullCallWrapper()); |
+ } else { |
+ // If we generate a global code snippet for deoptimization only, remember |
+ // the place to continue after deoptimization. |
+ masm->isolate()->heap()->SetSetterStubDeoptPCOffset(masm->pc_offset()); |
+ } |
+ |
+ // We have to return the passed value, not the return value of the setter. |
+ __ pop(r0); |
+ |
+ // Restore context register. |
+ __ ldr(cp, MemOperand(fp, StandardFrameConstants::kContextOffset)); |
+ } |
+ __ Ret(); |
+} |
+ |
+ |
void PropertyHandlerCompiler::GenerateDictionaryNegativeLookup( |
MacroAssembler* masm, Label* miss_label, Register receiver, |
Handle<Name> name, Register scratch0, Register scratch1) { |
@@ -210,6 +286,34 @@ void PropertyHandlerCompiler::GenerateFastApiCall( |
} |
+void ElementHandlerCompiler::GenerateLoadDictionaryElement( |
+ MacroAssembler* masm) { |
+ // The return address is in lr. |
+ Label slow, miss; |
+ |
+ Register key = LoadIC::NameRegister(); |
+ Register receiver = LoadIC::ReceiverRegister(); |
+ DCHECK(receiver.is(r1)); |
+ DCHECK(key.is(r2)); |
+ |
+ __ UntagAndJumpIfNotSmi(r6, key, &miss); |
+ __ ldr(r4, FieldMemOperand(receiver, JSObject::kElementsOffset)); |
+ __ LoadFromNumberDictionary(&slow, r4, key, r0, r6, r3, r5); |
+ __ Ret(); |
+ |
+ __ bind(&slow); |
+ __ IncrementCounter( |
+ masm->isolate()->counters()->keyed_load_external_array_slow(), 1, r2, r3); |
+ |
+ TailCallBuiltin(masm, Builtins::kKeyedLoadIC_Slow); |
+ |
+ // Miss case, call the runtime. |
+ __ bind(&miss); |
+ |
+ TailCallBuiltin(masm, Builtins::kKeyedLoadIC_Miss); |
+} |
+ |
+ |
#undef __ |
#define __ ACCESS_MASM(masm()) |
@@ -684,54 +788,6 @@ Handle<Code> NamedStoreHandlerCompiler::CompileStoreCallback( |
} |
-#undef __ |
-#define __ ACCESS_MASM(masm) |
- |
- |
-void NamedStoreHandlerCompiler::GenerateStoreViaSetter( |
- MacroAssembler* masm, Handle<HeapType> type, Register receiver, |
- Handle<JSFunction> setter) { |
- // ----------- S t a t e ------------- |
- // -- lr : return address |
- // ----------------------------------- |
- { |
- FrameAndConstantPoolScope scope(masm, StackFrame::INTERNAL); |
- |
- // Save value register, so we can restore it later. |
- __ push(value()); |
- |
- if (!setter.is_null()) { |
- // Call the JavaScript setter with receiver and value on the stack. |
- if (IC::TypeToMap(*type, masm->isolate())->IsJSGlobalObjectMap()) { |
- // Swap in the global receiver. |
- __ ldr(receiver, |
- FieldMemOperand(receiver, JSGlobalObject::kGlobalProxyOffset)); |
- } |
- __ Push(receiver, value()); |
- ParameterCount actual(1); |
- ParameterCount expected(setter); |
- __ InvokeFunction(setter, expected, actual, CALL_FUNCTION, |
- NullCallWrapper()); |
- } else { |
- // If we generate a global code snippet for deoptimization only, remember |
- // the place to continue after deoptimization. |
- masm->isolate()->heap()->SetSetterStubDeoptPCOffset(masm->pc_offset()); |
- } |
- |
- // We have to return the passed value, not the return value of the setter. |
- __ pop(r0); |
- |
- // Restore context register. |
- __ ldr(cp, MemOperand(fp, StandardFrameConstants::kContextOffset)); |
- } |
- __ Ret(); |
-} |
- |
- |
-#undef __ |
-#define __ ACCESS_MASM(masm()) |
- |
- |
Handle<Code> NamedStoreHandlerCompiler::CompileStoreInterceptor( |
Handle<Name> name) { |
__ Push(receiver(), this->name(), value()); |
@@ -749,50 +805,6 @@ Handle<Code> NamedStoreHandlerCompiler::CompileStoreInterceptor( |
Register NamedStoreHandlerCompiler::value() { return StoreIC::ValueRegister(); } |
-#undef __ |
-#define __ ACCESS_MASM(masm) |
- |
- |
-void NamedLoadHandlerCompiler::GenerateLoadViaGetter( |
- MacroAssembler* masm, Handle<HeapType> type, Register receiver, |
- Handle<JSFunction> getter) { |
- // ----------- S t a t e ------------- |
- // -- r0 : receiver |
- // -- r2 : name |
- // -- lr : return address |
- // ----------------------------------- |
- { |
- FrameAndConstantPoolScope scope(masm, StackFrame::INTERNAL); |
- |
- if (!getter.is_null()) { |
- // Call the JavaScript getter with the receiver on the stack. |
- if (IC::TypeToMap(*type, masm->isolate())->IsJSGlobalObjectMap()) { |
- // Swap in the global receiver. |
- __ ldr(receiver, |
- FieldMemOperand(receiver, JSGlobalObject::kGlobalProxyOffset)); |
- } |
- __ push(receiver); |
- ParameterCount actual(0); |
- ParameterCount expected(getter); |
- __ InvokeFunction(getter, expected, actual, CALL_FUNCTION, |
- NullCallWrapper()); |
- } else { |
- // If we generate a global code snippet for deoptimization only, remember |
- // the place to continue after deoptimization. |
- masm->isolate()->heap()->SetGetterStubDeoptPCOffset(masm->pc_offset()); |
- } |
- |
- // Restore context register. |
- __ ldr(cp, MemOperand(fp, StandardFrameConstants::kContextOffset)); |
- } |
- __ Ret(); |
-} |
- |
- |
-#undef __ |
-#define __ ACCESS_MASM(masm()) |
- |
- |
Handle<Code> NamedLoadHandlerCompiler::CompileLoadGlobal( |
Handle<PropertyCell> cell, Handle<Name> name, bool is_configurable) { |
Label miss; |
@@ -821,138 +833,6 @@ Handle<Code> NamedLoadHandlerCompiler::CompileLoadGlobal( |
} |
-Handle<Code> PropertyICCompiler::CompilePolymorphic(TypeHandleList* types, |
- CodeHandleList* handlers, |
- Handle<Name> name, |
- Code::StubType type, |
- IcCheckType check) { |
- Label miss; |
- |
- if (check == PROPERTY && |
- (kind() == Code::KEYED_LOAD_IC || kind() == Code::KEYED_STORE_IC)) { |
- // In case we are compiling an IC for dictionary loads and stores, just |
- // check whether the name is unique. |
- if (name.is_identical_to(isolate()->factory()->normal_ic_symbol())) { |
- __ JumpIfNotUniqueName(this->name(), &miss); |
- } else { |
- __ cmp(this->name(), Operand(name)); |
- __ b(ne, &miss); |
- } |
- } |
- |
- Label number_case; |
- Label* smi_target = IncludesNumberType(types) ? &number_case : &miss; |
- __ JumpIfSmi(receiver(), smi_target); |
- |
- // Polymorphic keyed stores may use the map register |
- Register map_reg = scratch1(); |
- DCHECK(kind() != Code::KEYED_STORE_IC || |
- map_reg.is(KeyedStoreIC::MapRegister())); |
- |
- int receiver_count = types->length(); |
- int number_of_handled_maps = 0; |
- __ ldr(map_reg, FieldMemOperand(receiver(), HeapObject::kMapOffset)); |
- for (int current = 0; current < receiver_count; ++current) { |
- Handle<HeapType> type = types->at(current); |
- Handle<Map> map = IC::TypeToMap(*type, isolate()); |
- if (!map->is_deprecated()) { |
- number_of_handled_maps++; |
- __ mov(ip, Operand(map)); |
- __ cmp(map_reg, ip); |
- if (type->Is(HeapType::Number())) { |
- DCHECK(!number_case.is_unused()); |
- __ bind(&number_case); |
- } |
- __ Jump(handlers->at(current), RelocInfo::CODE_TARGET, eq); |
- } |
- } |
- DCHECK(number_of_handled_maps != 0); |
- |
- __ bind(&miss); |
- TailCallBuiltin(masm(), MissBuiltin(kind())); |
- |
- // Return the generated code. |
- InlineCacheState state = |
- number_of_handled_maps > 1 ? POLYMORPHIC : MONOMORPHIC; |
- return GetCode(kind(), type, name, state); |
-} |
- |
- |
-Handle<Code> PropertyICCompiler::CompileKeyedStorePolymorphic( |
- MapHandleList* receiver_maps, CodeHandleList* handler_stubs, |
- MapHandleList* transitioned_maps) { |
- Label miss; |
- __ JumpIfSmi(receiver(), &miss); |
- |
- int receiver_count = receiver_maps->length(); |
- __ ldr(scratch1(), FieldMemOperand(receiver(), HeapObject::kMapOffset)); |
- for (int i = 0; i < receiver_count; ++i) { |
- __ mov(ip, Operand(receiver_maps->at(i))); |
- __ cmp(scratch1(), ip); |
- if (transitioned_maps->at(i).is_null()) { |
- __ Jump(handler_stubs->at(i), RelocInfo::CODE_TARGET, eq); |
- } else { |
- Label next_map; |
- __ b(ne, &next_map); |
- __ mov(transition_map(), Operand(transitioned_maps->at(i))); |
- __ Jump(handler_stubs->at(i), RelocInfo::CODE_TARGET, al); |
- __ bind(&next_map); |
- } |
- } |
- |
- __ bind(&miss); |
- TailCallBuiltin(masm(), MissBuiltin(kind())); |
- |
- // Return the generated code. |
- return GetCode(kind(), Code::NORMAL, factory()->empty_string(), POLYMORPHIC); |
-} |
- |
- |
-#undef __ |
-#define __ ACCESS_MASM(masm) |
- |
- |
-void ElementHandlerCompiler::GenerateLoadDictionaryElement( |
- MacroAssembler* masm) { |
- // The return address is in lr. |
- Label slow, miss; |
- |
- Register key = LoadIC::NameRegister(); |
- Register receiver = LoadIC::ReceiverRegister(); |
- DCHECK(receiver.is(r1)); |
- DCHECK(key.is(r2)); |
- |
- __ UntagAndJumpIfNotSmi(r6, key, &miss); |
- __ ldr(r4, FieldMemOperand(receiver, JSObject::kElementsOffset)); |
- __ LoadFromNumberDictionary(&slow, r4, key, r0, r6, r3, r5); |
- __ Ret(); |
- |
- __ bind(&slow); |
- __ IncrementCounter( |
- masm->isolate()->counters()->keyed_load_external_array_slow(), 1, r2, r3); |
- |
- TailCallBuiltin(masm, Builtins::kKeyedLoadIC_Slow); |
- |
- // Miss case, call the runtime. |
- __ bind(&miss); |
- |
- TailCallBuiltin(masm, Builtins::kKeyedLoadIC_Miss); |
-} |
- |
- |
-void PropertyICCompiler::GenerateRuntimeSetProperty(MacroAssembler* masm, |
- StrictMode strict_mode) { |
- __ Push(StoreIC::ReceiverRegister(), StoreIC::NameRegister(), |
- StoreIC::ValueRegister()); |
- |
- __ mov(r0, Operand(Smi::FromInt(strict_mode))); |
- __ Push(r0); |
- |
- // Do tail-call to runtime routine. |
- __ TailCallRuntime(Runtime::kSetProperty, 4, 1); |
-} |
- |
- |
#undef __ |
} |
} // namespace v8::internal |