| OLD | NEW |
| 1 // Copyright 2014 the V8 project authors. All rights reserved. | 1 // Copyright 2014 the V8 project authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 | 4 |
| 5 #include "src/v8.h" | 5 #include "src/v8.h" |
| 6 | 6 |
| 7 #if V8_TARGET_ARCH_X64 | 7 #if V8_TARGET_ARCH_X64 |
| 8 | 8 |
| 9 #include "src/ic/call-optimization.h" | 9 #include "src/ic/call-optimization.h" |
| 10 #include "src/ic/handler-compiler.h" | 10 #include "src/ic/handler-compiler.h" |
| (...skipping 112 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 123 Handle<JSObject> holder_obj, IC::UtilityId id) { | 123 Handle<JSObject> holder_obj, IC::UtilityId id) { |
| 124 PushInterceptorArguments(masm, receiver, holder, name, holder_obj); | 124 PushInterceptorArguments(masm, receiver, holder, name, holder_obj); |
| 125 __ CallExternalReference(ExternalReference(IC_Utility(id), masm->isolate()), | 125 __ CallExternalReference(ExternalReference(IC_Utility(id), masm->isolate()), |
| 126 NamedLoadHandlerCompiler::kInterceptorArgsLength); | 126 NamedLoadHandlerCompiler::kInterceptorArgsLength); |
| 127 } | 127 } |
| 128 | 128 |
| 129 | 129 |
| 130 // Generate call to api function. | 130 // Generate call to api function. |
| 131 void PropertyHandlerCompiler::GenerateApiAccessorCall( | 131 void PropertyHandlerCompiler::GenerateApiAccessorCall( |
| 132 MacroAssembler* masm, const CallOptimization& optimization, | 132 MacroAssembler* masm, const CallOptimization& optimization, |
| 133 Handle<Map> receiver_map, Register receiver, Register scratch_in, | 133 Handle<Map> receiver_map, Register receiver, Register scratch, |
| 134 bool is_store, Register store_parameter, Register accessor_holder, | 134 bool is_store, Register store_parameter, Register accessor_holder, |
| 135 int accessor_index) { | 135 int accessor_index) { |
| 136 DCHECK(!accessor_holder.is(scratch_in)); | 136 DCHECK(!accessor_holder.is(scratch)); |
| 137 DCHECK(optimization.is_simple_api_call()); | 137 DCHECK(optimization.is_simple_api_call()); |
| 138 | 138 |
| 139 __ PopReturnAddressTo(scratch_in); | 139 __ PopReturnAddressTo(scratch); |
| 140 // receiver | 140 // receiver |
| 141 __ Push(receiver); | 141 __ Push(receiver); |
| 142 // Write the arguments to stack frame. | 142 // Write the arguments to stack frame. |
| 143 if (is_store) { | 143 if (is_store) { |
| 144 DCHECK(!receiver.is(store_parameter)); | 144 DCHECK(!receiver.is(store_parameter)); |
| 145 DCHECK(!scratch_in.is(store_parameter)); | 145 DCHECK(!scratch.is(store_parameter)); |
| 146 __ Push(store_parameter); | 146 __ Push(store_parameter); |
| 147 } | 147 } |
| 148 __ PushReturnAddressFrom(scratch_in); | 148 __ PushReturnAddressFrom(scratch); |
| 149 // Stack now matches JSFunction abi. | 149 // Stack now matches JSFunction abi. |
| 150 | 150 |
| 151 // Abi for CallApiFunctionStub. | 151 // Abi for CallApiFunctionStub. |
| 152 Register callee = rdi; | 152 Register callee = rdi; |
| 153 Register call_data = rbx; | 153 Register data = rbx; |
| 154 Register holder = rcx; | 154 Register holder = rcx; |
| 155 Register api_function_address = rdx; | 155 Register api_function_address = rdx; |
| 156 Register scratch = rax; // scratch_in is no longer valid. | 156 scratch = no_reg; |
| 157 | 157 |
| 158 // Put callee in place. | 158 // Put callee in place. |
| 159 __ LoadAccessor(callee, accessor_holder, accessor_index, | 159 __ LoadAccessor(callee, accessor_holder, accessor_index, |
| 160 is_store ? ACCESSOR_SETTER : ACCESSOR_GETTER); | 160 is_store ? ACCESSOR_SETTER : ACCESSOR_GETTER); |
| 161 | 161 |
| 162 // Put holder in place. | 162 // Put holder in place. |
| 163 CallOptimization::HolderLookup holder_lookup; | 163 CallOptimization::HolderLookup holder_lookup; |
| 164 Handle<JSObject> api_holder = | 164 Handle<JSObject> api_holder = |
| 165 optimization.LookupHolderOfExpectedType(receiver_map, &holder_lookup); | 165 optimization.LookupHolderOfExpectedType(receiver_map, &holder_lookup); |
| 166 switch (holder_lookup) { | 166 switch (holder_lookup) { |
| 167 case CallOptimization::kHolderIsReceiver: | 167 case CallOptimization::kHolderIsReceiver: |
| 168 __ Move(holder, receiver); | 168 __ Move(holder, receiver); |
| 169 break; | 169 break; |
| 170 case CallOptimization::kHolderFound: | 170 case CallOptimization::kHolderFound: |
| 171 __ Move(holder, api_holder); | 171 __ Move(holder, api_holder); |
| 172 break; | 172 break; |
| 173 case CallOptimization::kHolderNotFound: | 173 case CallOptimization::kHolderNotFound: |
| 174 UNREACHABLE(); | 174 UNREACHABLE(); |
| 175 break; | 175 break; |
| 176 } | 176 } |
| 177 | 177 |
| 178 Isolate* isolate = masm->isolate(); | 178 Isolate* isolate = masm->isolate(); |
| 179 Handle<CallHandlerInfo> api_call_info = optimization.api_call_info(); | 179 Handle<CallHandlerInfo> api_call_info = optimization.api_call_info(); |
| 180 Handle<Object> call_data_obj(api_call_info->data(), isolate); | |
| 181 | |
| 182 bool call_data_undefined = false; | 180 bool call_data_undefined = false; |
| 183 // Put call_data in place. | 181 // Put call data in place. |
| 184 if (isolate->heap()->InNewSpace(*call_data_obj)) { | 182 if (api_call_info->data()->IsUndefined()) { |
| 185 __ Move(scratch, api_call_info); | |
| 186 __ movp(call_data, FieldOperand(scratch, CallHandlerInfo::kDataOffset)); | |
| 187 } else if (call_data_obj->IsUndefined()) { | |
| 188 call_data_undefined = true; | 183 call_data_undefined = true; |
| 189 __ LoadRoot(call_data, Heap::kUndefinedValueRootIndex); | 184 __ LoadRoot(data, Heap::kUndefinedValueRootIndex); |
| 190 } else { | 185 } else { |
| 191 __ Move(call_data, call_data_obj); | 186 __ movp(data, FieldOperand(callee, JSFunction::kSharedFunctionInfoOffset)); |
| 187 __ movp(data, FieldOperand(data, SharedFunctionInfo::kFunctionDataOffset)); |
| 188 __ movp(data, FieldOperand(data, FunctionTemplateInfo::kCallCodeOffset)); |
| 189 __ movp(data, FieldOperand(data, CallHandlerInfo::kDataOffset)); |
| 192 } | 190 } |
| 193 | 191 |
| 194 // Put api_function_address in place. | 192 // Put api_function_address in place. |
| 195 Address function_address = v8::ToCData<Address>(api_call_info->callback()); | 193 Address function_address = v8::ToCData<Address>(api_call_info->callback()); |
| 196 __ Move(api_function_address, function_address, | 194 __ Move(api_function_address, function_address, |
| 197 RelocInfo::EXTERNAL_REFERENCE); | 195 RelocInfo::EXTERNAL_REFERENCE); |
| 198 | 196 |
| 199 // Jump to stub. | 197 // Jump to stub. |
| 200 CallApiAccessorStub stub(isolate, is_store, call_data_undefined); | 198 CallApiAccessorStub stub(isolate, is_store, call_data_undefined); |
| 201 __ TailCallStub(&stub); | 199 __ TailCallStub(&stub); |
| (...skipping 552 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 754 // Return the generated code. | 752 // Return the generated code. |
| 755 return GetCode(kind(), Code::NORMAL, name); | 753 return GetCode(kind(), Code::NORMAL, name); |
| 756 } | 754 } |
| 757 | 755 |
| 758 | 756 |
| 759 #undef __ | 757 #undef __ |
| 760 } | 758 } |
| 761 } // namespace v8::internal | 759 } // namespace v8::internal |
| 762 | 760 |
| 763 #endif // V8_TARGET_ARCH_X64 | 761 #endif // V8_TARGET_ARCH_X64 |
| OLD | NEW |