diff options
Diffstat (limited to 'llvm/lib/CodeGen')
| -rw-r--r-- | llvm/lib/CodeGen/SelectionDAG/SelectionDAGBuilder.cpp | 71 | ||||
| -rw-r--r-- | llvm/lib/CodeGen/SelectionDAG/SelectionDAGBuilder.h | 6 | 
2 files changed, 65 insertions, 12 deletions
| diff --git a/llvm/lib/CodeGen/SelectionDAG/SelectionDAGBuilder.cpp b/llvm/lib/CodeGen/SelectionDAG/SelectionDAGBuilder.cpp index a561c5190ac..c177038102d 100644 --- a/llvm/lib/CodeGen/SelectionDAG/SelectionDAGBuilder.cpp +++ b/llvm/lib/CodeGen/SelectionDAG/SelectionDAGBuilder.cpp @@ -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      // +    // 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); +    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. +    // 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); +    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. diff --git a/llvm/lib/CodeGen/SelectionDAG/SelectionDAGBuilder.h b/llvm/lib/CodeGen/SelectionDAG/SelectionDAGBuilder.h index d32d13f343d..a8c6ab77c7c 100644 --- a/llvm/lib/CodeGen/SelectionDAG/SelectionDAGBuilder.h +++ b/llvm/lib/CodeGen/SelectionDAG/SelectionDAGBuilder.h @@ -612,11 +612,13 @@ public:    void FindMergedConditions(const Value *Cond, MachineBasicBlock *TBB,                              MachineBasicBlock *FBB, MachineBasicBlock *CurBB, -                            MachineBasicBlock *SwitchBB, unsigned Opc); +                            MachineBasicBlock *SwitchBB, unsigned Opc, +                            uint32_t TW, uint32_t FW);    void EmitBranchForMergedCondition(const Value *Cond, MachineBasicBlock *TBB,                                      MachineBasicBlock *FBB,                                      MachineBasicBlock *CurBB, -                                    MachineBasicBlock *SwitchBB); +                                    MachineBasicBlock *SwitchBB, +                                    uint32_t TW, uint32_t FW);    bool ShouldEmitAsBranches(const std::vector<CaseBlock> &Cases);    bool isExportableFromCurrentBlock(const Value *V, const BasicBlock *FromBB);    void CopyToExportRegsIfNeeded(const Value *V); | 

