| 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/compiler/js-inlining.h" | |
| 6 | |
| 7 #include "src/ast.h" | 5 #include "src/ast.h" |
| 8 #include "src/ast-numbering.h" | 6 #include "src/ast-numbering.h" |
| 9 #include "src/compiler/access-builder.h" | 7 #include "src/compiler/access-builder.h" |
| 10 #include "src/compiler/all-nodes.h" | |
| 11 #include "src/compiler/ast-graph-builder.h" | 8 #include "src/compiler/ast-graph-builder.h" |
| 12 #include "src/compiler/common-operator.h" | 9 #include "src/compiler/common-operator.h" |
| 10 #include "src/compiler/graph-inl.h" |
| 13 #include "src/compiler/graph-visualizer.h" | 11 #include "src/compiler/graph-visualizer.h" |
| 12 #include "src/compiler/js-inlining.h" |
| 14 #include "src/compiler/js-operator.h" | 13 #include "src/compiler/js-operator.h" |
| 15 #include "src/compiler/node-matchers.h" | 14 #include "src/compiler/node-matchers.h" |
| 16 #include "src/compiler/node-properties.h" | 15 #include "src/compiler/node-properties.h" |
| 17 #include "src/compiler/simplified-operator.h" | 16 #include "src/compiler/simplified-operator.h" |
| 18 #include "src/compiler/typer.h" | 17 #include "src/compiler/typer.h" |
| 19 #include "src/full-codegen.h" | 18 #include "src/full-codegen.h" |
| 20 #include "src/parser.h" | 19 #include "src/parser.h" |
| 21 #include "src/rewriter.h" | 20 #include "src/rewriter.h" |
| 22 #include "src/scopes.h" | 21 #include "src/scopes.h" |
| 23 | 22 |
| 23 |
| 24 namespace v8 { | 24 namespace v8 { |
| 25 namespace internal { | 25 namespace internal { |
| 26 namespace compiler { | 26 namespace compiler { |
| 27 | 27 |
| 28 | 28 |
| 29 // Provides convenience accessors for calls to JS functions. | 29 // Provides convenience accessors for calls to JS functions. |
| 30 class JSCallFunctionAccessor { | 30 class JSCallFunctionAccessor { |
| 31 public: | 31 public: |
| 32 explicit JSCallFunctionAccessor(Node* call) : call_(call) { | 32 explicit JSCallFunctionAccessor(Node* call) : call_(call) { |
| 33 DCHECK_EQ(IrOpcode::kJSCallFunction, call->opcode()); | 33 DCHECK_EQ(IrOpcode::kJSCallFunction, call->opcode()); |
| (...skipping 15 matching lines...) Expand all Loading... |
| 49 return value_inputs - 2; | 49 return value_inputs - 2; |
| 50 } | 50 } |
| 51 | 51 |
| 52 Node* frame_state() { return NodeProperties::GetFrameStateInput(call_); } | 52 Node* frame_state() { return NodeProperties::GetFrameStateInput(call_); } |
| 53 | 53 |
| 54 private: | 54 private: |
| 55 Node* call_; | 55 Node* call_; |
| 56 }; | 56 }; |
| 57 | 57 |
| 58 | 58 |
| 59 namespace { | 59 Reduction JSInliner::Reduce(Node* node) { |
| 60 if (node->opcode() != IrOpcode::kJSCallFunction) return NoChange(); |
| 61 |
| 62 JSCallFunctionAccessor call(node); |
| 63 HeapObjectMatcher<JSFunction> match(call.jsfunction()); |
| 64 if (!match.HasValue()) return NoChange(); |
| 65 |
| 66 Handle<JSFunction> jsfunction = match.Value().handle(); |
| 67 |
| 68 if (jsfunction->shared()->native()) { |
| 69 if (FLAG_trace_turbo_inlining) { |
| 70 SmartArrayPointer<char> name = |
| 71 jsfunction->shared()->DebugName()->ToCString(); |
| 72 PrintF("Not Inlining %s into %s because inlinee is native\n", name.get(), |
| 73 info_->shared_info()->DebugName()->ToCString().get()); |
| 74 } |
| 75 return NoChange(); |
| 76 } |
| 77 |
| 78 return TryInlineJSCall(node, jsfunction); |
| 79 } |
| 80 |
| 60 | 81 |
| 61 // A facade on a JSFunction's graph to facilitate inlining. It assumes the | 82 // A facade on a JSFunction's graph to facilitate inlining. It assumes the |
| 62 // that the function graph has only one return statement, and provides | 83 // that the function graph has only one return statement, and provides |
| 63 // {UnifyReturn} to convert a function graph to that end. | 84 // {UnifyReturn} to convert a function graph to that end. |
| 64 class Inlinee { | 85 class Inlinee { |
| 65 public: | 86 public: |
| 66 Inlinee(Node* start, Node* end) : start_(start), end_(end) {} | 87 Inlinee(Node* start, Node* end) : start_(start), end_(end) {} |
| 67 | 88 |
| 68 // Returns the last regular control node, that is | 89 // Returns the last regular control node, that is |
| 69 // the last control node before the end node. | 90 // the last control node before the end node. |
| (...skipping 76 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 146 Node* phi = | 167 Node* phi = |
| 147 graph->NewNode(op_phi, static_cast<int>(values.size()), &values.front()); | 168 graph->NewNode(op_phi, static_cast<int>(values.size()), &values.front()); |
| 148 Node* ephi = graph->NewNode(op_ephi, static_cast<int>(effects.size()), | 169 Node* ephi = graph->NewNode(op_ephi, static_cast<int>(effects.size()), |
| 149 &effects.front()); | 170 &effects.front()); |
| 150 Node* new_return = | 171 Node* new_return = |
| 151 graph->NewNode(jsgraph->common()->Return(), phi, ephi, final_merge); | 172 graph->NewNode(jsgraph->common()->Return(), phi, ephi, final_merge); |
| 152 graph->end()->ReplaceInput(0, new_return); | 173 graph->end()->ReplaceInput(0, new_return); |
| 153 } | 174 } |
| 154 | 175 |
| 155 | 176 |
| 156 class CopyVisitor { | 177 class CopyVisitor : public NullNodeVisitor { |
| 157 public: | 178 public: |
| 158 CopyVisitor(Graph* source_graph, Graph* target_graph, Zone* temp_zone) | 179 CopyVisitor(Graph* source_graph, Graph* target_graph, Zone* temp_zone) |
| 159 : sentinel_op_(IrOpcode::kDead, Operator::kNoProperties, "Sentinel", 0, 0, | 180 : copies_(source_graph->NodeCount(), NULL, temp_zone), |
| 160 0, 0, 0, 0), | 181 sentinels_(source_graph->NodeCount(), NULL, temp_zone), |
| 161 sentinel_(target_graph->NewNode(&sentinel_op_)), | |
| 162 copies_(source_graph->NodeCount(), sentinel_, temp_zone), | |
| 163 source_graph_(source_graph), | 182 source_graph_(source_graph), |
| 164 target_graph_(target_graph), | 183 target_graph_(target_graph), |
| 165 temp_zone_(temp_zone) {} | 184 temp_zone_(temp_zone), |
| 185 sentinel_op_(IrOpcode::kDead, Operator::kNoProperties, "sentinel", 0, 0, |
| 186 0, 0, 0, 0) {} |
| 166 | 187 |
| 167 Node* GetCopy(Node* orig) { return copies_[orig->id()]; } | 188 void Post(Node* original) { |
| 189 NodeVector inputs(temp_zone_); |
| 190 for (Node* const node : original->inputs()) { |
| 191 inputs.push_back(GetCopy(node)); |
| 192 } |
| 193 |
| 194 // Reuse the operator in the copy. This assumes that op lives in a zone |
| 195 // that lives longer than graph()'s zone. |
| 196 Node* copy = |
| 197 target_graph_->NewNode(original->op(), static_cast<int>(inputs.size()), |
| 198 (inputs.empty() ? NULL : &inputs.front())); |
| 199 copies_[original->id()] = copy; |
| 200 } |
| 201 |
| 202 Node* GetCopy(Node* original) { |
| 203 Node* copy = copies_[original->id()]; |
| 204 if (copy == NULL) { |
| 205 copy = GetSentinel(original); |
| 206 } |
| 207 DCHECK(copy); |
| 208 return copy; |
| 209 } |
| 168 | 210 |
| 169 void CopyGraph() { | 211 void CopyGraph() { |
| 170 NodeVector inputs(temp_zone_); | 212 source_graph_->VisitNodeInputsFromEnd(this); |
| 171 // TODO(bmeurer): AllNodes should be turned into something like | 213 ReplaceSentinels(); |
| 172 // Graph::CollectNodesReachableFromEnd() and the gray set stuff should be | 214 } |
| 173 // removed since it's only needed by the visualizer. | 215 |
| 174 AllNodes all(temp_zone_, source_graph_); | 216 const NodeVector& copies() { return copies_; } |
| 175 // Copy all nodes reachable from end. | 217 |
| 176 for (Node* orig : all.live) { | 218 private: |
| 177 Node* copy = GetCopy(orig); | 219 void ReplaceSentinels() { |
| 178 if (copy != sentinel_) { | 220 for (NodeId id = 0; id < source_graph_->NodeCount(); ++id) { |
| 179 // Mapping already exists. | 221 Node* sentinel = sentinels_[id]; |
| 180 continue; | 222 if (sentinel == NULL) continue; |
| 181 } | 223 Node* copy = copies_[id]; |
| 182 // Copy the node. | 224 DCHECK(copy); |
| 183 inputs.clear(); | 225 sentinel->ReplaceUses(copy); |
| 184 for (Node* input : orig->inputs()) inputs.push_back(copies_[input->id()]); | |
| 185 copy = target_graph_->NewNode(orig->op(), orig->InputCount(), &inputs[0]); | |
| 186 copies_[orig->id()] = copy; | |
| 187 } | |
| 188 // For missing inputs. | |
| 189 for (Node* orig : all.live) { | |
| 190 Node* copy = copies_[orig->id()]; | |
| 191 for (int i = 0; i < copy->InputCount(); ++i) { | |
| 192 Node* input = copy->InputAt(i); | |
| 193 if (input == sentinel_) { | |
| 194 copy->ReplaceInput(i, GetCopy(orig->InputAt(i))); | |
| 195 } | |
| 196 } | |
| 197 } | 226 } |
| 198 } | 227 } |
| 199 | 228 |
| 200 const NodeVector& copies() const { return copies_; } | 229 Node* GetSentinel(Node* original) { |
| 230 if (sentinels_[original->id()] == NULL) { |
| 231 sentinels_[original->id()] = target_graph_->NewNode(&sentinel_op_); |
| 232 } |
| 233 return sentinels_[original->id()]; |
| 234 } |
| 201 | 235 |
| 202 private: | |
| 203 Operator const sentinel_op_; | |
| 204 Node* const sentinel_; | |
| 205 NodeVector copies_; | 236 NodeVector copies_; |
| 206 Graph* const source_graph_; | 237 NodeVector sentinels_; |
| 207 Graph* const target_graph_; | 238 Graph* source_graph_; |
| 208 Zone* const temp_zone_; | 239 Graph* target_graph_; |
| 240 Zone* temp_zone_; |
| 241 Operator sentinel_op_; |
| 209 }; | 242 }; |
| 210 | 243 |
| 211 | 244 |
| 212 Reduction Inlinee::InlineAtCall(JSGraph* jsgraph, Node* call) { | 245 Reduction Inlinee::InlineAtCall(JSGraph* jsgraph, Node* call) { |
| 213 // The scheduler is smart enough to place our code; we just ensure {control} | 246 // The scheduler is smart enough to place our code; we just ensure {control} |
| 214 // becomes the control input of the start of the inlinee. | 247 // becomes the control input of the start of the inlinee. |
| 215 Node* control = NodeProperties::GetControlInput(call); | 248 Node* control = NodeProperties::GetControlInput(call); |
| 216 | 249 |
| 217 // The inlinee uses the context from the JSFunction object. This will | 250 // The inlinee uses the context from the JSFunction object. This will |
| 218 // also be the effect dependency for the inlinee as it produces an effect. | 251 // also be the effect dependency for the inlinee as it produces an effect. |
| (...skipping 40 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 259 UNREACHABLE(); | 292 UNREACHABLE(); |
| 260 } | 293 } |
| 261 break; | 294 break; |
| 262 } | 295 } |
| 263 } | 296 } |
| 264 | 297 |
| 265 NodeProperties::ReplaceWithValue(call, value_output(), effect_output()); | 298 NodeProperties::ReplaceWithValue(call, value_output(), effect_output()); |
| 266 return Reducer::Replace(value_output()); | 299 return Reducer::Replace(value_output()); |
| 267 } | 300 } |
| 268 | 301 |
| 269 } // namespace | |
| 270 | |
| 271 | 302 |
| 272 void JSInliner::AddClosureToFrameState(Node* frame_state, | 303 void JSInliner::AddClosureToFrameState(Node* frame_state, |
| 273 Handle<JSFunction> jsfunction) { | 304 Handle<JSFunction> jsfunction) { |
| 274 FrameStateCallInfo call_info = OpParameter<FrameStateCallInfo>(frame_state); | 305 FrameStateCallInfo call_info = OpParameter<FrameStateCallInfo>(frame_state); |
| 275 const Operator* op = jsgraph_->common()->FrameState( | 306 const Operator* op = jsgraph_->common()->FrameState( |
| 276 FrameStateType::JS_FRAME, call_info.bailout_id(), | 307 FrameStateType::JS_FRAME, call_info.bailout_id(), |
| 277 call_info.state_combine(), jsfunction); | 308 call_info.state_combine(), jsfunction); |
| 278 frame_state->set_op(op); | 309 frame_state->set_op(op); |
| 279 } | 310 } |
| 280 | 311 |
| (...skipping 14 matching lines...) Expand all Loading... |
| 295 const Operator* op_param = | 326 const Operator* op_param = |
| 296 jsgraph_->common()->StateValues(static_cast<int>(params.size())); | 327 jsgraph_->common()->StateValues(static_cast<int>(params.size())); |
| 297 Node* params_node = jsgraph_->graph()->NewNode( | 328 Node* params_node = jsgraph_->graph()->NewNode( |
| 298 op_param, static_cast<int>(params.size()), ¶ms.front()); | 329 op_param, static_cast<int>(params.size()), ¶ms.front()); |
| 299 return jsgraph_->graph()->NewNode(op, params_node, node0, node0, | 330 return jsgraph_->graph()->NewNode(op, params_node, node0, node0, |
| 300 jsgraph_->UndefinedConstant(), | 331 jsgraph_->UndefinedConstant(), |
| 301 call->frame_state()); | 332 call->frame_state()); |
| 302 } | 333 } |
| 303 | 334 |
| 304 | 335 |
| 305 Reduction JSInliner::Reduce(Node* node) { | 336 Reduction JSInliner::TryInlineJSCall(Node* call_node, |
| 306 if (node->opcode() != IrOpcode::kJSCallFunction) return NoChange(); | 337 Handle<JSFunction> function) { |
| 307 | 338 JSCallFunctionAccessor call(call_node); |
| 308 JSCallFunctionAccessor call(node); | |
| 309 HeapObjectMatcher<JSFunction> match(call.jsfunction()); | |
| 310 if (!match.HasValue()) return NoChange(); | |
| 311 | |
| 312 Handle<JSFunction> function = match.Value().handle(); | |
| 313 | |
| 314 if (function->shared()->native()) { | |
| 315 if (FLAG_trace_turbo_inlining) { | |
| 316 SmartArrayPointer<char> name = | |
| 317 function->shared()->DebugName()->ToCString(); | |
| 318 PrintF("Not Inlining %s into %s because inlinee is native\n", name.get(), | |
| 319 info_->shared_info()->DebugName()->ToCString().get()); | |
| 320 } | |
| 321 return NoChange(); | |
| 322 } | |
| 323 | |
| 324 CompilationInfoWithZone info(function); | 339 CompilationInfoWithZone info(function); |
| 325 | 340 |
| 326 if (!Compiler::ParseAndAnalyze(&info)) return NoChange(); | 341 if (!Compiler::ParseAndAnalyze(&info)) return NoChange(); |
| 327 if (!Compiler::EnsureDeoptimizationSupport(&info)) return NoChange(); | 342 if (!Compiler::EnsureDeoptimizationSupport(&info)) return NoChange(); |
| 328 | 343 |
| 329 if (info.scope()->arguments() != NULL && is_sloppy(info.language_mode())) { | 344 if (info.scope()->arguments() != NULL && is_sloppy(info.language_mode())) { |
| 330 // For now do not inline functions that use their arguments array. | 345 // For now do not inline functions that use their arguments array. |
| 331 SmartArrayPointer<char> name = function->shared()->DebugName()->ToCString(); | 346 SmartArrayPointer<char> name = function->shared()->DebugName()->ToCString(); |
| 332 if (FLAG_trace_turbo_inlining) { | 347 if (FLAG_trace_turbo_inlining) { |
| 333 PrintF( | 348 PrintF( |
| (...skipping 32 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 366 } | 381 } |
| 367 | 382 |
| 368 for (Node* node : visitor.copies()) { | 383 for (Node* node : visitor.copies()) { |
| 369 if (node && node->opcode() == IrOpcode::kFrameState) { | 384 if (node && node->opcode() == IrOpcode::kFrameState) { |
| 370 AddClosureToFrameState(node, function); | 385 AddClosureToFrameState(node, function); |
| 371 NodeProperties::ReplaceFrameStateInput(node, outer_frame_state); | 386 NodeProperties::ReplaceFrameStateInput(node, outer_frame_state); |
| 372 } | 387 } |
| 373 } | 388 } |
| 374 } | 389 } |
| 375 | 390 |
| 376 return inlinee.InlineAtCall(jsgraph_, node); | 391 return inlinee.InlineAtCall(jsgraph_, call_node); |
| 377 } | 392 } |
| 378 | 393 |
| 379 } // namespace compiler | 394 } // namespace compiler |
| 380 } // namespace internal | 395 } // namespace internal |
| 381 } // namespace v8 | 396 } // namespace v8 |
| OLD | NEW |