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/code-stubs.h" | 5 #include "src/code-stubs.h" |
6 #include "src/compiler.h" | 6 #include "src/compiler.h" |
7 #include "src/compiler/common-operator.h" | 7 #include "src/compiler/common-operator.h" |
8 #include "src/compiler/frame.h" | 8 #include "src/compiler/frame.h" |
9 #include "src/compiler/linkage.h" | 9 #include "src/compiler/linkage.h" |
10 #include "src/compiler/node.h" | 10 #include "src/compiler/node.h" |
(...skipping 71 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
82 bool CallDescriptor::HasSameReturnLocationsAs( | 82 bool CallDescriptor::HasSameReturnLocationsAs( |
83 const CallDescriptor* other) const { | 83 const CallDescriptor* other) const { |
84 if (ReturnCount() != other->ReturnCount()) return false; | 84 if (ReturnCount() != other->ReturnCount()) return false; |
85 for (size_t i = 0; i < ReturnCount(); ++i) { | 85 for (size_t i = 0; i < ReturnCount(); ++i) { |
86 if (GetReturnLocation(i) != other->GetReturnLocation(i)) return false; | 86 if (GetReturnLocation(i) != other->GetReturnLocation(i)) return false; |
87 } | 87 } |
88 return true; | 88 return true; |
89 } | 89 } |
90 | 90 |
91 | 91 |
92 bool CallDescriptor::CanTailCall(const Node* node) const { | 92 bool CallDescriptor::CanTailCall(const Node* node, |
93 // Determine the number of stack parameters passed in | 93 int* stack_param_delta) const { |
94 size_t stack_params = 0; | 94 // TODO(danno): TF only current supports tail calls where the number of stack |
95 for (size_t i = 0; i < InputCount(); ++i) { | 95 // parameters of the callee is the same or fewer of the caller. |
96 if (!GetInputLocation(i).IsRegister()) { | |
97 ++stack_params; | |
98 } | |
99 } | |
100 // Ensure the input linkage contains the stack parameters in the right order | |
101 size_t current_stack_param = 0; | |
102 for (size_t i = 0; i < InputCount(); ++i) { | |
103 if (!GetInputLocation(i).IsRegister()) { | |
104 if (GetInputLocation(i) != LinkageLocation::ForCallerFrameSlot( | |
105 static_cast<int>(current_stack_param) - | |
106 static_cast<int>(stack_params))) { | |
107 return false; | |
108 } | |
109 ++current_stack_param; | |
110 } | |
111 } | |
112 // Tail calling is currently allowed if return locations match and all | |
113 // parameters are either in registers or on the stack but match exactly in | |
114 // number and content. | |
115 CallDescriptor const* other = OpParameter<CallDescriptor const*>(node); | 96 CallDescriptor const* other = OpParameter<CallDescriptor const*>(node); |
116 if (!HasSameReturnLocationsAs(other)) return false; | 97 if (!HasSameReturnLocationsAs(other)) return false; |
117 size_t current_input = 0; | 98 size_t current_input = 0; |
118 size_t other_input = 0; | 99 size_t other_input = 0; |
119 while (true) { | 100 *stack_param_delta = 0; |
120 if (other_input >= other->InputCount()) { | 101 bool more_other = true; |
121 while (current_input < InputCount()) { | 102 bool more_this = true; |
122 if (!GetInputLocation(current_input).IsRegister()) { | 103 while (more_other || more_this) { |
123 return false; | 104 if (other_input < other->InputCount()) { |
124 } | 105 if (!other->GetInputLocation(other_input).IsRegister()) { |
125 ++current_input; | 106 (*stack_param_delta)++; |
126 } | 107 } |
127 return true; | 108 } else { |
| 109 more_other = false; |
128 } | 110 } |
129 if (current_input >= InputCount()) { | 111 if (current_input < InputCount()) { |
130 while (other_input < other->InputCount()) { | 112 if (!GetInputLocation(current_input).IsRegister()) { |
131 if (!other->GetInputLocation(other_input).IsRegister()) { | 113 (*stack_param_delta)--; |
132 return false; | |
133 } | |
134 ++other_input; | |
135 } | 114 } |
136 return true; | 115 } else { |
137 } | 116 more_this = false; |
138 if (GetInputLocation(current_input).IsRegister()) { | |
139 ++current_input; | |
140 continue; | |
141 } | |
142 if (other->GetInputLocation(other_input).IsRegister()) { | |
143 ++other_input; | |
144 continue; | |
145 } | |
146 if (GetInputLocation(current_input) != | |
147 other->GetInputLocation(other_input)) { | |
148 return false; | |
149 } | |
150 Node* input = node->InputAt(static_cast<int>(other_input)); | |
151 if (input->opcode() != IrOpcode::kParameter) { | |
152 return false; | |
153 } | |
154 // Make sure that the parameter input passed through to the tail call | |
155 // corresponds to the correct stack slot. | |
156 size_t param_index = ParameterIndexOf(input->op()); | |
157 if (param_index != current_input - 1) { | |
158 return false; | |
159 } | 117 } |
160 ++current_input; | 118 ++current_input; |
161 ++other_input; | 119 ++other_input; |
162 } | 120 } |
163 UNREACHABLE(); | 121 return *stack_param_delta <= 0; |
164 return false; | |
165 } | 122 } |
166 | 123 |
167 | 124 |
168 CallDescriptor* Linkage::ComputeIncoming(Zone* zone, CompilationInfo* info) { | 125 CallDescriptor* Linkage::ComputeIncoming(Zone* zone, CompilationInfo* info) { |
169 if (info->code_stub() != NULL) { | 126 if (info->code_stub() != NULL) { |
170 // Use the code stub interface descriptor. | 127 // Use the code stub interface descriptor. |
171 CodeStub* stub = info->code_stub(); | 128 CodeStub* stub = info->code_stub(); |
172 CallInterfaceDescriptor descriptor = stub->GetCallInterfaceDescriptor(); | 129 CallInterfaceDescriptor descriptor = stub->GetCallInterfaceDescriptor(); |
173 return GetStubCallDescriptor( | 130 return GetStubCallDescriptor( |
174 info->isolate(), zone, descriptor, stub->GetStackParameterCount(), | 131 info->isolate(), zone, descriptor, stub->GetStackParameterCount(), |
(...skipping 76 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
251 case Runtime::kInlineToLength: | 208 case Runtime::kInlineToLength: |
252 case Runtime::kInlineToName: | 209 case Runtime::kInlineToName: |
253 case Runtime::kInlineToNumber: | 210 case Runtime::kInlineToNumber: |
254 case Runtime::kInlineToObject: | 211 case Runtime::kInlineToObject: |
255 case Runtime::kInlineToPrimitive_Number: | 212 case Runtime::kInlineToPrimitive_Number: |
256 case Runtime::kInlineToPrimitive_String: | 213 case Runtime::kInlineToPrimitive_String: |
257 case Runtime::kInlineToPrimitive: | 214 case Runtime::kInlineToPrimitive: |
258 case Runtime::kInlineToString: | 215 case Runtime::kInlineToString: |
259 return 1; | 216 return 1; |
260 case Runtime::kInlineCall: | 217 case Runtime::kInlineCall: |
| 218 case Runtime::kInlineTailCall: |
261 case Runtime::kInlineDeoptimizeNow: | 219 case Runtime::kInlineDeoptimizeNow: |
262 case Runtime::kInlineThrowNotDateError: | 220 case Runtime::kInlineThrowNotDateError: |
263 return 2; | 221 return 2; |
264 default: | 222 default: |
265 break; | 223 break; |
266 } | 224 } |
267 | 225 |
268 // Most inlined runtime functions (except the ones listed above) can be called | 226 // Most inlined runtime functions (except the ones listed above) can be called |
269 // without a FrameState or will be lowered by JSIntrinsicLowering internally. | 227 // without a FrameState or will be lowered by JSIntrinsicLowering internally. |
270 const Runtime::Function* const f = Runtime::FunctionForId(function); | 228 const Runtime::Function* const f = Runtime::FunctionForId(function); |
(...skipping 291 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
562 return LinkageLocation::ForCalleeFrameSlot(spill_index); | 520 return LinkageLocation::ForCalleeFrameSlot(spill_index); |
563 } else { | 521 } else { |
564 // Parameter. Use the assigned location from the incoming call descriptor. | 522 // Parameter. Use the assigned location from the incoming call descriptor. |
565 int parameter_index = 1 + index; // skip index 0, which is the target. | 523 int parameter_index = 1 + index; // skip index 0, which is the target. |
566 return incoming_->GetInputLocation(parameter_index); | 524 return incoming_->GetInputLocation(parameter_index); |
567 } | 525 } |
568 } | 526 } |
569 } // namespace compiler | 527 } // namespace compiler |
570 } // namespace internal | 528 } // namespace internal |
571 } // namespace v8 | 529 } // namespace v8 |
OLD | NEW |