[Reduce] Rewrite function body delta pass again

It is not enough to replace all uses of users of the function with undef,
the users, we only drop instruction users, so they may stick around.

Let's try different approach - first drop bodies for all the functions
we will drop, which should take care of blockaddress issue the previous
rewrite was dealing with; then, after dropping *all* such bodies,
replace remaining uses with undef (thus all the uses are either
outside of functions, or are in kept functions)
and then finally drop functions.

This seems to work, and passes the *existing* test coverage,
but it is possible that a new issue will be discovered later :)

A new (previously crashing) test added.
This commit is contained in:
Roman Lebedev 2020-08-08 23:21:08 +03:00
parent 48887c4e81
commit d4c3f20285
No known key found for this signature in database
GPG Key ID: 083C3EBB4A1689E0
2 changed files with 33 additions and 25 deletions

View File

@ -0,0 +1,14 @@
; RUN: llvm-reduce --test FileCheck --test-arg --check-prefixes=CHECK-ALL,CHECK-INTERESTINGNESS --test-arg %s --test-arg --input-file %s -o %t
; RUN: cat %t | FileCheck --check-prefixes=CHECK-ALL,CHECK-FINAL %s
; CHECK-INTERESTINGNESS: @alias =
; CHECK-FINAL: @alias = alias void (i32), void (i32)* undef
@alias = alias void (i32), void (i32)* @func
; CHECK-FINAL-NOT: @func()
define void @func(i32 %arg) {
entry:
ret void
}

View File

@ -14,42 +14,36 @@
#include "ReduceFunctions.h" #include "ReduceFunctions.h"
#include "Delta.h" #include "Delta.h"
#include "llvm/ADT/SetVector.h" #include "llvm/ADT/STLExtras.h"
#include "llvm/IR/Instructions.h" #include "llvm/IR/Instructions.h"
#include <set> #include <iterator>
#include <vector>
using namespace llvm; using namespace llvm;
/// Removes all the Defined Functions (as well as their calls) /// Removes all the Defined Functions
/// that aren't inside any of the desired Chunks. /// that aren't inside any of the desired Chunks.
static void extractFunctionsFromModule(const std::vector<Chunk> &ChunksToKeep, static void extractFunctionsFromModule(const std::vector<Chunk> &ChunksToKeep,
Module *Program) { Module *Program) {
Oracle O(ChunksToKeep); Oracle O(ChunksToKeep);
// Get functions inside desired chunks // Record all out-of-chunk functions.
std::set<Function *> FuncsToKeep; std::vector<std::reference_wrapper<Function>> FuncsToRemove;
for (auto &F : *Program) copy_if(Program->functions(), std::back_inserter(FuncsToRemove),
if (O.shouldKeep()) [&O](auto &unused) { return !O.shouldKeep(); });
FuncsToKeep.insert(&F);
// Delete out-of-chunk functions, and replace their users with undef // Then, drop body of each of them. We want to batch this and do nothing else
std::vector<Function *> FuncsToRemove; // here so that minimal number of remaining exteranal uses will remain.
SetVector<Instruction *> InstrsToRemove; for (Function &F : FuncsToRemove)
for (auto &F : *Program) F.dropAllReferences();
if (!FuncsToKeep.count(&F)) {
for (auto U : F.users()) {
U->replaceAllUsesWith(UndefValue::get(U->getType()));
if (auto *I = dyn_cast<Instruction>(U))
InstrsToRemove.insert(I);
}
FuncsToRemove.push_back(&F);
}
for (auto *I : InstrsToRemove) // And finally, we can actually delete them.
I->eraseFromParent(); for (Function &F : FuncsToRemove) {
// Replace all *still* remaining uses with undef.
for (auto *F : FuncsToRemove) F.replaceAllUsesWith(UndefValue::get(F.getType()));
F->eraseFromParent(); // And finally, fully drop it.
F.eraseFromParent();
}
} }
/// Counts the amount of non-declaration functions and prints their /// Counts the amount of non-declaration functions and prints their