Commit 767b152a authored by Stephan Herhut's avatar Stephan Herhut Committed by Commit Bot

Revert "[regalloc] More aggressively reuse spill ranges for phi inputs"

Turns out TryReuseSpillForPhi does more than reusing a spill slot for a
phi (which is beneficial in general). It also decides whether a phi
should start out to be allocated in a spill slot. The latter, of course,
benefits from control flow knowledge. Hence, this change was detrimental
in cases where a common input to a phi is only spilled on few control
flow pathes.

To fix, we need to disentangle spill-slot reuse and the decision whether
a phi should start out spilled. I will look into this in a follow up
change.

This reverts commit b79cbd56.

Change-Id: I228185bb1a4b320d3115ba7f1d921593480d8e7d
Reviewed-on: https://chromium-review.googlesource.com/c/1304549Reviewed-by: 's avatarBenedikt Meurer <bmeurer@chromium.org>
Commit-Queue: Stephan Herhut <herhut@chromium.org>
Cr-Commit-Position: refs/heads/master@{#57113}
parent b589315d
...@@ -3355,6 +3355,7 @@ bool LinearScanAllocator::TryReuseSpillForPhi(TopLevelLiveRange* range) { ...@@ -3355,6 +3355,7 @@ bool LinearScanAllocator::TryReuseSpillForPhi(TopLevelLiveRange* range) {
RegisterAllocationData::PhiMapValue* phi_map_value = RegisterAllocationData::PhiMapValue* phi_map_value =
data()->GetPhiMapValueFor(range); data()->GetPhiMapValueFor(range);
const PhiInstruction* phi = phi_map_value->phi(); const PhiInstruction* phi = phi_map_value->phi();
const InstructionBlock* block = phi_map_value->block();
// Count the number of spilled operands. // Count the number of spilled operands.
size_t spilled_count = 0; size_t spilled_count = 0;
LiveRange* first_op = nullptr; LiveRange* first_op = nullptr;
...@@ -3362,9 +3363,15 @@ bool LinearScanAllocator::TryReuseSpillForPhi(TopLevelLiveRange* range) { ...@@ -3362,9 +3363,15 @@ bool LinearScanAllocator::TryReuseSpillForPhi(TopLevelLiveRange* range) {
int op = phi->operands()[i]; int op = phi->operands()[i];
LiveRange* op_range = data()->GetOrCreateLiveRangeFor(op); LiveRange* op_range = data()->GetOrCreateLiveRangeFor(op);
if (!op_range->TopLevel()->HasSpillRange()) continue; if (!op_range->TopLevel()->HasSpillRange()) continue;
if (!op_range->TopLevel()->IsSpilledOnlyInDeferredBlocks()) { const InstructionBlock* pred =
// This value was spilled at definition, so it is available in a spillslot code()->InstructionBlockAt(block->predecessors()[i]);
// regardless of control flow. LifetimePosition pred_end =
LifetimePosition::InstructionFromInstructionIndex(
pred->last_instruction_index());
while (op_range != nullptr && !op_range->CanCover(pred_end)) {
op_range = op_range->next();
}
if (op_range != nullptr && op_range->spilled()) {
spilled_count++; spilled_count++;
if (first_op == nullptr) { if (first_op == nullptr) {
first_op = op_range->TopLevel(); first_op = op_range->TopLevel();
......
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