Commit 9b859a88 authored by sigurds's avatar sigurds Committed by Commit bot

[turbofan] Add TRACE macro for escape analysis

R=jarin@chromium.org
BUG=v8:4586
LOG=n

Review URL: https://codereview.chromium.org/1608073003

Cr-Commit-Position: refs/heads/master@{#33468}
parent 867fc715
...@@ -11,6 +11,15 @@ namespace v8 { ...@@ -11,6 +11,15 @@ namespace v8 {
namespace internal { namespace internal {
namespace compiler { namespace compiler {
#ifdef DEBUG
#define TRACE(...) \
do { \
if (FLAG_trace_turbo_escape) PrintF(__VA_ARGS__); \
} while (false)
#else
#define TRACE(...)
#endif // DEBUG
EscapeAnalysisReducer::EscapeAnalysisReducer(Editor* editor, JSGraph* jsgraph, EscapeAnalysisReducer::EscapeAnalysisReducer(Editor* editor, JSGraph* jsgraph,
EscapeAnalysis* escape_analysis, EscapeAnalysis* escape_analysis,
Zone* zone) Zone* zone)
...@@ -89,10 +98,8 @@ Reduction EscapeAnalysisReducer::ReduceLoad(Node* node) { ...@@ -89,10 +98,8 @@ Reduction EscapeAnalysisReducer::ReduceLoad(Node* node) {
if (Node* rep = escape_analysis()->GetReplacement(node)) { if (Node* rep = escape_analysis()->GetReplacement(node)) {
visited_.Add(node->id()); visited_.Add(node->id());
counters()->turbo_escape_loads_replaced()->Increment(); counters()->turbo_escape_loads_replaced()->Increment();
if (FLAG_trace_turbo_escape) { TRACE("Replaced #%d (%s) with #%d (%s)\n", node->id(),
PrintF("Replaced #%d (%s) with #%d (%s)\n", node->id(), node->op()->mnemonic(), rep->id(), rep->op()->mnemonic());
node->op()->mnemonic(), rep->id(), rep->op()->mnemonic());
}
ReplaceWithValue(node, rep); ReplaceWithValue(node, rep);
return Changed(rep); return Changed(rep);
} }
...@@ -106,10 +113,8 @@ Reduction EscapeAnalysisReducer::ReduceStore(Node* node) { ...@@ -106,10 +113,8 @@ Reduction EscapeAnalysisReducer::ReduceStore(Node* node) {
if (visited_.Contains(node->id())) return NoChange(); if (visited_.Contains(node->id())) return NoChange();
visited_.Add(node->id()); visited_.Add(node->id());
if (escape_analysis()->IsVirtual(NodeProperties::GetValueInput(node, 0))) { if (escape_analysis()->IsVirtual(NodeProperties::GetValueInput(node, 0))) {
if (FLAG_trace_turbo_escape) { TRACE("Removed #%d (%s) from effect chain\n", node->id(),
PrintF("Removed #%d (%s) from effect chain\n", node->id(), node->op()->mnemonic());
node->op()->mnemonic());
}
RelaxEffectsAndControls(node); RelaxEffectsAndControls(node);
return Changed(node); return Changed(node);
} }
...@@ -124,9 +129,7 @@ Reduction EscapeAnalysisReducer::ReduceAllocate(Node* node) { ...@@ -124,9 +129,7 @@ Reduction EscapeAnalysisReducer::ReduceAllocate(Node* node) {
if (escape_analysis()->IsVirtual(node)) { if (escape_analysis()->IsVirtual(node)) {
RelaxEffectsAndControls(node); RelaxEffectsAndControls(node);
counters()->turbo_escape_allocs_replaced()->Increment(); counters()->turbo_escape_allocs_replaced()->Increment();
if (FLAG_trace_turbo_escape) { TRACE("Removed allocate #%d from effect chain\n", node->id());
PrintF("Removed allocate #%d from effect chain\n", node->id());
}
return Changed(node); return Changed(node);
} }
return NoChange(); return NoChange();
...@@ -139,6 +142,7 @@ Reduction EscapeAnalysisReducer::ReduceFinishRegion(Node* node) { ...@@ -139,6 +142,7 @@ Reduction EscapeAnalysisReducer::ReduceFinishRegion(Node* node) {
if (effect->opcode() == IrOpcode::kBeginRegion) { if (effect->opcode() == IrOpcode::kBeginRegion) {
RelaxEffectsAndControls(effect); RelaxEffectsAndControls(effect);
RelaxEffectsAndControls(node); RelaxEffectsAndControls(node);
#ifdef DEBUG
if (FLAG_trace_turbo_escape) { if (FLAG_trace_turbo_escape) {
PrintF("Removed region #%d / #%d from effect chain,", effect->id(), PrintF("Removed region #%d / #%d from effect chain,", effect->id(),
node->id()); node->id());
...@@ -148,6 +152,7 @@ Reduction EscapeAnalysisReducer::ReduceFinishRegion(Node* node) { ...@@ -148,6 +152,7 @@ Reduction EscapeAnalysisReducer::ReduceFinishRegion(Node* node) {
} }
PrintF("\n"); PrintF("\n");
} }
#endif // DEBUG
return Changed(node); return Changed(node);
} }
return NoChange(); return NoChange();
...@@ -162,22 +167,16 @@ Reduction EscapeAnalysisReducer::ReduceReferenceEqual(Node* node) { ...@@ -162,22 +167,16 @@ Reduction EscapeAnalysisReducer::ReduceReferenceEqual(Node* node) {
if (escape_analysis()->IsVirtual(right) && if (escape_analysis()->IsVirtual(right) &&
escape_analysis()->CompareVirtualObjects(left, right)) { escape_analysis()->CompareVirtualObjects(left, right)) {
ReplaceWithValue(node, jsgraph()->TrueConstant()); ReplaceWithValue(node, jsgraph()->TrueConstant());
if (FLAG_trace_turbo_escape) { TRACE("Replaced ref eq #%d with true\n", node->id());
PrintF("Replaced ref eq #%d with true\n", node->id());
}
} }
// Right-hand side is not a virtual object, or a different one. // Right-hand side is not a virtual object, or a different one.
ReplaceWithValue(node, jsgraph()->FalseConstant()); ReplaceWithValue(node, jsgraph()->FalseConstant());
if (FLAG_trace_turbo_escape) { TRACE("Replaced ref eq #%d with false\n", node->id());
PrintF("Replaced ref eq #%d with false\n", node->id());
}
return Replace(node); return Replace(node);
} else if (escape_analysis()->IsVirtual(right)) { } else if (escape_analysis()->IsVirtual(right)) {
// Left-hand side is not a virtual object. // Left-hand side is not a virtual object.
ReplaceWithValue(node, jsgraph()->FalseConstant()); ReplaceWithValue(node, jsgraph()->FalseConstant());
if (FLAG_trace_turbo_escape) { TRACE("Replaced ref eq #%d with false\n", node->id());
PrintF("Replaced ref eq #%d with false\n", node->id());
}
} }
return NoChange(); return NoChange();
} }
...@@ -188,9 +187,7 @@ Reduction EscapeAnalysisReducer::ReduceObjectIsSmi(Node* node) { ...@@ -188,9 +187,7 @@ Reduction EscapeAnalysisReducer::ReduceObjectIsSmi(Node* node) {
Node* input = NodeProperties::GetValueInput(node, 0); Node* input = NodeProperties::GetValueInput(node, 0);
if (escape_analysis()->IsVirtual(input)) { if (escape_analysis()->IsVirtual(input)) {
ReplaceWithValue(node, jsgraph()->FalseConstant()); ReplaceWithValue(node, jsgraph()->FalseConstant());
if (FLAG_trace_turbo_escape) { TRACE("Replaced ObjectIsSmi #%d with false\n", node->id());
PrintF("Replaced ObjectIsSmi #%d with false\n", node->id());
}
return Replace(node); return Replace(node);
} }
return NoChange(); return NoChange();
...@@ -227,9 +224,7 @@ Node* EscapeAnalysisReducer::ReduceDeoptState(Node* node, Node* effect, ...@@ -227,9 +224,7 @@ Node* EscapeAnalysisReducer::ReduceDeoptState(Node* node, Node* effect,
visited_.Contains(node->id())) { visited_.Contains(node->id())) {
return nullptr; return nullptr;
} }
if (FLAG_trace_turbo_escape) { TRACE("Reducing %s %d\n", node->op()->mnemonic(), node->id());
PrintF("Reducing %s %d\n", node->op()->mnemonic(), node->id());
}
Node* clone = nullptr; Node* clone = nullptr;
bool node_multiused = node->UseCount() > 1; bool node_multiused = node->UseCount() > 1;
bool multiple_users_rec = multiple_users || node_multiused; bool multiple_users_rec = multiple_users || node_multiused;
...@@ -238,13 +233,9 @@ Node* EscapeAnalysisReducer::ReduceDeoptState(Node* node, Node* effect, ...@@ -238,13 +233,9 @@ Node* EscapeAnalysisReducer::ReduceDeoptState(Node* node, Node* effect,
if (input->opcode() == IrOpcode::kStateValues) { if (input->opcode() == IrOpcode::kStateValues) {
if (Node* ret = ReduceDeoptState(input, effect, multiple_users_rec)) { if (Node* ret = ReduceDeoptState(input, effect, multiple_users_rec)) {
if (node_multiused || (multiple_users && !clone)) { if (node_multiused || (multiple_users && !clone)) {
if (FLAG_trace_turbo_escape) { TRACE(" Cloning #%d", node->id());
PrintF(" Cloning #%d", node->id());
}
node = clone = jsgraph()->graph()->CloneNode(node); node = clone = jsgraph()->graph()->CloneNode(node);
if (FLAG_trace_turbo_escape) { TRACE(" to #%d\n", node->id());
PrintF(" to #%d\n", node->id());
}
node_multiused = false; node_multiused = false;
} }
NodeProperties::ReplaceValueInput(node, ret, i); NodeProperties::ReplaceValueInput(node, ret, i);
...@@ -264,13 +255,9 @@ Node* EscapeAnalysisReducer::ReduceDeoptState(Node* node, Node* effect, ...@@ -264,13 +255,9 @@ Node* EscapeAnalysisReducer::ReduceDeoptState(Node* node, Node* effect,
if (Node* ret = if (Node* ret =
ReduceDeoptState(outer_frame_state, effect, multiple_users_rec)) { ReduceDeoptState(outer_frame_state, effect, multiple_users_rec)) {
if (node_multiused || (multiple_users && !clone)) { if (node_multiused || (multiple_users && !clone)) {
if (FLAG_trace_turbo_escape) { TRACE(" Cloning #%d", node->id());
PrintF(" Cloning #%d", node->id());
}
node = clone = jsgraph()->graph()->CloneNode(node); node = clone = jsgraph()->graph()->CloneNode(node);
if (FLAG_trace_turbo_escape) { TRACE(" to #%d\n", node->id());
PrintF(" to #%d\n", node->id());
}
} }
NodeProperties::ReplaceFrameStateInput(node, 0, ret); NodeProperties::ReplaceFrameStateInput(node, 0, ret);
} }
...@@ -287,10 +274,8 @@ Node* EscapeAnalysisReducer::ReduceStateValueInput(Node* node, int node_index, ...@@ -287,10 +274,8 @@ Node* EscapeAnalysisReducer::ReduceStateValueInput(Node* node, int node_index,
bool already_cloned, bool already_cloned,
bool multiple_users) { bool multiple_users) {
Node* input = NodeProperties::GetValueInput(node, node_index); Node* input = NodeProperties::GetValueInput(node, node_index);
if (FLAG_trace_turbo_escape) { TRACE("Reducing State Input #%d (%s)\n", input->id(),
PrintF("Reducing State Input #%d (%s)\n", input->id(), input->op()->mnemonic());
input->op()->mnemonic());
}
Node* clone = nullptr; Node* clone = nullptr;
if (input->opcode() == IrOpcode::kFinishRegion || if (input->opcode() == IrOpcode::kFinishRegion ||
input->opcode() == IrOpcode::kAllocate) { input->opcode() == IrOpcode::kAllocate) {
...@@ -298,25 +283,19 @@ Node* EscapeAnalysisReducer::ReduceStateValueInput(Node* node, int node_index, ...@@ -298,25 +283,19 @@ Node* EscapeAnalysisReducer::ReduceStateValueInput(Node* node, int node_index,
if (Node* object_state = if (Node* object_state =
escape_analysis()->GetOrCreateObjectState(effect, input)) { escape_analysis()->GetOrCreateObjectState(effect, input)) {
if (node_multiused || (multiple_users && !already_cloned)) { if (node_multiused || (multiple_users && !already_cloned)) {
if (FLAG_trace_turbo_escape) { TRACE("Cloning #%d", node->id());
PrintF("Cloning #%d", node->id());
}
node = clone = jsgraph()->graph()->CloneNode(node); node = clone = jsgraph()->graph()->CloneNode(node);
if (FLAG_trace_turbo_escape) { TRACE(" to #%d\n", node->id());
PrintF(" to #%d\n", node->id());
}
node_multiused = false; node_multiused = false;
already_cloned = true; already_cloned = true;
} }
NodeProperties::ReplaceValueInput(node, object_state, node_index); NodeProperties::ReplaceValueInput(node, object_state, node_index);
if (FLAG_trace_turbo_escape) { TRACE("Replaced state #%d input #%d with object state #%d\n",
PrintF("Replaced state #%d input #%d with object state #%d\n", node->id(), input->id(), object_state->id());
node->id(), input->id(), object_state->id());
}
} else { } else {
if (FLAG_trace_turbo_escape) { TRACE("No object state replacement for #%d at effect #%d available.\n",
PrintF("No object state replacement available.\n"); input->id(), effect->id());
} UNREACHABLE();
} }
} }
} }
......
...@@ -24,6 +24,15 @@ namespace v8 { ...@@ -24,6 +24,15 @@ namespace v8 {
namespace internal { namespace internal {
namespace compiler { namespace compiler {
#ifdef DEBUG
#define TRACE(...) \
do { \
if (FLAG_trace_turbo_escape) PrintF(__VA_ARGS__); \
} while (false)
#else
#define TRACE(...)
#endif
const EscapeAnalysis::Alias EscapeAnalysis::kNotReachable = const EscapeAnalysis::Alias EscapeAnalysis::kNotReachable =
std::numeric_limits<Alias>::max(); std::numeric_limits<Alias>::max();
const EscapeAnalysis::Alias EscapeAnalysis::kUntrackable = const EscapeAnalysis::Alias EscapeAnalysis::kUntrackable =
...@@ -75,9 +84,9 @@ class VirtualObject : public ZoneObject { ...@@ -75,9 +84,9 @@ class VirtualObject : public ZoneObject {
bool changed = fields_[offset] != node || phi_[offset] != created_phi; bool changed = fields_[offset] != node || phi_[offset] != created_phi;
fields_[offset] = node; fields_[offset] = node;
phi_[offset] = created_phi; phi_[offset] = created_phi;
if (changed && FLAG_trace_turbo_escape && node) { if (changed && node) {
PrintF("Setting field %zu of #%d to #%d (%s)\n", offset, id(), node->id(), TRACE("Setting field %zu of #%d to #%d (%s)\n", offset, id(), node->id(),
node->op()->mnemonic()); node->op()->mnemonic());
} }
return changed; return changed;
} }
...@@ -290,9 +299,7 @@ bool VirtualState::UpdateFrom(VirtualState* from, Zone* zone) { ...@@ -290,9 +299,7 @@ bool VirtualState::UpdateFrom(VirtualState* from, Zone* zone) {
continue; continue;
} }
if (FLAG_trace_turbo_escape) { TRACE(" Updating fields of @%d\n", alias);
PrintF(" Updating fields of @%d\n", alias);
}
changed = ls->UpdateFrom(*rs) || changed; changed = ls->UpdateFrom(*rs) || changed;
} }
...@@ -354,18 +361,23 @@ bool VirtualState::MergeFrom(MergeCache* cache, Zone* zone, Graph* graph, ...@@ -354,18 +361,23 @@ bool VirtualState::MergeFrom(MergeCache* cache, Zone* zone, Graph* graph,
for (EscapeAnalysis::Alias alias = 0; alias < size(); ++alias) { for (EscapeAnalysis::Alias alias = 0; alias < size(); ++alias) {
size_t fields = cache->LoadVirtualObjectsFromStatesFor(alias); size_t fields = cache->LoadVirtualObjectsFromStatesFor(alias);
if (cache->objects().size() == cache->states().size()) { if (cache->objects().size() == cache->states().size()) {
if (FLAG_trace_turbo_escape) {
PrintF(" Merging virtual objects of @%d\n", alias);
}
VirtualObject* mergeObject = GetOrCreateTrackedVirtualObject( VirtualObject* mergeObject = GetOrCreateTrackedVirtualObject(
alias, cache->objects().front()->id(), fields, zone); alias, cache->objects().front()->id(), fields, zone);
#ifdef DEBUG
if (FLAG_trace_turbo_escape) {
PrintF(" Alias @%d, merging into %p virtual objects", alias,
static_cast<void*>(mergeObject));
for (size_t i = 0; i < cache->objects().size(); i++) {
PrintF(" %p", static_cast<void*>(cache->objects()[i]));
}
PrintF("\n");
}
#endif // DEBUG
changed = mergeObject->ResizeFields(fields) || changed; changed = mergeObject->ResizeFields(fields) || changed;
for (size_t i = 0; i < fields; ++i) { for (size_t i = 0; i < fields; ++i) {
if (Node* field = cache->GetFields(i)) { if (Node* field = cache->GetFields(i)) {
changed = mergeObject->SetField(i, field) || changed; changed = mergeObject->SetField(i, field) || changed;
if (FLAG_trace_turbo_escape) { TRACE(" Field %zu agree on rep #%d\n", i, field->id());
PrintF(" Field %zu agree on rep #%d\n", i, field->id());
}
} else { } else {
int value_input_count = static_cast<int>(cache->fields().size()); int value_input_count = static_cast<int>(cache->fields().size());
if (cache->fields().size() == cache->objects().size()) { if (cache->fields().size() == cache->objects().size()) {
...@@ -377,6 +389,7 @@ bool VirtualState::MergeFrom(MergeCache* cache, Zone* zone, Graph* graph, ...@@ -377,6 +389,7 @@ bool VirtualState::MergeFrom(MergeCache* cache, Zone* zone, Graph* graph,
value_input_count), value_input_count),
value_input_count + 1, &cache->fields().front()); value_input_count + 1, &cache->fields().front());
mergeObject->SetField(i, phi, true); mergeObject->SetField(i, phi, true);
#ifdef DEBUG
if (FLAG_trace_turbo_escape) { if (FLAG_trace_turbo_escape) {
PrintF(" Creating Phi #%d as merge of", phi->id()); PrintF(" Creating Phi #%d as merge of", phi->id());
for (int i = 0; i < value_input_count; i++) { for (int i = 0; i < value_input_count; i++) {
...@@ -385,6 +398,7 @@ bool VirtualState::MergeFrom(MergeCache* cache, Zone* zone, Graph* graph, ...@@ -385,6 +398,7 @@ bool VirtualState::MergeFrom(MergeCache* cache, Zone* zone, Graph* graph,
} }
PrintF("\n"); PrintF("\n");
} }
#endif // DEBUG
changed = true; changed = true;
} else { } else {
DCHECK(rep->opcode() == IrOpcode::kPhi); DCHECK(rep->opcode() == IrOpcode::kPhi);
...@@ -402,18 +416,17 @@ bool VirtualState::MergeFrom(MergeCache* cache, Zone* zone, Graph* graph, ...@@ -402,18 +416,17 @@ bool VirtualState::MergeFrom(MergeCache* cache, Zone* zone, Graph* graph,
} }
} }
if (rep->op()->ValueInputCount() != value_input_count) { if (rep->op()->ValueInputCount() != value_input_count) {
if (FLAG_trace_turbo_escape) { TRACE(" Widening Phi #%d of arity %d to %d\n", rep->id(),
PrintF(" Widening Phi #%d of arity %d to %d", rep->id(), rep->op()->ValueInputCount(), value_input_count);
rep->op()->ValueInputCount(), value_input_count);
}
NodeProperties::ChangeOp( NodeProperties::ChangeOp(
rep, common->Phi(MachineRepresentation::kTagged, rep, common->Phi(MachineRepresentation::kTagged,
value_input_count)); value_input_count));
} }
} }
} else { } else {
if (FLAG_trace_turbo_escape) { if (mergeObject->GetField(i) != nullptr) {
PrintF(" Field %zu cleared\n", i); TRACE(" Field %zu cleared\n", i);
changed = true;
} }
changed = mergeObject->SetField(i, nullptr) || changed; changed = mergeObject->SetField(i, nullptr) || changed;
} }
...@@ -580,10 +593,8 @@ void EscapeStatusAnalysis::ProcessStoreField(Node* node) { ...@@ -580,10 +593,8 @@ void EscapeStatusAnalysis::ProcessStoreField(Node* node) {
if ((IsEscaped(to) || !IsAllocation(to)) && SetEscaped(val)) { if ((IsEscaped(to) || !IsAllocation(to)) && SetEscaped(val)) {
RevisitUses(val); RevisitUses(val);
RevisitInputs(val); RevisitInputs(val);
if (FLAG_trace_turbo_escape) { TRACE("Setting #%d (%s) to escaped because of store to field of #%d\n",
PrintF("Setting #%d (%s) to escaped because of store to field of #%d\n", val->id(), val->op()->mnemonic(), to->id());
val->id(), val->op()->mnemonic(), to->id());
}
} }
} }
...@@ -595,10 +606,8 @@ void EscapeStatusAnalysis::ProcessStoreElement(Node* node) { ...@@ -595,10 +606,8 @@ void EscapeStatusAnalysis::ProcessStoreElement(Node* node) {
if ((IsEscaped(to) || !IsAllocation(to)) && SetEscaped(val)) { if ((IsEscaped(to) || !IsAllocation(to)) && SetEscaped(val)) {
RevisitUses(val); RevisitUses(val);
RevisitInputs(val); RevisitInputs(val);
if (FLAG_trace_turbo_escape) { TRACE("Setting #%d (%s) to escaped because of store to field of #%d\n",
PrintF("Setting #%d (%s) to escaped because of store to field of #%d\n", val->id(), val->op()->mnemonic(), to->id());
val->id(), val->op()->mnemonic(), to->id());
}
} }
} }
...@@ -607,10 +616,8 @@ void EscapeStatusAnalysis::ProcessAllocate(Node* node) { ...@@ -607,10 +616,8 @@ void EscapeStatusAnalysis::ProcessAllocate(Node* node) {
DCHECK_EQ(node->opcode(), IrOpcode::kAllocate); DCHECK_EQ(node->opcode(), IrOpcode::kAllocate);
if (!HasEntry(node)) { if (!HasEntry(node)) {
status_[node->id()] |= kTracked; status_[node->id()] |= kTracked;
if (FLAG_trace_turbo_escape) { TRACE("Created status entry for node #%d (%s)\n", node->id(),
PrintF("Created status entry for node #%d (%s)\n", node->id(), node->op()->mnemonic());
node->op()->mnemonic());
}
NumberMatcher size(node->InputAt(0)); NumberMatcher size(node->InputAt(0));
DCHECK(node->InputAt(0)->opcode() != IrOpcode::kInt32Constant && DCHECK(node->InputAt(0)->opcode() != IrOpcode::kInt32Constant &&
node->InputAt(0)->opcode() != IrOpcode::kInt64Constant && node->InputAt(0)->opcode() != IrOpcode::kInt64Constant &&
...@@ -618,10 +625,7 @@ void EscapeStatusAnalysis::ProcessAllocate(Node* node) { ...@@ -618,10 +625,7 @@ void EscapeStatusAnalysis::ProcessAllocate(Node* node) {
node->InputAt(0)->opcode() != IrOpcode::kFloat64Constant); node->InputAt(0)->opcode() != IrOpcode::kFloat64Constant);
if (!size.HasValue() && SetEscaped(node)) { if (!size.HasValue() && SetEscaped(node)) {
RevisitUses(node); RevisitUses(node);
if (FLAG_trace_turbo_escape) { TRACE("Setting #%d to escaped because of non-const alloc\n", node->id());
PrintF("Setting #%d to escaped because of non-const alloc\n",
node->id());
}
// This node is known to escape, uses do not have to be checked. // This node is known to escape, uses do not have to be checked.
return; return;
} }
...@@ -642,13 +646,11 @@ bool EscapeStatusAnalysis::CheckUsesForEscape(Node* uses, Node* rep, ...@@ -642,13 +646,11 @@ bool EscapeStatusAnalysis::CheckUsesForEscape(Node* uses, Node* rep,
switch (use->opcode()) { switch (use->opcode()) {
case IrOpcode::kPhi: case IrOpcode::kPhi:
if (phi_escaping && SetEscaped(rep)) { if (phi_escaping && SetEscaped(rep)) {
if (FLAG_trace_turbo_escape) { TRACE(
PrintF( "Setting #%d (%s) to escaped because of use by phi node "
"Setting #%d (%s) to escaped because of use by phi node " "#%d (%s)\n",
"#%d (%s)\n", rep->id(), rep->op()->mnemonic(), use->id(),
rep->id(), rep->op()->mnemonic(), use->id(), use->op()->mnemonic());
use->op()->mnemonic());
}
return true; return true;
} }
// Fallthrough. // Fallthrough.
...@@ -661,45 +663,41 @@ bool EscapeStatusAnalysis::CheckUsesForEscape(Node* uses, Node* rep, ...@@ -661,45 +663,41 @@ bool EscapeStatusAnalysis::CheckUsesForEscape(Node* uses, Node* rep,
case IrOpcode::kReferenceEqual: case IrOpcode::kReferenceEqual:
case IrOpcode::kFinishRegion: case IrOpcode::kFinishRegion:
if (IsEscaped(use) && SetEscaped(rep)) { if (IsEscaped(use) && SetEscaped(rep)) {
if (FLAG_trace_turbo_escape) { TRACE(
PrintF( "Setting #%d (%s) to escaped because of use by escaping node "
"Setting #%d (%s) to escaped because of use by escaping node " "#%d (%s)\n",
"#%d (%s)\n", rep->id(), rep->op()->mnemonic(), use->id(),
rep->id(), rep->op()->mnemonic(), use->id(), use->op()->mnemonic());
use->op()->mnemonic());
}
return true; return true;
} }
break; break;
case IrOpcode::kObjectIsSmi: case IrOpcode::kObjectIsSmi:
if (!IsAllocation(rep) && SetEscaped(rep)) { if (!IsAllocation(rep) && SetEscaped(rep)) {
PrintF("Setting #%d (%s) to escaped because of use by #%d (%s)\n", TRACE("Setting #%d (%s) to escaped because of use by #%d (%s)\n",
rep->id(), rep->op()->mnemonic(), use->id(), rep->id(), rep->op()->mnemonic(), use->id(),
use->op()->mnemonic()); use->op()->mnemonic());
return true; return true;
} }
break; break;
case IrOpcode::kSelect: case IrOpcode::kSelect:
if (SetEscaped(rep)) { if (SetEscaped(rep)) {
PrintF("Setting #%d (%s) to escaped because of use by #%d (%s)\n", TRACE("Setting #%d (%s) to escaped because of use by #%d (%s)\n",
rep->id(), rep->op()->mnemonic(), use->id(), rep->id(), rep->op()->mnemonic(), use->id(),
use->op()->mnemonic()); use->op()->mnemonic());
return true; return true;
} }
break; break;
default: default:
if (use->op()->EffectInputCount() == 0 && if (use->op()->EffectInputCount() == 0 &&
uses->op()->EffectInputCount() > 0) { uses->op()->EffectInputCount() > 0) {
PrintF("Encountered unaccounted use by #%d (%s)\n", use->id(), TRACE("Encountered unaccounted use by #%d (%s)\n", use->id(),
use->op()->mnemonic()); use->op()->mnemonic());
UNREACHABLE(); UNREACHABLE();
} }
if (SetEscaped(rep)) { if (SetEscaped(rep)) {
if (FLAG_trace_turbo_escape) { TRACE("Setting #%d (%s) to escaped because of use by #%d (%s)\n",
PrintF("Setting #%d (%s) to escaped because of use by #%d (%s)\n", rep->id(), rep->op()->mnemonic(), use->id(),
rep->id(), rep->op()->mnemonic(), use->id(), use->op()->mnemonic());
use->op()->mnemonic());
}
return true; return true;
} }
} }
...@@ -762,6 +760,7 @@ void EscapeAnalysis::AssignAliases() { ...@@ -762,6 +760,7 @@ void EscapeAnalysis::AssignAliases() {
CHECK_LT(graph()->NodeCount(), kUntrackable); CHECK_LT(graph()->NodeCount(), kUntrackable);
aliases_.resize(graph()->NodeCount(), kNotReachable); aliases_.resize(graph()->NodeCount(), kNotReachable);
aliases_[graph()->end()->id()] = kUntrackable; aliases_[graph()->end()->id()] = kUntrackable;
TRACE("Discovering trackable nodes");
while (!stack.empty()) { while (!stack.empty()) {
Node* node = stack.back(); Node* node = stack.back();
stack.pop_back(); stack.pop_back();
...@@ -769,6 +768,8 @@ void EscapeAnalysis::AssignAliases() { ...@@ -769,6 +768,8 @@ void EscapeAnalysis::AssignAliases() {
case IrOpcode::kAllocate: case IrOpcode::kAllocate:
if (aliases_[node->id()] >= kUntrackable) { if (aliases_[node->id()] >= kUntrackable) {
aliases_[node->id()] = NextAlias(); aliases_[node->id()] = NextAlias();
TRACE(" @%d:%s#%u", aliases_[node->id()], node->op()->mnemonic(),
node->id());
} }
break; break;
case IrOpcode::kFinishRegion: { case IrOpcode::kFinishRegion: {
...@@ -779,10 +780,17 @@ void EscapeAnalysis::AssignAliases() { ...@@ -779,10 +780,17 @@ void EscapeAnalysis::AssignAliases() {
stack.push_back(allocate); stack.push_back(allocate);
} }
aliases_[allocate->id()] = NextAlias(); aliases_[allocate->id()] = NextAlias();
TRACE(" @%d:%s#%u", aliases_[allocate->id()],
allocate->op()->mnemonic(), allocate->id());
} }
aliases_[node->id()] = aliases_[allocate->id()]; aliases_[node->id()] = aliases_[allocate->id()];
TRACE(" @%d:%s#%u", aliases_[node->id()], node->op()->mnemonic(),
node->id());
} else { } else {
aliases_[node->id()] = NextAlias(); aliases_[node->id()] = NextAlias();
TRACE(" @%d:%s#%u", aliases_[node->id()], node->op()->mnemonic(),
node->id());
} }
break; break;
} }
...@@ -798,18 +806,7 @@ void EscapeAnalysis::AssignAliases() { ...@@ -798,18 +806,7 @@ void EscapeAnalysis::AssignAliases() {
} }
} }
} }
TRACE("\n");
if (FLAG_trace_turbo_escape) {
PrintF("Discovered trackable nodes");
for (EscapeAnalysis::Alias id = 0; id < graph()->NodeCount(); ++id) {
if (aliases_[id] < kUntrackable) {
if (FLAG_trace_turbo_escape) {
PrintF(" #%u", id);
}
}
}
PrintF("\n");
}
} }
...@@ -844,9 +841,11 @@ void EscapeAnalysis::RunObjectAnalysis() { ...@@ -844,9 +841,11 @@ void EscapeAnalysis::RunObjectAnalysis() {
} }
} }
} }
#ifdef DEBUG
if (FLAG_trace_turbo_escape) { if (FLAG_trace_turbo_escape) {
DebugPrint(); DebugPrint();
} }
#endif
} }
...@@ -971,11 +970,9 @@ void EscapeAnalysis::ForwardVirtualState(Node* node) { ...@@ -971,11 +970,9 @@ void EscapeAnalysis::ForwardVirtualState(Node* node) {
} }
DCHECK_NOT_NULL(virtual_states_[effect->id()]); DCHECK_NOT_NULL(virtual_states_[effect->id()]);
if (IsEffectBranchPoint(effect)) { if (IsEffectBranchPoint(effect)) {
if (FLAG_trace_turbo_escape) { TRACE("Copying virtual state %p from #%d (%s) to #%d (%s)\n",
PrintF("Copying object state %p from #%d (%s) to #%d (%s)\n", static_cast<void*>(virtual_states_[effect->id()]), effect->id(),
static_cast<void*>(virtual_states_[effect->id()]), effect->id(), effect->op()->mnemonic(), node->id(), node->op()->mnemonic());
effect->op()->mnemonic(), node->id(), node->op()->mnemonic());
}
if (!virtual_states_[node->id()]) { if (!virtual_states_[node->id()]) {
virtual_states_[node->id()] = virtual_states_[node->id()] =
new (zone()) VirtualState(*virtual_states_[effect->id()]); new (zone()) VirtualState(*virtual_states_[effect->id()]);
...@@ -985,11 +982,9 @@ void EscapeAnalysis::ForwardVirtualState(Node* node) { ...@@ -985,11 +982,9 @@ void EscapeAnalysis::ForwardVirtualState(Node* node) {
} }
} else { } else {
virtual_states_[node->id()] = virtual_states_[effect->id()]; virtual_states_[node->id()] = virtual_states_[effect->id()];
if (FLAG_trace_turbo_escape) { TRACE("Forwarding virtual state %p from #%d (%s) to #%d (%s)\n",
PrintF("Forwarding object state %p from #%d (%s) to #%d (%s)\n", static_cast<void*>(virtual_states_[effect->id()]), effect->id(),
static_cast<void*>(virtual_states_[effect->id()]), effect->id(), effect->op()->mnemonic(), node->id(), node->op()->mnemonic());
effect->op()->mnemonic(), node->id(), node->op()->mnemonic());
}
} }
} }
...@@ -1009,20 +1004,16 @@ bool EscapeAnalysis::ProcessEffectPhi(Node* node) { ...@@ -1009,20 +1004,16 @@ bool EscapeAnalysis::ProcessEffectPhi(Node* node) {
mergeState = new (zone()) VirtualState(zone(), AliasCount()); mergeState = new (zone()) VirtualState(zone(), AliasCount());
virtual_states_[node->id()] = mergeState; virtual_states_[node->id()] = mergeState;
changed = true; changed = true;
if (FLAG_trace_turbo_escape) { TRACE("Effect Phi #%d got new virtual state %p.\n", node->id(),
PrintF("Effect Phi #%d got new states map %p.\n", node->id(), static_cast<void*>(mergeState));
static_cast<void*>(mergeState));
}
} else if (mergeState->GetLastChanged() != node) { } else if (mergeState->GetLastChanged() != node) {
changed = true; changed = true;
} }
cache_->Clear(); cache_->Clear();
if (FLAG_trace_turbo_escape) { TRACE("At Effect Phi #%d, merging states into %p:", node->id(),
PrintF("At Effect Phi #%d, merging states into %p:", node->id(), static_cast<void*>(mergeState));
static_cast<void*>(mergeState));
}
for (int i = 0; i < node->op()->EffectInputCount(); ++i) { for (int i = 0; i < node->op()->EffectInputCount(); ++i) {
Node* input = NodeProperties::GetEffectInput(node, i); Node* input = NodeProperties::GetEffectInput(node, i);
...@@ -1030,14 +1021,10 @@ bool EscapeAnalysis::ProcessEffectPhi(Node* node) { ...@@ -1030,14 +1021,10 @@ bool EscapeAnalysis::ProcessEffectPhi(Node* node) {
if (state) { if (state) {
cache_->states().push_back(state); cache_->states().push_back(state);
} }
if (FLAG_trace_turbo_escape) { TRACE(" %p (from %d %s)", static_cast<void*>(state), input->id(),
PrintF(" %p (from %d %s)", static_cast<void*>(state), input->id(), input->op()->mnemonic());
input->op()->mnemonic());
}
}
if (FLAG_trace_turbo_escape) {
PrintF("\n");
} }
TRACE("\n");
if (cache_->states().size() == 0) { if (cache_->states().size() == 0) {
return changed; return changed;
...@@ -1047,9 +1034,7 @@ bool EscapeAnalysis::ProcessEffectPhi(Node* node) { ...@@ -1047,9 +1034,7 @@ bool EscapeAnalysis::ProcessEffectPhi(Node* node) {
NodeProperties::GetControlInput(node)) || NodeProperties::GetControlInput(node)) ||
changed; changed;
if (FLAG_trace_turbo_escape) { TRACE("Merge %s the node.\n", changed ? "changed" : "did not change");
PrintF("Merge %s the node.\n", changed ? "changed" : "did not change");
}
if (changed) { if (changed) {
mergeState->LastChangedAt(node); mergeState->LastChangedAt(node);
...@@ -1098,10 +1083,8 @@ void EscapeAnalysis::ProcessFinishRegion(Node* node) { ...@@ -1098,10 +1083,8 @@ void EscapeAnalysis::ProcessFinishRegion(Node* node) {
state->VirtualObjectFromAlias(aliases_[allocation->id()]); state->VirtualObjectFromAlias(aliases_[allocation->id()]);
DCHECK_NOT_NULL(vobj_alloc); DCHECK_NOT_NULL(vobj_alloc);
state->SetVirtualObject(aliases_[node->id()], vobj_alloc); state->SetVirtualObject(aliases_[node->id()], vobj_alloc);
if (FLAG_trace_turbo_escape) { TRACE("Linked finish region node #%d to node #%d\n", node->id(),
PrintF("Linked finish region node #%d to node #%d\n", node->id(), allocation->id());
allocation->id());
}
state->LastChangedAt(node); state->LastChangedAt(node);
} }
} }
...@@ -1130,13 +1113,11 @@ bool EscapeAnalysis::UpdateReplacement(VirtualState* state, Node* node, ...@@ -1130,13 +1113,11 @@ bool EscapeAnalysis::UpdateReplacement(VirtualState* state, Node* node,
Node* rep) { Node* rep) {
if (SetReplacement(node, rep)) { if (SetReplacement(node, rep)) {
state->LastChangedAt(node); state->LastChangedAt(node);
if (FLAG_trace_turbo_escape) { if (rep) {
if (rep) { TRACE("Replacement of #%d is #%d (%s)\n", node->id(), rep->id(),
PrintF("Replacement of #%d is #%d (%s)\n", node->id(), rep->id(), rep->op()->mnemonic());
rep->op()->mnemonic()); } else {
} else { TRACE("Replacement of #%d cleared\n", node->id());
PrintF("Replacement of #%d cleared\n", node->id());
}
} }
return true; return true;
} }
...@@ -1221,9 +1202,7 @@ int EscapeAnalysis::OffsetFromAccess(Node* node) { ...@@ -1221,9 +1202,7 @@ int EscapeAnalysis::OffsetFromAccess(Node* node) {
void EscapeAnalysis::ProcessLoadFromPhi(int offset, Node* from, Node* node, void EscapeAnalysis::ProcessLoadFromPhi(int offset, Node* from, Node* node,
VirtualState* state) { VirtualState* state) {
if (FLAG_trace_turbo_escape) { TRACE("Load #%d from phi #%d", node->id(), from->id());
PrintF("Load #%d from phi #%d", node->id(), from->id());
}
cache_->fields().clear(); cache_->fields().clear();
for (int i = 0; i < node->op()->ValueInputCount(); ++i) { for (int i = 0; i < node->op()->ValueInputCount(); ++i) {
...@@ -1245,17 +1224,15 @@ void EscapeAnalysis::ProcessLoadFromPhi(int offset, Node* from, Node* node, ...@@ -1245,17 +1224,15 @@ void EscapeAnalysis::ProcessLoadFromPhi(int offset, Node* from, Node* node,
escape_status_.Resize(); escape_status_.Resize();
SetReplacement(node, phi); SetReplacement(node, phi);
state->LastChangedAt(node); state->LastChangedAt(node);
if (FLAG_trace_turbo_escape) { TRACE(" got phi created.\n");
PrintF(" got phi created.\n"); } else {
} TRACE(" has already phi #%d.\n", rep->id());
} else if (FLAG_trace_turbo_escape) {
PrintF(" has already phi #%d.\n", rep->id());
} }
} else if (FLAG_trace_turbo_escape) { } else {
PrintF(" has incomplete field info.\n"); TRACE(" has incomplete field info.\n");
} }
} else if (FLAG_trace_turbo_escape) { } else {
PrintF(" has incomplete virtual object info.\n"); TRACE(" has incomplete virtual object info.\n");
} }
} }
...@@ -1321,14 +1298,11 @@ void EscapeAnalysis::ProcessLoadElement(Node* node) { ...@@ -1321,14 +1298,11 @@ void EscapeAnalysis::ProcessLoadElement(Node* node) {
} else { } else {
// We have a load from a non-const index, cannot eliminate object. // We have a load from a non-const index, cannot eliminate object.
if (SetEscaped(from)) { if (SetEscaped(from)) {
if (FLAG_trace_turbo_escape) { TRACE(
PrintF( "Setting #%d (%s) to escaped because load element #%d from non-const "
"Setting #%d (%s) to escaped because load element #%d from " "index #%d (%s)\n",
"non-const " from->id(), from->op()->mnemonic(), node->id(), index_node->id(),
"index #%d (%s)\n", index_node->op()->mnemonic());
from->id(), from->op()->mnemonic(), node->id(), index_node->id(),
index_node->op()->mnemonic());
}
} }
} }
} }
...@@ -1377,17 +1351,15 @@ void EscapeAnalysis::ProcessStoreElement(Node* node) { ...@@ -1377,17 +1351,15 @@ void EscapeAnalysis::ProcessStoreElement(Node* node) {
} else { } else {
// We have a store to a non-const index, cannot eliminate object. // We have a store to a non-const index, cannot eliminate object.
if (SetEscaped(to)) { if (SetEscaped(to)) {
if (FLAG_trace_turbo_escape) { TRACE(
PrintF( "Setting #%d (%s) to escaped because store element #%d to non-const "
"Setting #%d (%s) to escaped because store element #%d to " "index #%d (%s)\n",
"non-const " to->id(), to->op()->mnemonic(), node->id(), index_node->id(),
"index #%d (%s)\n", index_node->op()->mnemonic());
to->id(), to->op()->mnemonic(), node->id(), index_node->id(),
index_node->op()->mnemonic());
}
} }
if (obj && obj->IsTracked() && obj->ClearAllFields()) { if (obj && obj->IsTracked() && obj->ClearAllFields()) {
state->LastChangedAt(node); state->LastChangedAt(node);
TRACE("Cleared all fields of @%d:#%d\n", aliases_[obj->id()], obj->id());
} }
} }
} }
...@@ -1413,13 +1385,11 @@ Node* EscapeAnalysis::GetOrCreateObjectState(Node* effect, Node* node) { ...@@ -1413,13 +1385,11 @@ Node* EscapeAnalysis::GetOrCreateObjectState(Node* effect, Node* node) {
graph()->NewNode(common()->ObjectState(input_count, vobj->id()), graph()->NewNode(common()->ObjectState(input_count, vobj->id()),
input_count, &cache_->fields().front()); input_count, &cache_->fields().front());
vobj->SetObjectState(new_object_state); vobj->SetObjectState(new_object_state);
if (FLAG_trace_turbo_escape) { TRACE(
PrintF( "Creating object state #%d for vobj %p (from node #%d) at effect "
"Creating object state #%d for vobj %p (from node #%d) at effect " "#%d\n",
"#%d\n", new_object_state->id(), static_cast<void*>(vobj), node->id(),
new_object_state->id(), static_cast<void*>(vobj), node->id(), effect->id());
effect->id());
}
// Now fix uses of other objects. // Now fix uses of other objects.
for (size_t i = 0; i < vobj->field_count(); ++i) { for (size_t i = 0; i < vobj->field_count(); ++i) {
if (Node* field = vobj->GetField(i)) { if (Node* field = vobj->GetField(i)) {
...@@ -1450,7 +1420,7 @@ void EscapeAnalysis::DebugPrintObject(VirtualObject* object, Alias alias) { ...@@ -1450,7 +1420,7 @@ void EscapeAnalysis::DebugPrintObject(VirtualObject* object, Alias alias) {
void EscapeAnalysis::DebugPrintState(VirtualState* state) { void EscapeAnalysis::DebugPrintState(VirtualState* state) {
PrintF("Dumping object state %p\n", static_cast<void*>(state)); PrintF("Dumping virtual state %p\n", static_cast<void*>(state));
for (Alias alias = 0; alias < AliasCount(); ++alias) { for (Alias alias = 0; alias < AliasCount(); ++alias) {
if (VirtualObject* object = state->VirtualObjectFromAlias(alias)) { if (VirtualObject* object = state->VirtualObjectFromAlias(alias)) {
DebugPrintObject(object, alias); DebugPrintObject(object, alias);
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment