2015-01-12 11:39:48 +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.
|
|
|
|
|
|
|
|
#include "src/compiler.h"
|
2015-02-11 13:26:35 +00:00
|
|
|
#include "src/compiler/all-nodes.h"
|
2015-01-12 11:39:48 +00:00
|
|
|
#include "src/compiler/common-operator.h"
|
|
|
|
#include "src/compiler/control-reducer.h"
|
|
|
|
#include "src/compiler/frame.h"
|
|
|
|
#include "src/compiler/graph.h"
|
2015-02-11 13:26:35 +00:00
|
|
|
#include "src/compiler/graph-visualizer.h"
|
2015-01-12 11:39:48 +00:00
|
|
|
#include "src/compiler/js-graph.h"
|
2015-01-29 09:46:24 +00:00
|
|
|
#include "src/compiler/loop-analysis.h"
|
2015-01-12 11:39:48 +00:00
|
|
|
#include "src/compiler/node.h"
|
|
|
|
#include "src/compiler/node-marker.h"
|
|
|
|
#include "src/compiler/osr.h"
|
|
|
|
#include "src/scopes.h"
|
|
|
|
|
|
|
|
namespace v8 {
|
|
|
|
namespace internal {
|
|
|
|
namespace compiler {
|
|
|
|
|
|
|
|
OsrHelper::OsrHelper(CompilationInfo* info)
|
|
|
|
: parameter_count_(info->scope()->num_parameters()),
|
2015-02-02 10:19:52 +00:00
|
|
|
stack_slot_count_(info->scope()->num_stack_slots() +
|
|
|
|
info->osr_expr_stack_height()) {}
|
2015-01-12 11:39:48 +00:00
|
|
|
|
|
|
|
|
2015-02-11 13:26:35 +00:00
|
|
|
// Peel outer loops and rewire the graph so that control reduction can
|
|
|
|
// produce a properly formed graph.
|
|
|
|
static void PeelOuterLoopsForOsr(Graph* graph, CommonOperatorBuilder* common,
|
|
|
|
Zone* tmp_zone, Node* dead,
|
|
|
|
LoopTree* loop_tree, LoopTree::Loop* osr_loop,
|
|
|
|
Node* osr_normal_entry, Node* osr_loop_entry) {
|
|
|
|
const int original_count = graph->NodeCount();
|
|
|
|
AllNodes all(tmp_zone, graph);
|
|
|
|
NodeVector tmp_inputs(tmp_zone);
|
|
|
|
Node* sentinel = graph->NewNode(dead->op());
|
|
|
|
|
|
|
|
// Make a copy of the graph for each outer loop.
|
|
|
|
ZoneVector<NodeVector*> copies(tmp_zone);
|
|
|
|
for (LoopTree::Loop* loop = osr_loop->parent(); loop; loop = loop->parent()) {
|
|
|
|
void* stuff = tmp_zone->New(sizeof(NodeVector));
|
|
|
|
NodeVector* mapping =
|
|
|
|
new (stuff) NodeVector(original_count, sentinel, tmp_zone);
|
|
|
|
copies.push_back(mapping);
|
|
|
|
|
|
|
|
// Prepare the mapping for OSR values and the OSR loop entry.
|
|
|
|
mapping->at(osr_normal_entry->id()) = dead;
|
|
|
|
mapping->at(osr_loop_entry->id()) = dead;
|
|
|
|
|
|
|
|
// The outer loops are dead in this copy.
|
|
|
|
for (LoopTree::Loop* outer = loop->parent(); outer;
|
|
|
|
outer = outer->parent()) {
|
|
|
|
for (Node* node : loop_tree->HeaderNodes(outer)) {
|
|
|
|
mapping->at(node->id()) = dead;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Copy all nodes.
|
|
|
|
for (size_t i = 0; i < all.live.size(); i++) {
|
|
|
|
Node* orig = all.live[i];
|
|
|
|
Node* copy = mapping->at(orig->id());
|
|
|
|
if (copy != sentinel) {
|
|
|
|
// Mapping already exists.
|
|
|
|
continue;
|
|
|
|
}
|
2015-02-17 18:10:02 +00:00
|
|
|
if (orig->InputCount() == 0 || orig->opcode() == IrOpcode::kParameter ||
|
|
|
|
orig->opcode() == IrOpcode::kOsrValue) {
|
|
|
|
// No need to copy leaf nodes or parameters.
|
2015-02-11 13:26:35 +00:00
|
|
|
mapping->at(orig->id()) = orig;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Copy the node.
|
|
|
|
tmp_inputs.clear();
|
|
|
|
for (Node* input : orig->inputs()) {
|
|
|
|
tmp_inputs.push_back(mapping->at(input->id()));
|
|
|
|
}
|
|
|
|
copy = graph->NewNode(orig->op(), orig->InputCount(), &tmp_inputs[0]);
|
|
|
|
if (NodeProperties::IsTyped(orig)) {
|
|
|
|
NodeProperties::SetBounds(copy, NodeProperties::GetBounds(orig));
|
|
|
|
}
|
|
|
|
mapping->at(orig->id()) = copy;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Fix missing inputs.
|
|
|
|
for (size_t i = 0; i < all.live.size(); i++) {
|
|
|
|
Node* orig = all.live[i];
|
|
|
|
Node* copy = mapping->at(orig->id());
|
|
|
|
for (int j = 0; j < copy->InputCount(); j++) {
|
|
|
|
Node* input = copy->InputAt(j);
|
|
|
|
if (input == sentinel)
|
|
|
|
copy->ReplaceInput(j, mapping->at(orig->InputAt(j)->id()));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Construct the transfer from the previous graph copies to the new copy.
|
|
|
|
Node* loop_header = loop_tree->HeaderNode(loop);
|
|
|
|
NodeVector* previous =
|
|
|
|
copies.size() > 1 ? copies[copies.size() - 2] : nullptr;
|
|
|
|
const int backedges = loop_header->op()->ControlInputCount() - 1;
|
|
|
|
if (backedges == 1) {
|
|
|
|
// Simple case. Map the incoming edges to the loop to the previous copy.
|
|
|
|
for (Node* node : loop_tree->HeaderNodes(loop)) {
|
2015-02-19 11:36:38 +00:00
|
|
|
if (!all.IsLive(node)) continue; // dead phi hanging off loop.
|
2015-02-11 13:26:35 +00:00
|
|
|
Node* copy = mapping->at(node->id());
|
|
|
|
Node* backedge = node->InputAt(1);
|
|
|
|
if (previous) backedge = previous->at(backedge->id());
|
|
|
|
copy->ReplaceInput(0, backedge);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// Complex case. Multiple backedges. Introduce a merge for incoming edges.
|
|
|
|
tmp_inputs.clear();
|
|
|
|
for (int i = 0; i < backedges; i++) {
|
|
|
|
Node* backedge = loop_header->InputAt(i + 1);
|
|
|
|
if (previous) backedge = previous->at(backedge->id());
|
|
|
|
tmp_inputs.push_back(backedge);
|
|
|
|
}
|
|
|
|
Node* merge =
|
|
|
|
graph->NewNode(common->Merge(backedges), backedges, &tmp_inputs[0]);
|
|
|
|
for (Node* node : loop_tree->HeaderNodes(loop)) {
|
2015-02-19 11:36:38 +00:00
|
|
|
if (!all.IsLive(node)) continue; // dead phi hanging off loop.
|
2015-02-11 13:26:35 +00:00
|
|
|
Node* copy = mapping->at(node->id());
|
|
|
|
if (node == loop_header) {
|
|
|
|
// The entry to the loop is the merge.
|
|
|
|
copy->ReplaceInput(0, merge);
|
|
|
|
} else {
|
|
|
|
// Merge inputs to the phi at the loop entry.
|
|
|
|
tmp_inputs.clear();
|
|
|
|
for (int i = 0; i < backedges; i++) {
|
|
|
|
Node* backedge = node->InputAt(i + 1);
|
|
|
|
if (previous) backedge = previous->at(backedge->id());
|
|
|
|
tmp_inputs.push_back(backedge);
|
|
|
|
}
|
|
|
|
tmp_inputs.push_back(merge);
|
|
|
|
Node* phi =
|
|
|
|
graph->NewNode(common->ResizeMergeOrPhi(node->op(), backedges),
|
|
|
|
backedges + 1, &tmp_inputs[0]);
|
|
|
|
copy->ReplaceInput(0, phi);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Kill the outer loops in the original graph.
|
|
|
|
for (LoopTree::Loop* outer = osr_loop->parent(); outer;
|
|
|
|
outer = outer->parent()) {
|
|
|
|
loop_tree->HeaderNode(outer)->ReplaceUses(dead);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Merge the ends of the graph copies.
|
|
|
|
Node* end = graph->end();
|
|
|
|
tmp_inputs.clear();
|
|
|
|
for (int i = -1; i < static_cast<int>(copies.size()); i++) {
|
|
|
|
Node* input = end->InputAt(0);
|
|
|
|
if (i >= 0) input = copies[i]->at(input->id());
|
|
|
|
if (input->opcode() == IrOpcode::kMerge) {
|
|
|
|
for (Node* node : input->inputs()) tmp_inputs.push_back(node);
|
|
|
|
} else {
|
|
|
|
tmp_inputs.push_back(input);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
int count = static_cast<int>(tmp_inputs.size());
|
|
|
|
Node* merge = graph->NewNode(common->Merge(count), count, &tmp_inputs[0]);
|
|
|
|
end->ReplaceInput(0, merge);
|
|
|
|
|
|
|
|
if (FLAG_trace_turbo_graph) { // Simple textual RPO.
|
|
|
|
OFStream os(stdout);
|
|
|
|
os << "-- Graph after OSR duplication -- " << std::endl;
|
|
|
|
os << AsRPO(*graph);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-02-19 12:38:44 +00:00
|
|
|
static void TransferOsrValueTypesFromLoopPhis(Zone* zone, Node* osr_loop_entry,
|
|
|
|
Node* osr_loop) {
|
|
|
|
// Find the index of the osr loop entry into the loop.
|
|
|
|
int index = 0;
|
|
|
|
for (index = 0; index < osr_loop->InputCount(); index++) {
|
|
|
|
if (osr_loop->InputAt(index) == osr_loop_entry) break;
|
|
|
|
}
|
|
|
|
if (index == osr_loop->InputCount()) return;
|
|
|
|
|
|
|
|
for (Node* osr_value : osr_loop_entry->uses()) {
|
|
|
|
if (osr_value->opcode() != IrOpcode::kOsrValue) continue;
|
|
|
|
bool unknown = true;
|
|
|
|
for (Node* phi : osr_value->uses()) {
|
|
|
|
if (phi->opcode() != IrOpcode::kPhi) continue;
|
|
|
|
if (NodeProperties::GetControlInput(phi) != osr_loop) continue;
|
|
|
|
if (phi->InputAt(index) != osr_value) continue;
|
|
|
|
if (NodeProperties::IsTyped(phi)) {
|
|
|
|
// Transfer the type from the phi to the OSR value itself.
|
|
|
|
Bounds phi_bounds = NodeProperties::GetBounds(phi);
|
|
|
|
if (unknown) {
|
|
|
|
NodeProperties::SetBounds(osr_value, phi_bounds);
|
|
|
|
} else {
|
|
|
|
Bounds osr_bounds = NodeProperties::GetBounds(osr_value);
|
|
|
|
NodeProperties::SetBounds(osr_value,
|
|
|
|
Bounds::Both(phi_bounds, osr_bounds, zone));
|
|
|
|
}
|
|
|
|
unknown = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (unknown) NodeProperties::SetBounds(osr_value, Bounds::Unbounded(zone));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-01-29 09:46:24 +00:00
|
|
|
bool OsrHelper::Deconstruct(JSGraph* jsgraph, CommonOperatorBuilder* common,
|
2015-01-12 11:39:48 +00:00
|
|
|
Zone* tmp_zone) {
|
|
|
|
Graph* graph = jsgraph->graph();
|
2015-01-29 09:46:24 +00:00
|
|
|
Node* osr_normal_entry = nullptr;
|
|
|
|
Node* osr_loop_entry = nullptr;
|
|
|
|
Node* osr_loop = nullptr;
|
|
|
|
|
|
|
|
for (Node* node : graph->start()->uses()) {
|
|
|
|
if (node->opcode() == IrOpcode::kOsrLoopEntry) {
|
|
|
|
osr_loop_entry = node; // found the OSR loop entry
|
|
|
|
} else if (node->opcode() == IrOpcode::kOsrNormalEntry) {
|
|
|
|
osr_normal_entry = node;
|
2015-01-12 11:39:48 +00:00
|
|
|
}
|
2015-01-29 09:46:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (osr_loop_entry == nullptr) {
|
|
|
|
// No OSR entry found, do nothing.
|
2015-01-30 09:29:25 +00:00
|
|
|
CHECK(osr_normal_entry);
|
2015-01-29 09:46:24 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (Node* use : osr_loop_entry->uses()) {
|
|
|
|
if (use->opcode() == IrOpcode::kLoop) {
|
2015-01-30 09:29:25 +00:00
|
|
|
CHECK(!osr_loop); // should be only one OSR loop.
|
2015-01-29 09:46:24 +00:00
|
|
|
osr_loop = use; // found the OSR loop.
|
2015-01-12 11:39:48 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-30 09:29:25 +00:00
|
|
|
CHECK(osr_loop); // Should have found the OSR loop.
|
2015-01-29 09:46:24 +00:00
|
|
|
|
2015-02-19 12:38:44 +00:00
|
|
|
// Transfer the types from loop phis to the OSR values which flow into them.
|
|
|
|
TransferOsrValueTypesFromLoopPhis(graph->zone(), osr_loop_entry, osr_loop);
|
|
|
|
|
2015-01-29 09:46:24 +00:00
|
|
|
// Analyze the graph to determine how deeply nested the OSR loop is.
|
|
|
|
LoopTree* loop_tree = LoopFinder::BuildLoopTree(graph, tmp_zone);
|
|
|
|
|
2015-02-16 18:16:52 +00:00
|
|
|
Node* dead = jsgraph->DeadControl();
|
2015-01-29 09:46:24 +00:00
|
|
|
LoopTree::Loop* loop = loop_tree->ContainingLoop(osr_loop);
|
2015-02-11 13:26:35 +00:00
|
|
|
if (loop->depth() > 0) {
|
|
|
|
PeelOuterLoopsForOsr(graph, common, tmp_zone, dead, loop_tree, loop,
|
|
|
|
osr_normal_entry, osr_loop_entry);
|
|
|
|
}
|
2015-01-29 09:46:24 +00:00
|
|
|
|
|
|
|
// Replace the normal entry with {Dead} and the loop entry with {Start}
|
|
|
|
// and run the control reducer to clean up the graph.
|
2015-02-11 13:26:35 +00:00
|
|
|
osr_normal_entry->ReplaceUses(dead);
|
2015-02-19 11:36:38 +00:00
|
|
|
osr_normal_entry->Kill();
|
2015-01-29 09:46:24 +00:00
|
|
|
osr_loop_entry->ReplaceUses(graph->start());
|
2015-02-19 11:36:38 +00:00
|
|
|
osr_loop_entry->Kill();
|
2015-02-17 11:52:32 +00:00
|
|
|
|
|
|
|
// Normally the control reducer removes loops whose first input is dead,
|
|
|
|
// but we need to avoid that because the osr_loop is reachable through
|
|
|
|
// the second input, so reduce it and its phis manually.
|
|
|
|
osr_loop->ReplaceInput(0, dead);
|
|
|
|
Node* node = ControlReducer::ReduceMerge(jsgraph, common, osr_loop);
|
|
|
|
if (node != osr_loop) osr_loop->ReplaceUses(node);
|
|
|
|
|
|
|
|
// Run the normal control reduction, which naturally trims away the dead
|
|
|
|
// parts of the graph.
|
2015-01-12 11:39:48 +00:00
|
|
|
ControlReducer::ReduceGraph(tmp_zone, jsgraph, common);
|
2015-01-29 09:46:24 +00:00
|
|
|
|
|
|
|
return true;
|
2015-01-12 11:39:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void OsrHelper::SetupFrame(Frame* frame) {
|
|
|
|
// The optimized frame will subsume the unoptimized frame. Do so by reserving
|
|
|
|
// the first spill slots.
|
|
|
|
frame->ReserveSpillSlots(UnoptimizedFrameSlots());
|
2015-02-02 10:19:52 +00:00
|
|
|
// The frame needs to be adjusted by the number of unoptimized frame slots.
|
|
|
|
frame->SetOsrStackSlotCount(static_cast<int>(UnoptimizedFrameSlots()));
|
2015-01-12 11:39:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
} // namespace compiler
|
|
|
|
} // namespace internal
|
|
|
|
} // namespace v8
|