[MergeICmps] Collect block instructions once (NFC)

Collect the relevant instructions for a given BCECmpBlock once
on construction, rather than repeating this logic in multiple
places.
This commit is contained in:
Nikita Popov 2021-07-26 18:05:47 +02:00
parent 792c206e2b
commit f921bf6049
1 changed files with 30 additions and 37 deletions

View File

@ -201,8 +201,10 @@ struct BCECmp {
// (see canSplit()). // (see canSplit()).
class BCECmpBlock { class BCECmpBlock {
public: public:
BCECmpBlock(BCECmp Cmp, BasicBlock *BB, BranchInst *BranchI) typedef SmallDenseSet<const Instruction *, 8> InstructionSet;
: BB(BB), BranchI(BranchI), Cmp(std::move(Cmp)) {}
BCECmpBlock(BCECmp Cmp, BasicBlock *BB, InstructionSet BlockInsts)
: BB(BB), BlockInsts(std::move(BlockInsts)), Cmp(std::move(Cmp)) {}
const BCEAtom &Lhs() const { return Cmp.Lhs; } const BCEAtom &Lhs() const { return Cmp.Lhs; }
const BCEAtom &Rhs() const { return Cmp.Rhs; } const BCEAtom &Rhs() const { return Cmp.Rhs; }
@ -219,8 +221,7 @@ class BCECmpBlock {
// be sunk below this instruction. By doing this, we know we can separate the // be sunk below this instruction. By doing this, we know we can separate the
// BCE-cmp-block instructions from the non-BCE-cmp-block instructions in the // BCE-cmp-block instructions from the non-BCE-cmp-block instructions in the
// block. // block.
bool canSinkBCECmpInst(const Instruction *, DenseSet<const Instruction *> &, bool canSinkBCECmpInst(const Instruction *, AliasAnalysis &AA) const;
AliasAnalysis &AA) const;
// We can separate the BCE-cmp-block instructions and the non-BCE-cmp-block // We can separate the BCE-cmp-block instructions and the non-BCE-cmp-block
// instructions. Split the old block and move all non-BCE-cmp-insts into the // instructions. Split the old block and move all non-BCE-cmp-insts into the
@ -229,8 +230,8 @@ class BCECmpBlock {
// The basic block where this comparison happens. // The basic block where this comparison happens.
BasicBlock *BB; BasicBlock *BB;
// The terminating branch. // Instructions relating to the BCECmp and branch.
BranchInst *BranchI; InstructionSet BlockInsts;
// The block requires splitting. // The block requires splitting.
bool RequireSplit = false; bool RequireSplit = false;
@ -239,7 +240,6 @@ private:
}; };
bool BCECmpBlock::canSinkBCECmpInst(const Instruction *Inst, bool BCECmpBlock::canSinkBCECmpInst(const Instruction *Inst,
DenseSet<const Instruction *> &BlockInsts,
AliasAnalysis &AA) const { AliasAnalysis &AA) const {
// If this instruction may clobber the loads and is in middle of the BCE cmp // If this instruction may clobber the loads and is in middle of the BCE cmp
// block instructions, then bail for now. // block instructions, then bail for now.
@ -263,15 +263,11 @@ bool BCECmpBlock::canSinkBCECmpInst(const Instruction *Inst,
} }
void BCECmpBlock::split(BasicBlock *NewParent, AliasAnalysis &AA) const { void BCECmpBlock::split(BasicBlock *NewParent, AliasAnalysis &AA) const {
DenseSet<const Instruction *> BlockInsts(
{Cmp.Lhs.GEP, Cmp.Rhs.GEP, Cmp.Lhs.LoadI, Cmp.Rhs.LoadI, Cmp.CmpI,
BranchI});
llvm::SmallVector<Instruction *, 4> OtherInsts; llvm::SmallVector<Instruction *, 4> OtherInsts;
for (Instruction &Inst : *BB) { for (Instruction &Inst : *BB) {
if (BlockInsts.count(&Inst)) if (BlockInsts.count(&Inst))
continue; continue;
assert(canSinkBCECmpInst(&Inst, BlockInsts, AA) && assert(canSinkBCECmpInst(&Inst, AA) && "Split unsplittable block");
"Split unsplittable block");
// This is a non-BCE-cmp-block instruction. And it can be separated // This is a non-BCE-cmp-block instruction. And it can be separated
// from the BCE-cmp-block instruction. // from the BCE-cmp-block instruction.
OtherInsts.push_back(&Inst); OtherInsts.push_back(&Inst);
@ -284,12 +280,9 @@ void BCECmpBlock::split(BasicBlock *NewParent, AliasAnalysis &AA) const {
} }
bool BCECmpBlock::canSplit(AliasAnalysis &AA) const { bool BCECmpBlock::canSplit(AliasAnalysis &AA) const {
DenseSet<const Instruction *> BlockInsts(
{Cmp.Lhs.GEP, Cmp.Rhs.GEP, Cmp.Lhs.LoadI, Cmp.Rhs.LoadI, Cmp.CmpI,
BranchI});
for (Instruction &Inst : *BB) { for (Instruction &Inst : *BB) {
if (!BlockInsts.count(&Inst)) { if (!BlockInsts.count(&Inst)) {
if (!canSinkBCECmpInst(&Inst, BlockInsts, AA)) if (!canSinkBCECmpInst(&Inst, AA))
return false; return false;
} }
} }
@ -297,10 +290,6 @@ bool BCECmpBlock::canSplit(AliasAnalysis &AA) const {
} }
bool BCECmpBlock::doesOtherWork() const { bool BCECmpBlock::doesOtherWork() const {
// All the instructions we care about in the BCE cmp block.
DenseSet<const Instruction *> BlockInsts(
{Cmp.Lhs.GEP, Cmp.Rhs.GEP, Cmp.Lhs.LoadI, Cmp.Rhs.LoadI, Cmp.CmpI,
BranchI});
// TODO(courbet): Can we allow some other things ? This is very conservative. // TODO(courbet): Can we allow some other things ? This is very conservative.
// We might be able to get away with anything does not have any side // We might be able to get away with anything does not have any side
// effects outside of the basic block. // effects outside of the basic block.
@ -351,37 +340,41 @@ Optional<BCECmpBlock> visitCmpBlock(Value *const Val, BasicBlock *const Block,
auto *const BranchI = dyn_cast<BranchInst>(Block->getTerminator()); auto *const BranchI = dyn_cast<BranchInst>(Block->getTerminator());
if (!BranchI) return None; if (!BranchI) return None;
LLVM_DEBUG(dbgs() << "branch\n"); LLVM_DEBUG(dbgs() << "branch\n");
Value *Cond;
ICmpInst::Predicate ExpectedPredicate;
if (BranchI->isUnconditional()) { if (BranchI->isUnconditional()) {
// In this case, we expect an incoming value which is the result of the // In this case, we expect an incoming value which is the result of the
// comparison. This is the last link in the chain of comparisons (note // comparison. This is the last link in the chain of comparisons (note
// that this does not mean that this is the last incoming value, blocks // that this does not mean that this is the last incoming value, blocks
// can be reordered). // can be reordered).
auto *const CmpI = dyn_cast<ICmpInst>(Val); Cond = Val;
if (!CmpI) return None; ExpectedPredicate = ICmpInst::ICMP_EQ;
LLVM_DEBUG(dbgs() << "icmp\n");
Optional<BCECmp> Result = visitICmp(CmpI, ICmpInst::ICMP_EQ, BaseId);
if (!Result)
return None;
return BCECmpBlock(std::move(*Result), Block, BranchI);
} else { } else {
// In this case, we expect a constant incoming value (the comparison is // In this case, we expect a constant incoming value (the comparison is
// chained). // chained).
const auto *const Const = cast<ConstantInt>(Val); const auto *const Const = cast<ConstantInt>(Val);
LLVM_DEBUG(dbgs() << "const\n"); LLVM_DEBUG(dbgs() << "const\n");
if (!Const->isZero()) return {}; if (!Const->isZero()) return None;
LLVM_DEBUG(dbgs() << "false\n"); LLVM_DEBUG(dbgs() << "false\n");
auto *const CmpI = dyn_cast<ICmpInst>(BranchI->getCondition());
if (!CmpI) return {};
LLVM_DEBUG(dbgs() << "icmp\n");
assert(BranchI->getNumSuccessors() == 2 && "expecting a cond branch"); assert(BranchI->getNumSuccessors() == 2 && "expecting a cond branch");
BasicBlock *const FalseBlock = BranchI->getSuccessor(1); BasicBlock *const FalseBlock = BranchI->getSuccessor(1);
Optional<BCECmp> Result = visitICmp( Cond = BranchI->getCondition();
CmpI, FalseBlock == PhiBlock ? ICmpInst::ICMP_EQ : ICmpInst::ICMP_NE, ExpectedPredicate =
BaseId); FalseBlock == PhiBlock ? ICmpInst::ICMP_EQ : ICmpInst::ICMP_NE;
if (!Result)
return None;
return BCECmpBlock(std::move(*Result), Block, BranchI);
} }
auto *CmpI = dyn_cast<ICmpInst>(Cond);
if (!CmpI) return None;
LLVM_DEBUG(dbgs() << "icmp\n");
Optional<BCECmp> Result = visitICmp(CmpI, ExpectedPredicate, BaseId);
if (!Result)
return None;
BCECmpBlock::InstructionSet BlockInsts(
{Result->Lhs.GEP, Result->Rhs.GEP, Result->Lhs.LoadI, Result->Rhs.LoadI,
Result->CmpI, BranchI});
return BCECmpBlock(std::move(*Result), Block, BlockInsts);
} }
static inline void enqueueBlock(std::vector<BCECmpBlock> &Comparisons, static inline void enqueueBlock(std::vector<BCECmpBlock> &Comparisons,