mirror of
https://github.com/c64scene-ar/llvm-6502.git
synced 2026-04-23 22:23:00 +00:00
PGO branch weight: update edge weights in SelectionDAGBuilder.
When converting from "or + br" to two branches, or converting from "and + br" to two branches, we correctly update the edge weights of the two branches. The previous attempt at r200431 was reverted at r200434 because of two testing case failures. I modified my patch a little, but forgot to re-run "make check-all". Testing case CodeGen/ARM/lsr-unfolded-offset.ll is updated because of the patch's impact on branch probability which causes changes in spill placement. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@200502 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
@@ -1393,7 +1393,9 @@ SelectionDAGBuilder::EmitBranchForMergedCondition(const Value *Cond,
|
||||
MachineBasicBlock *TBB,
|
||||
MachineBasicBlock *FBB,
|
||||
MachineBasicBlock *CurBB,
|
||||
MachineBasicBlock *SwitchBB) {
|
||||
MachineBasicBlock *SwitchBB,
|
||||
uint32_t TWeight,
|
||||
uint32_t FWeight) {
|
||||
const BasicBlock *BB = CurBB->getBasicBlock();
|
||||
|
||||
// If the leaf of the tree is a comparison, merge the condition into
|
||||
@@ -1418,7 +1420,7 @@ SelectionDAGBuilder::EmitBranchForMergedCondition(const Value *Cond,
|
||||
}
|
||||
|
||||
CaseBlock CB(Condition, BOp->getOperand(0),
|
||||
BOp->getOperand(1), NULL, TBB, FBB, CurBB);
|
||||
BOp->getOperand(1), NULL, TBB, FBB, CurBB, TWeight, FWeight);
|
||||
SwitchCases.push_back(CB);
|
||||
return;
|
||||
}
|
||||
@@ -1426,17 +1428,26 @@ SelectionDAGBuilder::EmitBranchForMergedCondition(const Value *Cond,
|
||||
|
||||
// Create a CaseBlock record representing this branch.
|
||||
CaseBlock CB(ISD::SETEQ, Cond, ConstantInt::getTrue(*DAG.getContext()),
|
||||
NULL, TBB, FBB, CurBB);
|
||||
NULL, TBB, FBB, CurBB, TWeight, FWeight);
|
||||
SwitchCases.push_back(CB);
|
||||
}
|
||||
|
||||
/// Scale down both weights to fit into uint32_t.
|
||||
static void ScaleWeights(uint64_t &NewTrue, uint64_t &NewFalse) {
|
||||
uint64_t NewMax = (NewTrue > NewFalse) ? NewTrue : NewFalse;
|
||||
uint32_t Scale = (NewMax / UINT32_MAX) + 1;
|
||||
NewTrue = NewTrue / Scale;
|
||||
NewFalse = NewFalse / Scale;
|
||||
}
|
||||
|
||||
/// FindMergedConditions - If Cond is an expression like
|
||||
void SelectionDAGBuilder::FindMergedConditions(const Value *Cond,
|
||||
MachineBasicBlock *TBB,
|
||||
MachineBasicBlock *FBB,
|
||||
MachineBasicBlock *CurBB,
|
||||
MachineBasicBlock *SwitchBB,
|
||||
unsigned Opc) {
|
||||
unsigned Opc, uint32_t TWeight,
|
||||
uint32_t FWeight) {
|
||||
// If this node is not part of the or/and tree, emit it as a branch.
|
||||
const Instruction *BOp = dyn_cast<Instruction>(Cond);
|
||||
if (!BOp || !(isa<BinaryOperator>(BOp) || isa<CmpInst>(BOp)) ||
|
||||
@@ -1444,7 +1455,8 @@ void SelectionDAGBuilder::FindMergedConditions(const Value *Cond,
|
||||
BOp->getParent() != CurBB->getBasicBlock() ||
|
||||
!InBlock(BOp->getOperand(0), CurBB->getBasicBlock()) ||
|
||||
!InBlock(BOp->getOperand(1), CurBB->getBasicBlock())) {
|
||||
EmitBranchForMergedCondition(Cond, TBB, FBB, CurBB, SwitchBB);
|
||||
EmitBranchForMergedCondition(Cond, TBB, FBB, CurBB, SwitchBB,
|
||||
TWeight, FWeight);
|
||||
return;
|
||||
}
|
||||
|
||||
@@ -1456,6 +1468,7 @@ void SelectionDAGBuilder::FindMergedConditions(const Value *Cond,
|
||||
|
||||
if (Opc == Instruction::Or) {
|
||||
// Codegen X | Y as:
|
||||
// BB1:
|
||||
// jmp_if_X TBB
|
||||
// jmp TmpBB
|
||||
// TmpBB:
|
||||
@@ -1463,14 +1476,34 @@ void SelectionDAGBuilder::FindMergedConditions(const Value *Cond,
|
||||
// jmp FBB
|
||||
//
|
||||
|
||||
// Emit the LHS condition.
|
||||
FindMergedConditions(BOp->getOperand(0), TBB, TmpBB, CurBB, SwitchBB, Opc);
|
||||
// We have flexibility in setting Prob for BB1 and Prob for TmpBB.
|
||||
// The requirement is that
|
||||
// TrueProb for BB1 + (FalseProb for BB1 * TrueProb for TmpBB)
|
||||
// = TrueProb for orignal BB.
|
||||
// Assuming the orignal weights are A and B, one choice is to set BB1's
|
||||
// weights to A and A+2B, and set TmpBB's weights to A and 2B. This choice
|
||||
// assumes that
|
||||
// TrueProb for BB1 == FalseProb for BB1 * TrueProb for TmpBB.
|
||||
// Another choice is to assume TrueProb for BB1 equals to TrueProb for
|
||||
// TmpBB, but the math is more complicated.
|
||||
|
||||
uint64_t NewTrueWeight = TWeight;
|
||||
uint64_t NewFalseWeight = (uint64_t)TWeight + 2 * (uint64_t)FWeight;
|
||||
ScaleWeights(NewTrueWeight, NewFalseWeight);
|
||||
// Emit the LHS condition.
|
||||
FindMergedConditions(BOp->getOperand(0), TBB, TmpBB, CurBB, SwitchBB, Opc,
|
||||
NewTrueWeight, NewFalseWeight);
|
||||
|
||||
NewTrueWeight = TWeight;
|
||||
NewFalseWeight = 2 * (uint64_t)FWeight;
|
||||
ScaleWeights(NewTrueWeight, NewFalseWeight);
|
||||
// Emit the RHS condition into TmpBB.
|
||||
FindMergedConditions(BOp->getOperand(1), TBB, FBB, TmpBB, SwitchBB, Opc);
|
||||
FindMergedConditions(BOp->getOperand(1), TBB, FBB, TmpBB, SwitchBB, Opc,
|
||||
NewTrueWeight, NewFalseWeight);
|
||||
} else {
|
||||
assert(Opc == Instruction::And && "Unknown merge op!");
|
||||
// Codegen X & Y as:
|
||||
// BB1:
|
||||
// jmp_if_X TmpBB
|
||||
// jmp FBB
|
||||
// TmpBB:
|
||||
@@ -1479,11 +1512,28 @@ void SelectionDAGBuilder::FindMergedConditions(const Value *Cond,
|
||||
//
|
||||
// This requires creation of TmpBB after CurBB.
|
||||
|
||||
// Emit the LHS condition.
|
||||
FindMergedConditions(BOp->getOperand(0), TmpBB, FBB, CurBB, SwitchBB, Opc);
|
||||
// We have flexibility in setting Prob for BB1 and Prob for TmpBB.
|
||||
// The requirement is that
|
||||
// FalseProb for BB1 + (TrueProb for BB1 * FalseProb for TmpBB)
|
||||
// = FalseProb for orignal BB.
|
||||
// Assuming the orignal weights are A and B, one choice is to set BB1's
|
||||
// weights to 2A+B and B, and set TmpBB's weights to 2A and B. This choice
|
||||
// assumes that
|
||||
// FalseProb for BB1 == TrueProb for BB1 * FalseProb for TmpBB.
|
||||
|
||||
uint64_t NewTrueWeight = 2 * (uint64_t)TWeight + (uint64_t)FWeight;
|
||||
uint64_t NewFalseWeight = FWeight;
|
||||
ScaleWeights(NewTrueWeight, NewFalseWeight);
|
||||
// Emit the LHS condition.
|
||||
FindMergedConditions(BOp->getOperand(0), TmpBB, FBB, CurBB, SwitchBB, Opc,
|
||||
NewTrueWeight, NewFalseWeight);
|
||||
|
||||
NewTrueWeight = 2 * (uint64_t)TWeight;
|
||||
NewFalseWeight = FWeight;
|
||||
ScaleWeights(NewTrueWeight, NewFalseWeight);
|
||||
// Emit the RHS condition into TmpBB.
|
||||
FindMergedConditions(BOp->getOperand(1), TBB, FBB, TmpBB, SwitchBB, Opc);
|
||||
FindMergedConditions(BOp->getOperand(1), TBB, FBB, TmpBB, SwitchBB, Opc,
|
||||
NewTrueWeight, NewFalseWeight);
|
||||
}
|
||||
}
|
||||
|
||||
@@ -1570,7 +1620,8 @@ void SelectionDAGBuilder::visitBr(const BranchInst &I) {
|
||||
(BOp->getOpcode() == Instruction::And ||
|
||||
BOp->getOpcode() == Instruction::Or)) {
|
||||
FindMergedConditions(BOp, Succ0MBB, Succ1MBB, BrMBB, BrMBB,
|
||||
BOp->getOpcode());
|
||||
BOp->getOpcode(), getEdgeWeight(BrMBB, Succ0MBB),
|
||||
getEdgeWeight(BrMBB, Succ1MBB));
|
||||
// If the compares in later blocks need to use values not currently
|
||||
// exported from this block, export them now. This block should always
|
||||
// be the first entry.
|
||||
|
||||
Reference in New Issue
Block a user