Commit 2e360cd8 authored by jacob.bramley's avatar jacob.bramley Committed by Commit bot

Improve phi hinting heuristics.

The basic intention is to try to remove unnecessary moves caused by
hints in otherwise empty blocks. Roughly:

Before                          After
-----------------------------------------------------------
B0: add   x1, ...               B0: add   x1, ...
    b.ne  B2                        b.eq  B3
B1: mov   x0, x1                B1: [empty]
    b     B3
B2: add   x0, x1, ...           B2: add   x1, x1, ...
B3: phi(B1,B2) in x0            B3: phi(B0,B1) in x1

Hinting is also improved in cases where one of the inputs is already
allocated. This occurs commonly on architectures with instructions which
write into fixed registers, for example.

BUG=

Review-Url: https://chromiumcodereview.appspot.com/2125463002
Cr-Commit-Position: refs/heads/master@{#40498}
parent 14be3846
...@@ -2164,33 +2164,112 @@ void LiveRangeBuilder::ProcessPhis(const InstructionBlock* block, ...@@ -2164,33 +2164,112 @@ void LiveRangeBuilder::ProcessPhis(const InstructionBlock* block,
// block. // block.
int phi_vreg = phi->virtual_register(); int phi_vreg = phi->virtual_register();
live->Remove(phi_vreg); live->Remove(phi_vreg);
// Select the hint from the first predecessor block that preceeds this block // Select a hint from a predecessor block that preceeds this block in the
// in the rpo ordering. Prefer non-deferred blocks. The enforcement of // rpo order. In order of priority:
// hinting in rpo order is required because hint resolution that happens // - Avoid hints from deferred blocks.
// later in the compiler pipeline visits instructions in reverse rpo, // - Prefer hints from allocated (or explicit) operands.
// relying on the fact that phis are encountered before their hints. // - Prefer hints from empty blocks (containing just parallel moves and a
const Instruction* instr = nullptr; // jump). In these cases, if we can elide the moves, the jump threader
const InstructionBlock::Predecessors& predecessors = block->predecessors(); // is likely to be able to elide the jump.
for (size_t i = 0; i < predecessors.size(); ++i) { // The enforcement of hinting in rpo order is required because hint
// resolution that happens later in the compiler pipeline visits
// instructions in reverse rpo order, relying on the fact that phis are
// encountered before their hints.
InstructionOperand* hint = nullptr;
int hint_preference = 0;
// The cost of hinting increases with the number of predecessors. At the
// same time, the typical benefit decreases, since this hinting only
// optimises the execution path through one predecessor. A limit of 2 is
// sufficient to hit the common if/else pattern.
int predecessor_limit = 2;
for (RpoNumber predecessor : block->predecessors()) {
const InstructionBlock* predecessor_block = const InstructionBlock* predecessor_block =
code()->InstructionBlockAt(predecessors[i]); code()->InstructionBlockAt(predecessor);
if (predecessor_block->rpo_number() < block->rpo_number()) { DCHECK_EQ(predecessor_block->rpo_number(), predecessor);
instr = GetLastInstruction(code(), predecessor_block);
if (!predecessor_block->IsDeferred()) break; // Only take hints from earlier rpo numbers.
if (predecessor >= block->rpo_number()) continue;
// Look up the predecessor instruction.
const Instruction* predecessor_instr =
GetLastInstruction(code(), predecessor_block);
InstructionOperand* predecessor_hint = nullptr;
// Phis are assigned in the END position of the last instruction in each
// predecessor block.
for (MoveOperands* move :
*predecessor_instr->GetParallelMove(Instruction::END)) {
InstructionOperand& to = move->destination();
if (to.IsUnallocated() &&
UnallocatedOperand::cast(to).virtual_register() == phi_vreg) {
predecessor_hint = &move->source();
break;
}
}
DCHECK_NOT_NULL(predecessor_hint);
// For each predecessor, generate a score according to the priorities
// described above, and pick the best one. Flags in higher-order bits have
// a higher priority than those in lower-order bits.
int predecessor_hint_preference = 0;
const int kNotDeferredBlockPreference = (1 << 2);
const int kMoveIsAllocatedPreference = (1 << 1);
const int kBlockIsEmptyPreference = (1 << 0);
// - Avoid hints from deferred blocks.
if (!predecessor_block->IsDeferred()) {
predecessor_hint_preference |= kNotDeferredBlockPreference;
} }
}
DCHECK_NOT_NULL(instr);
InstructionOperand* hint = nullptr; // - Prefer hints from allocated (or explicit) operands.
for (MoveOperands* move : *instr->GetParallelMove(Instruction::END)) { //
InstructionOperand& to = move->destination(); // Already-allocated or explicit operands are typically assigned using
if (to.IsUnallocated() && // the parallel moves on the last instruction. For example:
UnallocatedOperand::cast(to).virtual_register() == phi_vreg) { //
hint = &move->source(); // gap (v101 = [x0|R|w32]) (v100 = v101)
break; // ArchJmp
// ...
// phi: v100 = v101 v102
//
// We have already found the END move, so look for a matching START move
// from an allocated (or explicit) operand.
//
// Note that we cannot simply look up data()->live_ranges()[vreg] here
// because the live ranges are still being built when this function is
// called.
// TODO(v8): Find a way to separate hinting from live range analysis in
// BuildLiveRanges so that we can use the O(1) live-range look-up.
auto moves = predecessor_instr->GetParallelMove(Instruction::START);
if (moves != nullptr) {
for (MoveOperands* move : *moves) {
InstructionOperand& to = move->destination();
if (predecessor_hint->Equals(to)) {
if (move->source().IsAllocated() || move->source().IsExplicit()) {
predecessor_hint_preference |= kMoveIsAllocatedPreference;
}
break;
}
}
}
// - Prefer hints from empty blocks.
if (predecessor_block->last_instruction_index() ==
predecessor_block->first_instruction_index()) {
predecessor_hint_preference |= kBlockIsEmptyPreference;
} }
if ((hint == nullptr) ||
(predecessor_hint_preference > hint_preference)) {
// Take the hint from this predecessor.
hint = predecessor_hint;
hint_preference = predecessor_hint_preference;
}
if (--predecessor_limit <= 0) break;
} }
DCHECK(hint != nullptr); DCHECK_NOT_NULL(hint);
LifetimePosition block_start = LifetimePosition::GapFromInstructionIndex( LifetimePosition block_start = LifetimePosition::GapFromInstructionIndex(
block->first_instruction_index()); block->first_instruction_index());
UsePosition* use_pos = Define(block_start, &phi->output(), hint, UsePosition* use_pos = Define(block_start, &phi->output(), hint,
......
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