2014-08-20 13:05:03 +00:00
|
|
|
// Copyright 2014 the V8 project authors. All rights reserved.
|
|
|
|
// Use of this source code is governed by a BSD-style license that can be
|
|
|
|
// found in the LICENSE file.
|
|
|
|
|
2015-02-20 10:11:45 +00:00
|
|
|
#include "src/compiler/js-inlining.h"
|
|
|
|
|
2014-10-21 12:16:37 +00:00
|
|
|
#include "src/ast.h"
|
|
|
|
#include "src/ast-numbering.h"
|
2015-02-20 10:11:45 +00:00
|
|
|
#include "src/compiler/all-nodes.h"
|
2014-08-20 13:05:03 +00:00
|
|
|
#include "src/compiler/ast-graph-builder.h"
|
|
|
|
#include "src/compiler/common-operator.h"
|
2015-03-10 08:37:16 +00:00
|
|
|
#include "src/compiler/js-context-specialization.h"
|
2014-08-20 13:05:03 +00:00
|
|
|
#include "src/compiler/js-operator.h"
|
|
|
|
#include "src/compiler/node-matchers.h"
|
2015-01-29 09:17:45 +00:00
|
|
|
#include "src/compiler/node-properties.h"
|
2015-03-09 08:37:01 +00:00
|
|
|
#include "src/compiler/operator-properties.h"
|
2014-09-18 08:56:52 +00:00
|
|
|
#include "src/full-codegen.h"
|
2014-08-20 13:05:03 +00:00
|
|
|
#include "src/parser.h"
|
|
|
|
#include "src/rewriter.h"
|
|
|
|
#include "src/scopes.h"
|
|
|
|
|
|
|
|
namespace v8 {
|
|
|
|
namespace internal {
|
|
|
|
namespace compiler {
|
|
|
|
|
2015-03-17 18:51:08 +00:00
|
|
|
#define TRACE(...) \
|
|
|
|
do { \
|
|
|
|
if (FLAG_trace_turbo_inlining) PrintF(__VA_ARGS__); \
|
|
|
|
} while (false)
|
|
|
|
|
2015-02-17 10:30:54 +00:00
|
|
|
|
|
|
|
// Provides convenience accessors for calls to JS functions.
|
|
|
|
class JSCallFunctionAccessor {
|
2014-08-20 13:05:03 +00:00
|
|
|
public:
|
2015-02-17 10:30:54 +00:00
|
|
|
explicit JSCallFunctionAccessor(Node* call) : call_(call) {
|
|
|
|
DCHECK_EQ(IrOpcode::kJSCallFunction, call->opcode());
|
|
|
|
}
|
2014-08-20 13:05:03 +00:00
|
|
|
|
2015-02-17 10:30:54 +00:00
|
|
|
Node* jsfunction() { return call_->InputAt(0); }
|
|
|
|
|
|
|
|
Node* receiver() { return call_->InputAt(1); }
|
|
|
|
|
|
|
|
Node* formal_argument(size_t index) {
|
|
|
|
DCHECK(index < formal_arguments());
|
|
|
|
return call_->InputAt(static_cast<int>(2 + index));
|
2014-08-20 13:05:03 +00:00
|
|
|
}
|
|
|
|
|
2015-02-17 10:30:54 +00:00
|
|
|
size_t formal_arguments() {
|
|
|
|
// {value_inputs} includes jsfunction and receiver.
|
|
|
|
size_t value_inputs = call_->op()->ValueInputCount();
|
|
|
|
DCHECK_GE(call_->InputCount(), 2);
|
|
|
|
return value_inputs - 2;
|
|
|
|
}
|
|
|
|
|
2015-03-09 08:37:01 +00:00
|
|
|
Node* frame_state() { return NodeProperties::GetFrameStateInput(call_, 0); }
|
2015-02-17 10:30:54 +00:00
|
|
|
|
2014-08-20 13:05:03 +00:00
|
|
|
private:
|
2015-02-17 10:30:54 +00:00
|
|
|
Node* call_;
|
2014-08-20 13:05:03 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2015-02-20 10:11:45 +00:00
|
|
|
class CopyVisitor {
|
2014-09-12 11:06:37 +00:00
|
|
|
public:
|
|
|
|
CopyVisitor(Graph* source_graph, Graph* target_graph, Zone* temp_zone)
|
2015-02-20 10:11:45 +00:00
|
|
|
: sentinel_op_(IrOpcode::kDead, Operator::kNoProperties, "Sentinel", 0, 0,
|
|
|
|
0, 0, 0, 0),
|
|
|
|
sentinel_(target_graph->NewNode(&sentinel_op_)),
|
|
|
|
copies_(source_graph->NodeCount(), sentinel_, temp_zone),
|
2014-09-12 11:06:37 +00:00
|
|
|
source_graph_(source_graph),
|
|
|
|
target_graph_(target_graph),
|
2015-02-20 10:11:45 +00:00
|
|
|
temp_zone_(temp_zone) {}
|
|
|
|
|
|
|
|
Node* GetCopy(Node* orig) { return copies_[orig->id()]; }
|
2014-09-12 11:06:37 +00:00
|
|
|
|
2015-02-20 10:11:45 +00:00
|
|
|
void CopyGraph() {
|
2014-09-12 11:06:37 +00:00
|
|
|
NodeVector inputs(temp_zone_);
|
2015-02-20 10:11:45 +00:00
|
|
|
// TODO(bmeurer): AllNodes should be turned into something like
|
|
|
|
// Graph::CollectNodesReachableFromEnd() and the gray set stuff should be
|
|
|
|
// removed since it's only needed by the visualizer.
|
|
|
|
AllNodes all(temp_zone_, source_graph_);
|
|
|
|
// Copy all nodes reachable from end.
|
|
|
|
for (Node* orig : all.live) {
|
|
|
|
Node* copy = GetCopy(orig);
|
|
|
|
if (copy != sentinel_) {
|
|
|
|
// Mapping already exists.
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
// Copy the node.
|
|
|
|
inputs.clear();
|
|
|
|
for (Node* input : orig->inputs()) inputs.push_back(copies_[input->id()]);
|
|
|
|
copy = target_graph_->NewNode(orig->op(), orig->InputCount(),
|
|
|
|
inputs.empty() ? nullptr : &inputs[0]);
|
|
|
|
copies_[orig->id()] = copy;
|
2014-09-12 11:06:37 +00:00
|
|
|
}
|
2015-02-20 10:11:45 +00:00
|
|
|
// For missing inputs.
|
|
|
|
for (Node* orig : all.live) {
|
|
|
|
Node* copy = copies_[orig->id()];
|
|
|
|
for (int i = 0; i < copy->InputCount(); ++i) {
|
|
|
|
Node* input = copy->InputAt(i);
|
|
|
|
if (input == sentinel_) {
|
|
|
|
copy->ReplaceInput(i, GetCopy(orig->InputAt(i)));
|
|
|
|
}
|
|
|
|
}
|
2014-09-12 11:06:37 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-02-20 10:11:45 +00:00
|
|
|
const NodeVector& copies() const { return copies_; }
|
2014-09-12 11:06:37 +00:00
|
|
|
|
|
|
|
private:
|
2015-02-20 10:11:45 +00:00
|
|
|
Operator const sentinel_op_;
|
|
|
|
Node* const sentinel_;
|
2014-09-12 11:06:37 +00:00
|
|
|
NodeVector copies_;
|
2015-02-20 10:11:45 +00:00
|
|
|
Graph* const source_graph_;
|
|
|
|
Graph* const target_graph_;
|
|
|
|
Zone* const temp_zone_;
|
2014-09-12 11:06:37 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2015-05-21 13:02:18 +00:00
|
|
|
Reduction JSInliner::InlineCall(Node* call, Node* start, Node* end) {
|
2014-08-28 12:18:25 +00:00
|
|
|
// The scheduler is smart enough to place our code; we just ensure {control}
|
2015-03-09 13:01:30 +00:00
|
|
|
// becomes the control input of the start of the inlinee, and {effect} becomes
|
|
|
|
// the effect input of the start of the inlinee.
|
2014-08-20 13:05:03 +00:00
|
|
|
Node* control = NodeProperties::GetControlInput(call);
|
2015-03-09 13:01:30 +00:00
|
|
|
Node* effect = NodeProperties::GetEffectInput(call);
|
2014-08-20 13:05:03 +00:00
|
|
|
|
|
|
|
// Context is last argument.
|
2015-05-21 13:02:18 +00:00
|
|
|
int const inlinee_context_index =
|
|
|
|
static_cast<int>(start->op()->ValueOutputCount()) - 1;
|
|
|
|
|
2014-08-20 13:05:03 +00:00
|
|
|
// {inliner_inputs} counts JSFunction, Receiver, arguments, but not
|
|
|
|
// context, effect, control.
|
2014-10-29 18:46:44 +00:00
|
|
|
int inliner_inputs = call->op()->ValueInputCount();
|
2014-08-20 13:05:03 +00:00
|
|
|
// Iterate over all uses of the start node.
|
2015-05-21 13:02:18 +00:00
|
|
|
for (Edge edge : start->use_edges()) {
|
2014-12-02 14:38:55 +00:00
|
|
|
Node* use = edge.from();
|
2014-08-20 13:05:03 +00:00
|
|
|
switch (use->opcode()) {
|
|
|
|
case IrOpcode::kParameter: {
|
2015-04-22 11:34:43 +00:00
|
|
|
int index = 1 + ParameterIndexOf(use->op());
|
2014-08-20 13:05:03 +00:00
|
|
|
if (index < inliner_inputs && index < inlinee_context_index) {
|
|
|
|
// There is an input from the call, and the index is a value
|
|
|
|
// projection but not the context, so rewire the input.
|
2015-05-12 12:41:41 +00:00
|
|
|
ReplaceWithValue(use, call->InputAt(index));
|
2014-08-20 13:05:03 +00:00
|
|
|
} else if (index == inlinee_context_index) {
|
2015-03-10 08:37:16 +00:00
|
|
|
// TODO(turbofan): We always context specialize inlinees currently, so
|
|
|
|
// we should never get here.
|
|
|
|
UNREACHABLE();
|
2014-08-20 13:05:03 +00:00
|
|
|
} else if (index < inlinee_context_index) {
|
|
|
|
// Call has fewer arguments than required, fill with undefined.
|
2015-05-12 12:41:41 +00:00
|
|
|
ReplaceWithValue(use, jsgraph_->UndefinedConstant());
|
2014-08-20 13:05:03 +00:00
|
|
|
} else {
|
|
|
|
// We got too many arguments, discard for now.
|
|
|
|
// TODO(sigurds): Fix to treat arguments array correctly.
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
default:
|
2014-12-02 14:38:55 +00:00
|
|
|
if (NodeProperties::IsEffectEdge(edge)) {
|
2015-03-09 13:01:30 +00:00
|
|
|
edge.UpdateTo(effect);
|
2014-12-02 14:38:55 +00:00
|
|
|
} else if (NodeProperties::IsControlEdge(edge)) {
|
|
|
|
edge.UpdateTo(control);
|
2014-08-20 13:05:03 +00:00
|
|
|
} else {
|
|
|
|
UNREACHABLE();
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-05-26 10:31:55 +00:00
|
|
|
NodeVector values(local_zone_);
|
|
|
|
NodeVector effects(local_zone_);
|
|
|
|
NodeVector controls(local_zone_);
|
|
|
|
for (Node* const input : end->inputs()) {
|
|
|
|
switch (input->opcode()) {
|
|
|
|
case IrOpcode::kReturn:
|
|
|
|
values.push_back(NodeProperties::GetValueInput(input, 0));
|
|
|
|
effects.push_back(NodeProperties::GetEffectInput(input));
|
|
|
|
controls.push_back(NodeProperties::GetControlInput(input));
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
// TODO(turbofan): Handle Throw, Terminate and Deoptimize here.
|
|
|
|
UNREACHABLE();
|
|
|
|
break;
|
2015-05-21 13:02:18 +00:00
|
|
|
}
|
|
|
|
}
|
2015-05-26 10:31:55 +00:00
|
|
|
DCHECK_NE(0u, values.size());
|
|
|
|
DCHECK_EQ(values.size(), effects.size());
|
|
|
|
DCHECK_EQ(values.size(), controls.size());
|
|
|
|
int const input_count = static_cast<int>(controls.size());
|
|
|
|
Node* control_output = jsgraph_->graph()->NewNode(
|
|
|
|
jsgraph_->common()->Merge(input_count), input_count, &controls.front());
|
|
|
|
values.push_back(control_output);
|
|
|
|
effects.push_back(control_output);
|
|
|
|
Node* value_output = jsgraph_->graph()->NewNode(
|
|
|
|
jsgraph_->common()->Phi(kMachAnyTagged, input_count),
|
|
|
|
static_cast<int>(values.size()), &values.front());
|
|
|
|
Node* effect_output = jsgraph_->graph()->NewNode(
|
|
|
|
jsgraph_->common()->EffectPhi(input_count),
|
|
|
|
static_cast<int>(effects.size()), &effects.front());
|
2015-05-21 13:02:18 +00:00
|
|
|
|
|
|
|
ReplaceWithValue(call, value_output, effect_output, control_output);
|
2015-03-03 13:09:49 +00:00
|
|
|
|
2015-05-21 13:02:18 +00:00
|
|
|
return Changed(value_output);
|
2014-08-20 13:05:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2014-09-18 08:56:52 +00:00
|
|
|
Node* JSInliner::CreateArgumentsAdaptorFrameState(JSCallFunctionAccessor* call,
|
|
|
|
Zone* temp_zone) {
|
2014-09-29 13:37:58 +00:00
|
|
|
const Operator* op = jsgraph_->common()->FrameState(
|
|
|
|
FrameStateType::ARGUMENTS_ADAPTOR, BailoutId(-1),
|
2015-05-15 12:17:15 +00:00
|
|
|
OutputFrameStateCombine::Ignore());
|
2014-09-18 08:56:52 +00:00
|
|
|
const Operator* op0 = jsgraph_->common()->StateValues(0);
|
|
|
|
Node* node0 = jsgraph_->graph()->NewNode(op0);
|
|
|
|
NodeVector params(temp_zone);
|
|
|
|
params.push_back(call->receiver());
|
|
|
|
for (size_t argument = 0; argument != call->formal_arguments(); ++argument) {
|
|
|
|
params.push_back(call->formal_argument(argument));
|
|
|
|
}
|
|
|
|
const Operator* op_param =
|
|
|
|
jsgraph_->common()->StateValues(static_cast<int>(params.size()));
|
|
|
|
Node* params_node = jsgraph_->graph()->NewNode(
|
|
|
|
op_param, static_cast<int>(params.size()), ¶ms.front());
|
|
|
|
return jsgraph_->graph()->NewNode(op, params_node, node0, node0,
|
|
|
|
jsgraph_->UndefinedConstant(),
|
2015-05-15 12:17:15 +00:00
|
|
|
call->jsfunction(), call->frame_state());
|
2014-09-18 08:56:52 +00:00
|
|
|
}
|
|
|
|
|
2014-08-20 13:05:03 +00:00
|
|
|
|
2015-02-20 10:11:45 +00:00
|
|
|
Reduction JSInliner::Reduce(Node* node) {
|
|
|
|
if (node->opcode() != IrOpcode::kJSCallFunction) return NoChange();
|
|
|
|
|
|
|
|
JSCallFunctionAccessor call(node);
|
|
|
|
HeapObjectMatcher<JSFunction> match(call.jsfunction());
|
|
|
|
if (!match.HasValue()) return NoChange();
|
|
|
|
|
|
|
|
Handle<JSFunction> function = match.Value().handle();
|
2015-03-09 08:05:24 +00:00
|
|
|
if (!function->IsJSFunction()) return NoChange();
|
2015-05-20 12:48:02 +00:00
|
|
|
if (mode_ == kRestrictedInlining && !function->shared()->force_inline()) {
|
2015-03-09 08:05:24 +00:00
|
|
|
return NoChange();
|
|
|
|
}
|
2015-02-20 10:11:45 +00:00
|
|
|
|
2015-03-24 14:17:05 +00:00
|
|
|
Zone zone;
|
|
|
|
ParseInfo parse_info(&zone, function);
|
|
|
|
CompilationInfo info(&parse_info);
|
2015-05-21 11:33:42 +00:00
|
|
|
if (info_->is_deoptimization_enabled()) info.MarkAsDeoptimizationEnabled();
|
2015-02-17 10:30:54 +00:00
|
|
|
|
2015-03-09 14:51:13 +00:00
|
|
|
if (!Compiler::ParseAndAnalyze(info.parse_info())) return NoChange();
|
2015-02-17 10:30:54 +00:00
|
|
|
if (!Compiler::EnsureDeoptimizationSupport(&info)) return NoChange();
|
2014-08-20 13:05:03 +00:00
|
|
|
|
2015-02-04 09:34:05 +00:00
|
|
|
if (info.scope()->arguments() != NULL && is_sloppy(info.language_mode())) {
|
2014-08-20 13:05:03 +00:00
|
|
|
// For now do not inline functions that use their arguments array.
|
2015-03-17 18:51:08 +00:00
|
|
|
TRACE("Not Inlining %s into %s because inlinee uses arguments array\n",
|
|
|
|
function->shared()->DebugName()->ToCString().get(),
|
|
|
|
info_->shared_info()->DebugName()->ToCString().get());
|
2015-02-17 10:30:54 +00:00
|
|
|
return NoChange();
|
2014-08-20 13:05:03 +00:00
|
|
|
}
|
|
|
|
|
2015-03-17 18:51:08 +00:00
|
|
|
TRACE("Inlining %s into %s\n",
|
|
|
|
function->shared()->DebugName()->ToCString().get(),
|
|
|
|
info_->shared_info()->DebugName()->ToCString().get());
|
2014-08-20 13:05:03 +00:00
|
|
|
|
2014-09-12 11:06:37 +00:00
|
|
|
Graph graph(info.zone());
|
2015-01-23 15:19:34 +00:00
|
|
|
JSGraph jsgraph(info.isolate(), &graph, jsgraph_->common(),
|
|
|
|
jsgraph_->javascript(), jsgraph_->machine());
|
2014-08-20 13:05:03 +00:00
|
|
|
|
2015-03-10 08:37:16 +00:00
|
|
|
// The inlinee specializes to the context from the JSFunction object.
|
|
|
|
// TODO(turbofan): We might want to load the context from the JSFunction at
|
|
|
|
// runtime in case we only know the SharedFunctionInfo once we have dynamic
|
|
|
|
// type feedback in the compiler.
|
2014-10-21 14:44:50 +00:00
|
|
|
AstGraphBuilder graph_builder(local_zone_, &info, &jsgraph);
|
2015-03-10 08:37:16 +00:00
|
|
|
graph_builder.CreateGraph(true, false);
|
|
|
|
JSContextSpecializer context_specializer(&jsgraph);
|
|
|
|
GraphReducer graph_reducer(&graph, local_zone_);
|
|
|
|
graph_reducer.AddReducer(&context_specializer);
|
|
|
|
graph_reducer.ReduceGraph();
|
2014-08-20 13:05:03 +00:00
|
|
|
|
2014-09-12 11:06:37 +00:00
|
|
|
CopyVisitor visitor(&graph, jsgraph_->graph(), info.zone());
|
|
|
|
visitor.CopyGraph();
|
2014-08-20 13:05:03 +00:00
|
|
|
|
2015-05-21 13:02:18 +00:00
|
|
|
Node* start = visitor.GetCopy(graph.start());
|
|
|
|
Node* end = visitor.GetCopy(graph.end());
|
2014-09-18 08:56:52 +00:00
|
|
|
|
2015-04-23 09:04:37 +00:00
|
|
|
Node* outer_frame_state = call.frame_state();
|
2015-05-21 13:02:18 +00:00
|
|
|
size_t const inlinee_formal_parameters = start->op()->ValueOutputCount() - 3;
|
2015-04-23 09:04:37 +00:00
|
|
|
// Insert argument adaptor frame if required.
|
2015-05-21 13:02:18 +00:00
|
|
|
if (call.formal_arguments() != inlinee_formal_parameters) {
|
2015-05-11 17:20:43 +00:00
|
|
|
// In strong mode, in case of too few arguments we need to throw a
|
|
|
|
// TypeError so we must not inline this call.
|
|
|
|
if (is_strong(info.language_mode()) &&
|
2015-05-21 13:02:18 +00:00
|
|
|
call.formal_arguments() < inlinee_formal_parameters) {
|
2015-05-11 17:20:43 +00:00
|
|
|
return NoChange();
|
|
|
|
}
|
2015-05-15 12:17:15 +00:00
|
|
|
outer_frame_state = CreateArgumentsAdaptorFrameState(&call, info.zone());
|
2015-04-23 09:04:37 +00:00
|
|
|
}
|
2014-09-18 08:56:52 +00:00
|
|
|
|
2015-05-19 09:11:30 +00:00
|
|
|
// Fix up all outer frame states from the inlinee.
|
|
|
|
for (Node* const node : visitor.copies()) {
|
|
|
|
if (node->opcode() == IrOpcode::kFrameState) {
|
2015-04-23 09:04:37 +00:00
|
|
|
DCHECK_EQ(1, OperatorProperties::GetFrameStateInputCount(node->op()));
|
2015-05-19 09:11:30 +00:00
|
|
|
// Don't touch this frame state, if it already has an "outer frame state".
|
|
|
|
if (NodeProperties::GetFrameStateInput(node, 0)->opcode() !=
|
|
|
|
IrOpcode::kFrameState) {
|
|
|
|
NodeProperties::ReplaceFrameStateInput(node, 0, outer_frame_state);
|
|
|
|
}
|
2014-09-18 08:56:52 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-05-21 13:02:18 +00:00
|
|
|
return InlineCall(node, start, end);
|
2014-08-20 13:05:03 +00:00
|
|
|
}
|
2014-10-20 07:56:50 +00:00
|
|
|
|
2015-01-26 09:05:47 +00:00
|
|
|
} // namespace compiler
|
|
|
|
} // namespace internal
|
|
|
|
} // namespace v8
|