earlier I fixed a bug where the BB removal pass sometimes created

invalid IR. the fix was incomplete, this one is better and is believed
to be complete

Differential Revision: https://reviews.llvm.org/D131132
This commit is contained in:
John Regehr 2022-08-04 10:21:20 -06:00
parent b06da9c183
commit 213c21fe10
2 changed files with 60 additions and 19 deletions

View File

@ -0,0 +1,28 @@
; RUN: llvm-reduce --delta-passes=basic-blocks --test %python --test-arg %p/Inputs/remove-bbs.py -abort-on-invalid-reduction %s -o %t
declare void @0()
define internal void @1(ptr %x0, i1 %x1) {
interesting0:
%x3 = alloca i32, align 4
store ptr null, ptr %x0, align 8
br label %interesting1
interesting1: ; preds = %x2
call void @0()
br label %x2
x2:
br i1 %x1, label %interesting3, label %interesting4
interesting3:
call void @0()
br label %x2
interesting4:
br label %x5
x5:
store i32 0, ptr %x3, align 4
ret void
}

View File

@ -102,35 +102,48 @@ removeUninterestingBBsFromSwitch(SwitchInst &SwInst,
} }
} }
/// A BB is ok to remove if it's not the entry block, or else it is /// It's OK to add a block to the set of removed blocks if the first
/// the entry block but the next block in the function has just one /// basic block in the function that survives all of the deletions is
/// predecessor -- this property is required because that block is /// a legal entry block
/// going to become the new entry block static bool okToRemove(BasicBlock &Candidate, Function &F,
static bool okToRemove(BasicBlock &BB) { const DenseSet<BasicBlock *> &BBsToDelete) {
if (!BB.isEntryBlock()) for (auto &B : F) {
if (&B == &Candidate)
continue;
if (BBsToDelete.count(&B))
continue;
/// Ok we've found the first block that's not going to be deleted,
/// it will be the new entry block -- that's only legal if this
/// block has no predecessors among blocks that survive the
/// deletions
for (BasicBlock *Pred : predecessors(&B)) {
if (!BBsToDelete.contains(Pred))
return false;
}
return true; return true;
auto F = BB.getParent(); }
auto it = F->begin(); return true;
++it;
return (it == F->end()) || (*it).hasNPredecessors(1);
} }
/// Removes out-of-chunk arguments from functions, and modifies their calls /// Removes out-of-chunk arguments from functions, and modifies their calls
/// accordingly. It also removes allocations of out-of-chunk arguments. /// accordingly. It also removes allocations of out-of-chunk arguments.
static void extractBasicBlocksFromModule(Oracle &O, Module &Program) { static void extractBasicBlocksFromModule(Oracle &O, Module &Program) {
DenseSet<BasicBlock *> BBsToKeep; DenseSet<BasicBlock *> BBsToKeep, BBsToDelete;
SmallVector<BasicBlock *> BBsToDelete;
for (auto &F : Program) { for (auto &F : Program) {
for (auto &BB : F) { for (auto &BB : F) {
if (!okToRemove(BB) || O.shouldKeep()) if (!okToRemove(BB, F, BBsToDelete) || O.shouldKeep())
BBsToKeep.insert(&BB); BBsToKeep.insert(&BB);
else { else
BBsToDelete.push_back(&BB); BBsToDelete.insert(&BB);
// Remove out-of-chunk BB from successor phi nodes }
for (auto *Succ : successors(&BB)) }
Succ->removePredecessor(&BB);
} // Remove out-of-chunk BB from successor phi nodes
for (auto &F : Program) {
for (auto &BB : F) {
for (auto *Succ : successors(&BB))
Succ->removePredecessor(&BB);
} }
} }