#include "llvm/Transforms/Utils/CanonicalizeFreezeInLoops.h"
#include "llvm/ADT/STLExtras.h"
#include "llvm/ADT/SmallVector.h"
#include "llvm/Analysis/IVDescriptors.h"
#include "llvm/Analysis/LoopAnalysisManager.h"
#include "llvm/Analysis/LoopInfo.h"
#include "llvm/Analysis/LoopPass.h"
#include "llvm/Analysis/ScalarEvolution.h"
#include "llvm/Analysis/ValueTracking.h"
#include "llvm/IR/Dominators.h"
#include "llvm/InitializePasses.h"
#include "llvm/Pass.h"
#include "llvm/Support/Debug.h"
#include "llvm/Transforms/Utils.h"
using namespace llvm;
#define DEBUG_TYPE "canon-freeze"
namespace {
class CanonicalizeFreezeInLoops : public LoopPass {
public:
static char ID;
CanonicalizeFreezeInLoops();
private:
bool runOnLoop(Loop *L, LPPassManager &LPM) override;
void getAnalysisUsage(AnalysisUsage &AU) const override;
};
class CanonicalizeFreezeInLoopsImpl {
Loop *L;
ScalarEvolution &SE;
DominatorTree &DT;
struct FrozenIndPHIInfo {
FreezeInst *FI = nullptr;
PHINode *PHI;
BinaryOperator *StepInst;
unsigned StepValIdx = 0;
FrozenIndPHIInfo(PHINode *PHI, BinaryOperator *StepInst)
: PHI(PHI), StepInst(StepInst) {}
};
bool canHandleInst(const Instruction *I) {
auto Opc = I->getOpcode();
return Opc == Instruction::Add || Opc == Instruction::Sub ||
Opc == Instruction::Mul;
}
void InsertFreezeAndForgetFromSCEV(Use &U);
public:
CanonicalizeFreezeInLoopsImpl(Loop *L, ScalarEvolution &SE, DominatorTree &DT)
: L(L), SE(SE), DT(DT) {}
bool run();
};
}
void CanonicalizeFreezeInLoopsImpl::InsertFreezeAndForgetFromSCEV(Use &U) {
auto *PH = L->getLoopPreheader();
auto *UserI = cast<Instruction>(U.getUser());
auto *ValueToFr = U.get();
assert(L->contains(UserI->getParent()) &&
"Should not process an instruction that isn't inside the loop");
if (isGuaranteedNotToBeUndefOrPoison(ValueToFr, nullptr, UserI, &DT))
return;
LLVM_DEBUG(dbgs() << "canonfr: inserting freeze:\n");
LLVM_DEBUG(dbgs() << "\tUser: " << *U.getUser() << "\n");
LLVM_DEBUG(dbgs() << "\tOperand: " << *U.get() << "\n");
U.set(new FreezeInst(ValueToFr, ValueToFr->getName() + ".frozen",
PH->getTerminator()));
SE.forgetValue(UserI);
}
bool CanonicalizeFreezeInLoopsImpl::run() {
if (!L->isLoopSimplifyForm())
return false;
SmallVector<FrozenIndPHIInfo, 4> Candidates;
for (auto &PHI : L->getHeader()->phis()) {
InductionDescriptor ID;
if (!InductionDescriptor::isInductionPHI(&PHI, L, &SE, ID))
continue;
LLVM_DEBUG(dbgs() << "canonfr: PHI: " << PHI << "\n");
FrozenIndPHIInfo Info(&PHI, ID.getInductionBinOp());
if (!Info.StepInst || !canHandleInst(Info.StepInst)) {
continue;
}
Info.StepValIdx = Info.StepInst->getOperand(0) == &PHI;
Value *StepV = Info.StepInst->getOperand(Info.StepValIdx);
if (auto *StepI = dyn_cast<Instruction>(StepV)) {
if (L->contains(StepI->getParent())) {
continue;
}
}
auto Visit = [&](User *U) {
if (auto *FI = dyn_cast<FreezeInst>(U)) {
LLVM_DEBUG(dbgs() << "canonfr: found: " << *FI << "\n");
Info.FI = FI;
Candidates.push_back(Info);
}
};
for_each(PHI.users(), Visit);
for_each(Info.StepInst->users(), Visit);
}
if (Candidates.empty())
return false;
SmallSet<PHINode *, 8> ProcessedPHIs;
for (const auto &Info : Candidates) {
PHINode *PHI = Info.PHI;
if (!ProcessedPHIs.insert(Info.PHI).second)
continue;
BinaryOperator *StepI = Info.StepInst;
assert(StepI && "Step instruction should have been found");
if (!isGuaranteedNotToBeUndefOrPoison(StepI, nullptr, StepI, &DT)) {
LLVM_DEBUG(dbgs() << "canonfr: drop flags: " << *StepI << "\n");
StepI->dropPoisonGeneratingFlags();
SE.forgetValue(StepI);
}
InsertFreezeAndForgetFromSCEV(StepI->getOperandUse(Info.StepValIdx));
unsigned OperandIdx =
PHI->getOperandNumForIncomingValue(PHI->getIncomingValue(0) == StepI);
InsertFreezeAndForgetFromSCEV(PHI->getOperandUse(OperandIdx));
}
for (const auto &Item : Candidates) {
auto *FI = Item.FI;
LLVM_DEBUG(dbgs() << "canonfr: removing " << *FI << "\n");
SE.forgetValue(FI);
FI->replaceAllUsesWith(FI->getOperand(0));
FI->eraseFromParent();
}
return true;
}
CanonicalizeFreezeInLoops::CanonicalizeFreezeInLoops() : LoopPass(ID) {
initializeCanonicalizeFreezeInLoopsPass(*PassRegistry::getPassRegistry());
}
void CanonicalizeFreezeInLoops::getAnalysisUsage(AnalysisUsage &AU) const {
AU.addPreservedID(LoopSimplifyID);
AU.addRequired<LoopInfoWrapperPass>();
AU.addPreserved<LoopInfoWrapperPass>();
AU.addRequiredID(LoopSimplifyID);
AU.addRequired<ScalarEvolutionWrapperPass>();
AU.addPreserved<ScalarEvolutionWrapperPass>();
AU.addRequired<DominatorTreeWrapperPass>();
AU.addPreserved<DominatorTreeWrapperPass>();
}
bool CanonicalizeFreezeInLoops::runOnLoop(Loop *L, LPPassManager &) {
if (skipLoop(L))
return false;
auto &SE = getAnalysis<ScalarEvolutionWrapperPass>().getSE();
auto &DT = getAnalysis<DominatorTreeWrapperPass>().getDomTree();
return CanonicalizeFreezeInLoopsImpl(L, SE, DT).run();
}
PreservedAnalyses
CanonicalizeFreezeInLoopsPass::run(Loop &L, LoopAnalysisManager &AM,
LoopStandardAnalysisResults &AR,
LPMUpdater &U) {
if (!CanonicalizeFreezeInLoopsImpl(&L, AR.SE, AR.DT).run())
return PreservedAnalyses::all();
return getLoopPassPreservedAnalyses();
}
INITIALIZE_PASS_BEGIN(CanonicalizeFreezeInLoops, "canon-freeze",
"Canonicalize Freeze Instructions in Loops", false, false)
INITIALIZE_PASS_DEPENDENCY(DominatorTreeWrapperPass)
INITIALIZE_PASS_DEPENDENCY(ScalarEvolutionWrapperPass)
INITIALIZE_PASS_DEPENDENCY(LoopSimplify)
INITIALIZE_PASS_END(CanonicalizeFreezeInLoops, "canon-freeze",
"Canonicalize Freeze Instructions in Loops", false, false)
Pass *llvm::createCanonicalizeFreezeInLoopsPass() {
return new CanonicalizeFreezeInLoops();
}
char CanonicalizeFreezeInLoops::ID = 0;