forked from OSchip/llvm-project
PredicateInfo: Support switch statements
Summary: Depends on D29606 and D29682 Makes us pass GVN's edge.ll (we also will pass a few other testcases they just need cleaning up). Thoughts on the Predicate* hiearchy of classes especially welcome :) (it's not clear to me how best to organize it, and currently, the getBlock* seems ... uglier than maybe wasting a field somewhere or something). Reviewers: davide Subscribers: llvm-commits Differential Revision: https://reviews.llvm.org/D29747 llvm-svn: 295889
This commit is contained in:
parent
211a1209a5
commit
fccbda967a
|
@ -92,7 +92,7 @@ class LLVMContext;
|
||||||
class raw_ostream;
|
class raw_ostream;
|
||||||
class OrderedBasicBlock;
|
class OrderedBasicBlock;
|
||||||
|
|
||||||
enum PredicateType { PT_Branch, PT_Assume };
|
enum PredicateType { PT_Branch, PT_Assume, PT_Switch };
|
||||||
|
|
||||||
// Base class for all predicate information we provide.
|
// Base class for all predicate information we provide.
|
||||||
// All of our predicate information has at least a comparison.
|
// All of our predicate information has at least a comparison.
|
||||||
|
@ -103,50 +103,91 @@ public:
|
||||||
// This can be use by passes, when destroying predicateinfo, to know
|
// This can be use by passes, when destroying predicateinfo, to know
|
||||||
// whether they can just drop the intrinsic, or have to merge metadata.
|
// whether they can just drop the intrinsic, or have to merge metadata.
|
||||||
Value *OriginalOp;
|
Value *OriginalOp;
|
||||||
Value *Condition;
|
|
||||||
PredicateBase(const PredicateBase &) = delete;
|
PredicateBase(const PredicateBase &) = delete;
|
||||||
PredicateBase &operator=(const PredicateBase &) = delete;
|
PredicateBase &operator=(const PredicateBase &) = delete;
|
||||||
PredicateBase() = delete;
|
PredicateBase() = delete;
|
||||||
virtual ~PredicateBase() = default;
|
virtual ~PredicateBase() = default;
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
PredicateBase(PredicateType PT, Value *Op, Value *Condition)
|
PredicateBase(PredicateType PT, Value *Op) : Type(PT), OriginalOp(Op) {}
|
||||||
: Type(PT), OriginalOp(Op), Condition(Condition) {}
|
};
|
||||||
|
|
||||||
|
class PredicateWithCondition : public PredicateBase {
|
||||||
|
public:
|
||||||
|
Value *Condition;
|
||||||
|
static inline bool classof(const PredicateBase *PB) {
|
||||||
|
return PB->Type == PT_Assume || PB->Type == PT_Branch || PB->Type == PT_Switch;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected:
|
||||||
|
PredicateWithCondition(PredicateType PT, Value *Op, Value *Condition)
|
||||||
|
: PredicateBase(PT, Op), Condition(Condition) {}
|
||||||
};
|
};
|
||||||
|
|
||||||
// Provides predicate information for assumes. Since assumes are always true,
|
// Provides predicate information for assumes. Since assumes are always true,
|
||||||
// we simply provide the assume instruction, so you can tell your relative
|
// we simply provide the assume instruction, so you can tell your relative
|
||||||
// position to it.
|
// position to it.
|
||||||
class PredicateAssume : public PredicateBase {
|
class PredicateAssume : public PredicateWithCondition {
|
||||||
public:
|
public:
|
||||||
IntrinsicInst *AssumeInst;
|
IntrinsicInst *AssumeInst;
|
||||||
PredicateAssume(Value *Op, IntrinsicInst *AssumeInst, Value *Condition)
|
PredicateAssume(Value *Op, IntrinsicInst *AssumeInst, Value *Condition)
|
||||||
: PredicateBase(PT_Assume, Op, Condition), AssumeInst(AssumeInst) {}
|
: PredicateWithCondition(PT_Assume, Op, Condition),
|
||||||
|
AssumeInst(AssumeInst) {}
|
||||||
PredicateAssume() = delete;
|
PredicateAssume() = delete;
|
||||||
static inline bool classof(const PredicateBase *PB) {
|
static inline bool classof(const PredicateBase *PB) {
|
||||||
return PB->Type == PT_Assume;
|
return PB->Type == PT_Assume;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
// Provides predicate information for branches.
|
// Mixin class for edge predicates. The FROM block is the block where the
|
||||||
class PredicateBranch : public PredicateBase {
|
// predicate originates, and the TO block is the block where the predicate is
|
||||||
|
// valid.
|
||||||
|
class PredicateWithEdge : public PredicateWithCondition {
|
||||||
|
public:
|
||||||
|
BasicBlock *From;
|
||||||
|
BasicBlock *To;
|
||||||
|
PredicateWithEdge() = delete;
|
||||||
|
static inline bool classof(const PredicateBase *PB) {
|
||||||
|
return PB->Type == PT_Branch || PB->Type == PT_Switch;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected:
|
||||||
|
PredicateWithEdge(PredicateType PType, Value *Op, BasicBlock *From,
|
||||||
|
BasicBlock *To, Value *Cond)
|
||||||
|
: PredicateWithCondition(PType, Op, Cond), From(From), To(To) {}
|
||||||
|
};
|
||||||
|
|
||||||
|
// Provides predicate information for branches.
|
||||||
|
class PredicateBranch : public PredicateWithEdge {
|
||||||
public:
|
public:
|
||||||
// This is the block that is conditional upon the condition.
|
|
||||||
BasicBlock *BranchBB;
|
|
||||||
// This is one of the true/false successors of BranchBB.
|
|
||||||
BasicBlock *SplitBB;
|
|
||||||
// If true, SplitBB is the true successor, otherwise it's the false successor.
|
// If true, SplitBB is the true successor, otherwise it's the false successor.
|
||||||
bool TrueEdge;
|
bool TrueEdge;
|
||||||
PredicateBranch(Value *Op, BasicBlock *BranchBB, BasicBlock *SplitBB,
|
PredicateBranch(Value *Op, BasicBlock *BranchBB, BasicBlock *SplitBB,
|
||||||
Value *Condition, bool TakenEdge)
|
Value *Condition, bool TakenEdge)
|
||||||
: PredicateBase(PT_Branch, Op, Condition), BranchBB(BranchBB),
|
: PredicateWithEdge(PT_Branch, Op, BranchBB, SplitBB, Condition),
|
||||||
SplitBB(SplitBB), TrueEdge(TakenEdge) {}
|
TrueEdge(TakenEdge) {}
|
||||||
PredicateBranch() = delete;
|
PredicateBranch() = delete;
|
||||||
static inline bool classof(const PredicateBase *PB) {
|
static inline bool classof(const PredicateBase *PB) {
|
||||||
return PB->Type == PT_Branch;
|
return PB->Type == PT_Branch;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
class PredicateSwitch : public PredicateWithEdge {
|
||||||
|
public:
|
||||||
|
Value *CaseValue;
|
||||||
|
// This is the switch instruction.
|
||||||
|
SwitchInst *Switch;
|
||||||
|
PredicateSwitch(Value *Op, BasicBlock *SwitchBB, BasicBlock *TargetBB,
|
||||||
|
Value *CaseValue, SwitchInst *SI)
|
||||||
|
: PredicateWithEdge(PT_Switch, Op, SwitchBB, TargetBB,
|
||||||
|
SI->getCondition()),
|
||||||
|
CaseValue(CaseValue), Switch(SI) {}
|
||||||
|
PredicateSwitch() = delete;
|
||||||
|
static inline bool classof(const PredicateBase *PB) {
|
||||||
|
return PB->Type == PT_Switch;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
// This name is used in a few places, so kick it into their own namespace
|
// This name is used in a few places, so kick it into their own namespace
|
||||||
namespace PredicateInfoClasses {
|
namespace PredicateInfoClasses {
|
||||||
struct ValueDFS;
|
struct ValueDFS;
|
||||||
|
@ -189,6 +230,7 @@ private:
|
||||||
void buildPredicateInfo();
|
void buildPredicateInfo();
|
||||||
void processAssume(IntrinsicInst *, BasicBlock *, SmallPtrSetImpl<Value *> &);
|
void processAssume(IntrinsicInst *, BasicBlock *, SmallPtrSetImpl<Value *> &);
|
||||||
void processBranch(BranchInst *, BasicBlock *, SmallPtrSetImpl<Value *> &);
|
void processBranch(BranchInst *, BasicBlock *, SmallPtrSetImpl<Value *> &);
|
||||||
|
void processSwitch(SwitchInst *, BasicBlock *, SmallPtrSetImpl<Value *> &);
|
||||||
void renameUses(SmallPtrSetImpl<Value *> &);
|
void renameUses(SmallPtrSetImpl<Value *> &);
|
||||||
using ValueDFS = PredicateInfoClasses::ValueDFS;
|
using ValueDFS = PredicateInfoClasses::ValueDFS;
|
||||||
typedef SmallVectorImpl<ValueDFS> ValueDFSStack;
|
typedef SmallVectorImpl<ValueDFS> ValueDFSStack;
|
||||||
|
|
|
@ -855,15 +855,19 @@ NewGVN::performSymbolicPredicateInfoEvaluation(Instruction *I) {
|
||||||
return nullptr;
|
return nullptr;
|
||||||
|
|
||||||
DEBUG(dbgs() << "Found predicate info from instruction !\n");
|
DEBUG(dbgs() << "Found predicate info from instruction !\n");
|
||||||
auto *CopyOf = I->getOperand(0);
|
|
||||||
auto *Cond = dyn_cast<Instruction>(PI->Condition);
|
auto *PWC = dyn_cast<PredicateWithCondition>(PI);
|
||||||
if (!Cond)
|
if (!PWC)
|
||||||
return nullptr;
|
return nullptr;
|
||||||
|
|
||||||
|
auto *CopyOf = I->getOperand(0);
|
||||||
|
auto *Cond = PWC->Condition;
|
||||||
|
|
||||||
// If this a copy of the condition, it must be either true or false depending
|
// If this a copy of the condition, it must be either true or false depending
|
||||||
// on the predicate info type and edge
|
// on the predicate info type and edge
|
||||||
if (CopyOf == Cond) {
|
if (CopyOf == Cond) {
|
||||||
addPredicateUsers(PI, I);
|
// We should not need to add predicate users because the predicate info is
|
||||||
|
// already a use of this operand.
|
||||||
if (isa<PredicateAssume>(PI))
|
if (isa<PredicateAssume>(PI))
|
||||||
return createConstantExpression(ConstantInt::getTrue(Cond->getType()));
|
return createConstantExpression(ConstantInt::getTrue(Cond->getType()));
|
||||||
if (auto *PBranch = dyn_cast<PredicateBranch>(PI)) {
|
if (auto *PBranch = dyn_cast<PredicateBranch>(PI)) {
|
||||||
|
@ -871,32 +875,36 @@ NewGVN::performSymbolicPredicateInfoEvaluation(Instruction *I) {
|
||||||
return createConstantExpression(ConstantInt::getTrue(Cond->getType()));
|
return createConstantExpression(ConstantInt::getTrue(Cond->getType()));
|
||||||
return createConstantExpression(ConstantInt::getFalse(Cond->getType()));
|
return createConstantExpression(ConstantInt::getFalse(Cond->getType()));
|
||||||
}
|
}
|
||||||
|
if (auto *PSwitch = dyn_cast<PredicateSwitch>(PI))
|
||||||
|
return createConstantExpression(cast<Constant>(PSwitch->CaseValue));
|
||||||
}
|
}
|
||||||
// Not a copy of the condition, so see what the predicates tell us about this
|
|
||||||
// value.
|
|
||||||
// Not a copy of the condition, so see what the predicates tell us about this
|
// Not a copy of the condition, so see what the predicates tell us about this
|
||||||
// value. First, though, we check to make sure the value is actually a copy
|
// value. First, though, we check to make sure the value is actually a copy
|
||||||
// of one of the condition operands. It's possible, in certain cases, for it
|
// of one of the condition operands. It's possible, in certain cases, for it
|
||||||
// to be a copy of a predicateinfo copy. In particular, if two branch
|
// to be a copy of a predicateinfo copy. In particular, if two branch
|
||||||
// operations use the same condition, and one branch dominates the other, we
|
// operations use the same condition, and one branch dominates the other, we
|
||||||
// will end up with a copy of a copy. This is currently a small deficiency in
|
// will end up with a copy of a copy. This is currently a small deficiency in
|
||||||
// predicateinfo. What will end up happening here is that we will value
|
// predicateinfo. What will end up happening here is that we will value
|
||||||
// number both copies the same anyway.
|
// number both copies the same anyway.
|
||||||
if (CopyOf != Cond->getOperand(0) && CopyOf != Cond->getOperand(1)) {
|
|
||||||
|
// Everything below relies on the condition being a comparison.
|
||||||
|
auto *Cmp = dyn_cast<CmpInst>(Cond);
|
||||||
|
if (!Cmp)
|
||||||
|
return nullptr;
|
||||||
|
|
||||||
|
if (CopyOf != Cmp->getOperand(0) && CopyOf != Cmp->getOperand(1)) {
|
||||||
DEBUG(dbgs() << "Copy is not of any condition operands!");
|
DEBUG(dbgs() << "Copy is not of any condition operands!");
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
Value *FirstOp = lookupOperandLeader(Cond->getOperand(0));
|
Value *FirstOp = lookupOperandLeader(Cmp->getOperand(0));
|
||||||
Value *SecondOp = lookupOperandLeader(Cond->getOperand(1));
|
Value *SecondOp = lookupOperandLeader(Cmp->getOperand(1));
|
||||||
bool SwappedOps = false;
|
bool SwappedOps = false;
|
||||||
// Sort the ops
|
// Sort the ops
|
||||||
if (shouldSwapOperands(FirstOp, SecondOp)) {
|
if (shouldSwapOperands(FirstOp, SecondOp)) {
|
||||||
std::swap(FirstOp, SecondOp);
|
std::swap(FirstOp, SecondOp);
|
||||||
SwappedOps = true;
|
SwappedOps = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Everything below relies on the condition being a comparison.
|
|
||||||
auto *Cmp = dyn_cast<CmpInst>(Cond);
|
|
||||||
CmpInst::Predicate Predicate =
|
CmpInst::Predicate Predicate =
|
||||||
SwappedOps ? Cmp->getSwappedPredicate() : Cmp->getPredicate();
|
SwappedOps ? Cmp->getSwappedPredicate() : Cmp->getPredicate();
|
||||||
|
|
||||||
|
@ -1095,7 +1103,6 @@ const Expression *NewGVN::performSymbolicCmpEvaluation(Instruction *I) {
|
||||||
|
|
||||||
// Avoid processing the same info twice
|
// Avoid processing the same info twice
|
||||||
const PredicateBase *LastPredInfo = nullptr;
|
const PredicateBase *LastPredInfo = nullptr;
|
||||||
|
|
||||||
// See if we know something about the comparison itself, like it is the target
|
// See if we know something about the comparison itself, like it is the target
|
||||||
// of an assume.
|
// of an assume.
|
||||||
auto *CmpPI = PredInfo->getPredicateInfoFor(I);
|
auto *CmpPI = PredInfo->getPredicateInfoFor(I);
|
||||||
|
@ -1141,6 +1148,7 @@ const Expression *NewGVN::performSymbolicCmpEvaluation(Instruction *I) {
|
||||||
if (PI == LastPredInfo)
|
if (PI == LastPredInfo)
|
||||||
continue;
|
continue;
|
||||||
LastPredInfo = PI;
|
LastPredInfo = PI;
|
||||||
|
|
||||||
// TODO: Along the false edge, we may know more things too, like icmp of
|
// TODO: Along the false edge, we may know more things too, like icmp of
|
||||||
// same operands is false.
|
// same operands is false.
|
||||||
// TODO: We only handle actual comparison conditions below, not and/or.
|
// TODO: We only handle actual comparison conditions below, not and/or.
|
||||||
|
|
|
@ -49,8 +49,36 @@ INITIALIZE_PASS_END(PredicateInfoPrinterLegacyPass, "print-predicateinfo",
|
||||||
static cl::opt<bool> VerifyPredicateInfo(
|
static cl::opt<bool> VerifyPredicateInfo(
|
||||||
"verify-predicateinfo", cl::init(false), cl::Hidden,
|
"verify-predicateinfo", cl::init(false), cl::Hidden,
|
||||||
cl::desc("Verify PredicateInfo in legacy printer pass."));
|
cl::desc("Verify PredicateInfo in legacy printer pass."));
|
||||||
|
namespace {
|
||||||
DEBUG_COUNTER(RenameCounter, "predicateinfo-rename",
|
DEBUG_COUNTER(RenameCounter, "predicateinfo-rename",
|
||||||
"Controls which variables are renamed with predicateinfo")
|
"Controls which variables are renamed with predicateinfo")
|
||||||
|
// Given a predicate info that is a type of branching terminator, get the
|
||||||
|
// branching block.
|
||||||
|
const BasicBlock *getBranchBlock(const PredicateBase *PB) {
|
||||||
|
assert(isa<PredicateWithEdge>(PB) &&
|
||||||
|
"Only branches and switches should have PHIOnly defs that "
|
||||||
|
"require branch blocks.");
|
||||||
|
return cast<PredicateWithEdge>(PB)->From;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Given a predicate info that is a type of branching terminator, get the
|
||||||
|
// branching terminator.
|
||||||
|
static Instruction *getBranchTerminator(const PredicateBase *PB) {
|
||||||
|
assert(isa<PredicateWithEdge>(PB) &&
|
||||||
|
"Not a predicate info type we know how to get a terminator from.");
|
||||||
|
return cast<PredicateWithEdge>(PB)->From->getTerminator();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Given a predicate info that is a type of branching terminator, get the
|
||||||
|
// edge this predicate info represents
|
||||||
|
const std::pair<BasicBlock *, BasicBlock *>
|
||||||
|
getBlockEdge(const PredicateBase *PB) {
|
||||||
|
assert(isa<PredicateWithEdge>(PB) &&
|
||||||
|
"Not a predicate info type we know how to get an edge from.");
|
||||||
|
const auto *PEdge = cast<PredicateWithEdge>(PB);
|
||||||
|
return std::make_pair(PEdge->From, PEdge->To);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
namespace llvm {
|
namespace llvm {
|
||||||
namespace PredicateInfoClasses {
|
namespace PredicateInfoClasses {
|
||||||
|
@ -110,15 +138,14 @@ struct ValueDFS_Compare {
|
||||||
}
|
}
|
||||||
|
|
||||||
// For a phi use, or a non-materialized def, return the edge it represents.
|
// For a phi use, or a non-materialized def, return the edge it represents.
|
||||||
const std::pair<const BasicBlock *, const BasicBlock *>
|
const std::pair<BasicBlock *, BasicBlock *>
|
||||||
getBlockEdge(const ValueDFS &VD) const {
|
getBlockEdge(const ValueDFS &VD) const {
|
||||||
if (!VD.Def && VD.U) {
|
if (!VD.Def && VD.U) {
|
||||||
auto *PHI = cast<PHINode>(VD.U->getUser());
|
auto *PHI = cast<PHINode>(VD.U->getUser());
|
||||||
return std::make_pair(PHI->getIncomingBlock(*VD.U), PHI->getParent());
|
return std::make_pair(PHI->getIncomingBlock(*VD.U), PHI->getParent());
|
||||||
}
|
}
|
||||||
// This is really a non-materialized def.
|
// This is really a non-materialized def.
|
||||||
auto *PBranch = cast<PredicateBranch>(VD.PInfo);
|
return ::getBlockEdge(VD.PInfo);
|
||||||
return std::make_pair(PBranch->BranchBB, PBranch->SplitBB);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// For two phi related values, return the ordering.
|
// For two phi related values, return the ordering.
|
||||||
|
@ -208,17 +235,13 @@ bool PredicateInfo::stackIsInScope(const ValueDFSStack &Stack,
|
||||||
auto *PHI = dyn_cast<PHINode>(VDUse.U->getUser());
|
auto *PHI = dyn_cast<PHINode>(VDUse.U->getUser());
|
||||||
if (!PHI)
|
if (!PHI)
|
||||||
return false;
|
return false;
|
||||||
// The only EdgeOnly defs should be branch info.
|
// Check edge
|
||||||
auto *PBranch = dyn_cast<PredicateBranch>(Stack.back().PInfo);
|
|
||||||
assert(PBranch && "Only branches should have EdgeOnly defs");
|
|
||||||
// Check edge matches us.
|
|
||||||
BasicBlock *EdgePred = PHI->getIncomingBlock(*VDUse.U);
|
BasicBlock *EdgePred = PHI->getIncomingBlock(*VDUse.U);
|
||||||
if (EdgePred != PBranch->BranchBB)
|
if (EdgePred != getBranchBlock(Stack.back().PInfo))
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
// Use dominates, which knows how to handle edge dominance.
|
// Use dominates, which knows how to handle edge dominance.
|
||||||
return DT.dominates(BasicBlockEdge(PBranch->BranchBB, PBranch->SplitBB),
|
return DT.dominates(getBlockEdge(Stack.back().PInfo), *VDUse.U);
|
||||||
*VDUse.U);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return (VDUse.DFSIn >= Stack.back().DFSIn &&
|
return (VDUse.DFSIn >= Stack.back().DFSIn &&
|
||||||
|
@ -400,6 +423,33 @@ void PredicateInfo::processBranch(BranchInst *BI, BasicBlock *BranchBB,
|
||||||
CmpOperands.clear();
|
CmpOperands.clear();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
// Process a block terminating switch, and place relevant operations to be
|
||||||
|
// renamed into OpsToRename.
|
||||||
|
void PredicateInfo::processSwitch(SwitchInst *SI, BasicBlock *BranchBB,
|
||||||
|
SmallPtrSetImpl<Value *> &OpsToRename) {
|
||||||
|
Value *Op = SI->getCondition();
|
||||||
|
if ((!isa<Instruction>(Op) && !isa<Argument>(Op)) || Op->hasOneUse())
|
||||||
|
return;
|
||||||
|
|
||||||
|
// Remember how many outgoing edges there are to every successor.
|
||||||
|
SmallDenseMap<BasicBlock *, unsigned, 16> SwitchEdges;
|
||||||
|
for (unsigned i = 0, e = SI->getNumSuccessors(); i != e; ++i) {
|
||||||
|
BasicBlock *TargetBlock = SI->getSuccessor(i);
|
||||||
|
++SwitchEdges[TargetBlock];
|
||||||
|
}
|
||||||
|
|
||||||
|
// Now propagate info for each case value
|
||||||
|
for (auto C : SI->cases()) {
|
||||||
|
BasicBlock *TargetBlock = C.getCaseSuccessor();
|
||||||
|
if (SwitchEdges.lookup(TargetBlock) == 1) {
|
||||||
|
PredicateSwitch *PS = new PredicateSwitch(
|
||||||
|
Op, SI->getParent(), TargetBlock, C.getCaseValue(), SI);
|
||||||
|
addInfoFor(OpsToRename, Op, PS);
|
||||||
|
if (!TargetBlock->getSinglePredecessor())
|
||||||
|
EdgeUsesOnly.insert({BranchBB, TargetBlock});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Build predicate info for our function
|
// Build predicate info for our function
|
||||||
void PredicateInfo::buildPredicateInfo() {
|
void PredicateInfo::buildPredicateInfo() {
|
||||||
|
@ -413,6 +463,8 @@ void PredicateInfo::buildPredicateInfo() {
|
||||||
if (!BI->isConditional())
|
if (!BI->isConditional())
|
||||||
continue;
|
continue;
|
||||||
processBranch(BI, BranchBB, OpsToRename);
|
processBranch(BI, BranchBB, OpsToRename);
|
||||||
|
} else if (auto *SI = dyn_cast<SwitchInst>(BranchBB->getTerminator())) {
|
||||||
|
processSwitch(SI, BranchBB, OpsToRename);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
for (auto &Assume : AC.assumptions()) {
|
for (auto &Assume : AC.assumptions()) {
|
||||||
|
@ -422,6 +474,9 @@ void PredicateInfo::buildPredicateInfo() {
|
||||||
// Now rename all our operations.
|
// Now rename all our operations.
|
||||||
renameUses(OpsToRename);
|
renameUses(OpsToRename);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Given the renaming stack, make all the operands currently on the stack real
|
||||||
|
// by inserting them into the IR. Return the last operation's value.
|
||||||
Value *PredicateInfo::materializeStack(unsigned int &Counter,
|
Value *PredicateInfo::materializeStack(unsigned int &Counter,
|
||||||
ValueDFSStack &RenameStack,
|
ValueDFSStack &RenameStack,
|
||||||
Value *OrigOp) {
|
Value *OrigOp) {
|
||||||
|
@ -441,14 +496,13 @@ Value *PredicateInfo::materializeStack(unsigned int &Counter,
|
||||||
RenameIter == RenameStack.begin() ? OrigOp : (RenameIter - 1)->Def;
|
RenameIter == RenameStack.begin() ? OrigOp : (RenameIter - 1)->Def;
|
||||||
ValueDFS &Result = *RenameIter;
|
ValueDFS &Result = *RenameIter;
|
||||||
auto *ValInfo = Result.PInfo;
|
auto *ValInfo = Result.PInfo;
|
||||||
// For branches, we can just place the operand in the branch block before
|
// For edge predicates, we can just place the operand in the block before
|
||||||
// the terminator. For assume, we have to place it right before the assume
|
// the terminator. For assume, we have to place it right before the assume
|
||||||
// to ensure we dominate all of our uses. Always insert right before the
|
// to ensure we dominate all of our uses. Always insert right before the
|
||||||
// relevant instruction (terminator, assume), so that we insert in proper
|
// relevant instruction (terminator, assume), so that we insert in proper
|
||||||
// order in the case of multiple predicateinfo in the same block.
|
// order in the case of multiple predicateinfo in the same block.
|
||||||
if (isa<PredicateBranch>(ValInfo)) {
|
if (isa<PredicateWithEdge>(ValInfo)) {
|
||||||
auto *PBranch = cast<PredicateBranch>(ValInfo);
|
IRBuilder<> B(getBranchTerminator(ValInfo));
|
||||||
IRBuilder<> B(PBranch->BranchBB->getTerminator());
|
|
||||||
Function *IF = Intrinsic::getDeclaration(
|
Function *IF = Intrinsic::getDeclaration(
|
||||||
F.getParent(), Intrinsic::ssa_copy, Op->getType());
|
F.getParent(), Intrinsic::ssa_copy, Op->getType());
|
||||||
CallInst *PIC =
|
CallInst *PIC =
|
||||||
|
@ -515,14 +569,14 @@ void PredicateInfo::renameUses(SmallPtrSetImpl<Value *> &OpsToRename) {
|
||||||
VD.DFSOut = DomNode->getDFSNumOut();
|
VD.DFSOut = DomNode->getDFSNumOut();
|
||||||
VD.PInfo = PossibleCopy;
|
VD.PInfo = PossibleCopy;
|
||||||
OrderedUses.push_back(VD);
|
OrderedUses.push_back(VD);
|
||||||
} else if (const auto *PBranch =
|
} else if (isa<PredicateWithEdge>(PossibleCopy)) {
|
||||||
dyn_cast<PredicateBranch>(PossibleCopy)) {
|
|
||||||
// If we can only do phi uses, we treat it like it's in the branch
|
// If we can only do phi uses, we treat it like it's in the branch
|
||||||
// block, and handle it specially. We know that it goes last, and only
|
// block, and handle it specially. We know that it goes last, and only
|
||||||
// dominate phi uses.
|
// dominate phi uses.
|
||||||
if (EdgeUsesOnly.count({PBranch->BranchBB, PBranch->SplitBB})) {
|
auto BlockEdge = getBlockEdge(PossibleCopy);
|
||||||
|
if (EdgeUsesOnly.count(BlockEdge)) {
|
||||||
VD.LocalNum = LN_Last;
|
VD.LocalNum = LN_Last;
|
||||||
auto *DomNode = DT.getNode(PBranch->BranchBB);
|
auto *DomNode = DT.getNode(BlockEdge.first);
|
||||||
if (DomNode) {
|
if (DomNode) {
|
||||||
VD.DFSIn = DomNode->getDFSNumIn();
|
VD.DFSIn = DomNode->getDFSNumIn();
|
||||||
VD.DFSOut = DomNode->getDFSNumOut();
|
VD.DFSOut = DomNode->getDFSNumOut();
|
||||||
|
@ -535,7 +589,7 @@ void PredicateInfo::renameUses(SmallPtrSetImpl<Value *> &OpsToRename) {
|
||||||
// insertion in the branch block).
|
// insertion in the branch block).
|
||||||
// Insert a possible copy at the split block and before the branch.
|
// Insert a possible copy at the split block and before the branch.
|
||||||
VD.LocalNum = LN_First;
|
VD.LocalNum = LN_First;
|
||||||
auto *DomNode = DT.getNode(PBranch->SplitBB);
|
auto *DomNode = DT.getNode(BlockEdge.second);
|
||||||
if (DomNode) {
|
if (DomNode) {
|
||||||
VD.DFSIn = DomNode->getDFSNumIn();
|
VD.DFSIn = DomNode->getDFSNumIn();
|
||||||
VD.DFSOut = DomNode->getDFSNumOut();
|
VD.DFSOut = DomNode->getDFSNumOut();
|
||||||
|
@ -687,12 +741,24 @@ public:
|
||||||
formatted_raw_ostream &OS) {
|
formatted_raw_ostream &OS) {
|
||||||
if (const auto *PI = PredInfo->getPredicateInfoFor(I)) {
|
if (const auto *PI = PredInfo->getPredicateInfoFor(I)) {
|
||||||
OS << "; Has predicate info\n";
|
OS << "; Has predicate info\n";
|
||||||
if (const auto *PB = dyn_cast<PredicateBranch>(PI))
|
if (const auto *PB = dyn_cast<PredicateBranch>(PI)) {
|
||||||
OS << "; branch predicate info { TrueEdge: " << PB->TrueEdge
|
OS << "; branch predicate info { TrueEdge: " << PB->TrueEdge
|
||||||
<< " Comparison:" << *PB->Condition << " }\n";
|
<< " Comparison:" << *PB->Condition << " Edge: [";
|
||||||
else if (const auto *PA = dyn_cast<PredicateAssume>(PI))
|
PB->From->printAsOperand(OS);
|
||||||
|
OS << ",";
|
||||||
|
PB->To->printAsOperand(OS);
|
||||||
|
OS << "] }\n";
|
||||||
|
} else if (const auto *PS = dyn_cast<PredicateSwitch>(PI)) {
|
||||||
|
OS << "; switch predicate info { CaseValue: " << *PS->CaseValue
|
||||||
|
<< " Switch:" << *PS->Switch << " Edge: [";
|
||||||
|
PS->From->printAsOperand(OS);
|
||||||
|
OS << ",";
|
||||||
|
PS->To->printAsOperand(OS);
|
||||||
|
OS << "] }\n";
|
||||||
|
} else if (const auto *PA = dyn_cast<PredicateAssume>(PI)) {
|
||||||
OS << "; assume predicate info {"
|
OS << "; assume predicate info {"
|
||||||
<< " Comparison:" << *PA->Condition << " }\n";
|
<< " Comparison:" << *PA->Condition << " }\n";
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
|
@ -1,116 +1,211 @@
|
||||||
|
; NOTE: Assertions have been autogenerated by utils/update_test_checks.py
|
||||||
; RUN: opt < %s -basicaa -newgvn -S | FileCheck %s
|
; RUN: opt < %s -basicaa -newgvn -S | FileCheck %s
|
||||||
|
|
||||||
|
|
||||||
declare void @foo(i1)
|
declare void @foo(i1)
|
||||||
declare void @bar(i32)
|
declare void @bar(i32)
|
||||||
|
|
||||||
; CHECK-LABEL: @test3(
|
|
||||||
define void @test3(i32 %x, i32 %y) {
|
define void @test3(i32 %x, i32 %y) {
|
||||||
|
; CHECK-LABEL: @test3(
|
||||||
|
; CHECK-NEXT: [[XZ:%.*]] = icmp eq i32 [[X:%.*]], 0
|
||||||
|
; CHECK-NEXT: [[YZ:%.*]] = icmp eq i32 [[Y:%.*]], 0
|
||||||
|
; CHECK-NEXT: [[Z:%.*]] = and i1 [[XZ]], [[YZ]]
|
||||||
|
; CHECK-NEXT: br i1 [[Z]], label [[BOTH_ZERO:%.*]], label [[NOPE:%.*]]
|
||||||
|
; CHECK: both_zero:
|
||||||
|
; CHECK-NEXT: call void @foo(i1 true)
|
||||||
|
; CHECK-NEXT: call void @foo(i1 true)
|
||||||
|
; CHECK-NEXT: call void @bar(i32 0)
|
||||||
|
; CHECK-NEXT: call void @bar(i32 0)
|
||||||
|
; CHECK-NEXT: ret void
|
||||||
|
; CHECK: nope:
|
||||||
|
; CHECK-NEXT: call void @foo(i1 false)
|
||||||
|
; CHECK-NEXT: ret void
|
||||||
|
;
|
||||||
%xz = icmp eq i32 %x, 0
|
%xz = icmp eq i32 %x, 0
|
||||||
%yz = icmp eq i32 %y, 0
|
%yz = icmp eq i32 %y, 0
|
||||||
%z = and i1 %xz, %yz
|
%z = and i1 %xz, %yz
|
||||||
br i1 %z, label %both_zero, label %nope
|
br i1 %z, label %both_zero, label %nope
|
||||||
both_zero:
|
both_zero:
|
||||||
call void @foo(i1 %xz)
|
call void @foo(i1 %xz)
|
||||||
; CHECK: call void @foo(i1 true)
|
|
||||||
call void @foo(i1 %yz)
|
call void @foo(i1 %yz)
|
||||||
; CHECK: call void @foo(i1 true)
|
|
||||||
call void @bar(i32 %x)
|
call void @bar(i32 %x)
|
||||||
; CHECK: call void @bar(i32 0)
|
|
||||||
call void @bar(i32 %y)
|
call void @bar(i32 %y)
|
||||||
; CHECK: call void @bar(i32 0)
|
|
||||||
ret void
|
ret void
|
||||||
nope:
|
nope:
|
||||||
call void @foo(i1 %z)
|
call void @foo(i1 %z)
|
||||||
; CHECK: call void @foo(i1 false)
|
ret void
|
||||||
|
}
|
||||||
|
define void @test4(i1 %b, i32 %x) {
|
||||||
|
; CHECK-LABEL: @test4(
|
||||||
|
; CHECK-NEXT: br i1 [[B:%.*]], label [[SW:%.*]], label [[CASE3:%.*]]
|
||||||
|
; CHECK: sw:
|
||||||
|
; CHECK-NEXT: switch i32 [[X:%.*]], label [[DEFAULT:%.*]] [
|
||||||
|
; CHECK-NEXT: i32 0, label [[CASE0:%.*]]
|
||||||
|
; CHECK-NEXT: i32 1, label [[CASE1:%.*]]
|
||||||
|
; CHECK-NEXT: i32 2, label [[CASE0]]
|
||||||
|
; CHECK-NEXT: i32 3, label [[CASE3]]
|
||||||
|
; CHECK-NEXT: i32 4, label [[DEFAULT]]
|
||||||
|
; CHECK-NEXT: ]
|
||||||
|
; CHECK: default:
|
||||||
|
; CHECK-NEXT: call void @bar(i32 [[X]])
|
||||||
|
; CHECK-NEXT: ret void
|
||||||
|
; CHECK: case0:
|
||||||
|
; CHECK-NEXT: call void @bar(i32 [[X]])
|
||||||
|
; CHECK-NEXT: ret void
|
||||||
|
; CHECK: case1:
|
||||||
|
; CHECK-NEXT: call void @bar(i32 1)
|
||||||
|
; CHECK-NEXT: ret void
|
||||||
|
; CHECK: case3:
|
||||||
|
; CHECK-NEXT: call void @bar(i32 [[X]])
|
||||||
|
; CHECK-NEXT: ret void
|
||||||
|
;
|
||||||
|
br i1 %b, label %sw, label %case3
|
||||||
|
sw:
|
||||||
|
switch i32 %x, label %default [
|
||||||
|
i32 0, label %case0
|
||||||
|
i32 1, label %case1
|
||||||
|
i32 2, label %case0
|
||||||
|
i32 3, label %case3
|
||||||
|
i32 4, label %default
|
||||||
|
]
|
||||||
|
default:
|
||||||
|
call void @bar(i32 %x)
|
||||||
|
ret void
|
||||||
|
case0:
|
||||||
|
call void @bar(i32 %x)
|
||||||
|
ret void
|
||||||
|
case1:
|
||||||
|
call void @bar(i32 %x)
|
||||||
|
ret void
|
||||||
|
case3:
|
||||||
|
call void @bar(i32 %x)
|
||||||
ret void
|
ret void
|
||||||
}
|
}
|
||||||
|
|
||||||
; CHECK-LABEL: @test5(
|
|
||||||
define i1 @test5(i32 %x, i32 %y) {
|
define i1 @test5(i32 %x, i32 %y) {
|
||||||
|
; CHECK-LABEL: @test5(
|
||||||
|
; CHECK-NEXT: [[CMP:%.*]] = icmp eq i32 [[X:%.*]], [[Y:%.*]]
|
||||||
|
; CHECK-NEXT: br i1 [[CMP]], label [[SAME:%.*]], label [[DIFFERENT:%.*]]
|
||||||
|
; CHECK: same:
|
||||||
|
; CHECK-NEXT: ret i1 false
|
||||||
|
; CHECK: different:
|
||||||
|
; CHECK-NEXT: ret i1 false
|
||||||
|
;
|
||||||
%cmp = icmp eq i32 %x, %y
|
%cmp = icmp eq i32 %x, %y
|
||||||
br i1 %cmp, label %same, label %different
|
br i1 %cmp, label %same, label %different
|
||||||
|
|
||||||
same:
|
same:
|
||||||
%cmp2 = icmp ne i32 %x, %y
|
%cmp2 = icmp ne i32 %x, %y
|
||||||
; CHECK: ret i1 false
|
|
||||||
ret i1 %cmp2
|
ret i1 %cmp2
|
||||||
|
|
||||||
different:
|
different:
|
||||||
%cmp3 = icmp eq i32 %x, %y
|
%cmp3 = icmp eq i32 %x, %y
|
||||||
; CHECK: ret i1 false
|
|
||||||
ret i1 %cmp3
|
ret i1 %cmp3
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
; CHECK-LABEL: @test7(
|
|
||||||
define i1 @test7(i32 %x, i32 %y) {
|
define i1 @test7(i32 %x, i32 %y) {
|
||||||
|
; CHECK-LABEL: @test7(
|
||||||
|
; CHECK-NEXT: [[CMP:%.*]] = icmp sgt i32 [[X:%.*]], [[Y:%.*]]
|
||||||
|
; CHECK-NEXT: br i1 [[CMP]], label [[SAME:%.*]], label [[DIFFERENT:%.*]]
|
||||||
|
; CHECK: same:
|
||||||
|
; CHECK-NEXT: ret i1 false
|
||||||
|
; CHECK: different:
|
||||||
|
; CHECK-NEXT: ret i1 false
|
||||||
|
;
|
||||||
%cmp = icmp sgt i32 %x, %y
|
%cmp = icmp sgt i32 %x, %y
|
||||||
br i1 %cmp, label %same, label %different
|
br i1 %cmp, label %same, label %different
|
||||||
|
|
||||||
same:
|
same:
|
||||||
%cmp2 = icmp sle i32 %x, %y
|
%cmp2 = icmp sle i32 %x, %y
|
||||||
; CHECK: ret i1 false
|
|
||||||
ret i1 %cmp2
|
ret i1 %cmp2
|
||||||
|
|
||||||
different:
|
different:
|
||||||
%cmp3 = icmp sgt i32 %x, %y
|
%cmp3 = icmp sgt i32 %x, %y
|
||||||
; CHECK: ret i1 false
|
|
||||||
ret i1 %cmp3
|
ret i1 %cmp3
|
||||||
}
|
}
|
||||||
|
|
||||||
; CHECK-LABEL: @test7_fp(
|
|
||||||
define i1 @test7_fp(float %x, float %y) {
|
define i1 @test7_fp(float %x, float %y) {
|
||||||
|
; CHECK-LABEL: @test7_fp(
|
||||||
|
; CHECK-NEXT: [[CMP:%.*]] = fcmp ogt float [[X:%.*]], [[Y:%.*]]
|
||||||
|
; CHECK-NEXT: br i1 [[CMP]], label [[SAME:%.*]], label [[DIFFERENT:%.*]]
|
||||||
|
; CHECK: same:
|
||||||
|
; CHECK-NEXT: ret i1 false
|
||||||
|
; CHECK: different:
|
||||||
|
; CHECK-NEXT: ret i1 false
|
||||||
|
;
|
||||||
%cmp = fcmp ogt float %x, %y
|
%cmp = fcmp ogt float %x, %y
|
||||||
br i1 %cmp, label %same, label %different
|
br i1 %cmp, label %same, label %different
|
||||||
|
|
||||||
same:
|
same:
|
||||||
%cmp2 = fcmp ule float %x, %y
|
%cmp2 = fcmp ule float %x, %y
|
||||||
; CHECK: ret i1 false
|
|
||||||
ret i1 %cmp2
|
ret i1 %cmp2
|
||||||
|
|
||||||
different:
|
different:
|
||||||
%cmp3 = fcmp ogt float %x, %y
|
%cmp3 = fcmp ogt float %x, %y
|
||||||
; CHECK: ret i1 false
|
|
||||||
ret i1 %cmp3
|
ret i1 %cmp3
|
||||||
}
|
}
|
||||||
|
|
||||||
; PR1768
|
; PR1768
|
||||||
; CHECK-LABEL: @test9(
|
|
||||||
define i32 @test9(i32 %i, i32 %j) {
|
define i32 @test9(i32 %i, i32 %j) {
|
||||||
|
; CHECK-LABEL: @test9(
|
||||||
|
; CHECK-NEXT: [[CMP:%.*]] = icmp eq i32 [[I:%.*]], [[J:%.*]]
|
||||||
|
; CHECK-NEXT: br i1 [[CMP]], label [[COND_TRUE:%.*]], label [[RET:%.*]]
|
||||||
|
; CHECK: cond_true:
|
||||||
|
; CHECK-NEXT: ret i32 0
|
||||||
|
; CHECK: ret:
|
||||||
|
; CHECK-NEXT: ret i32 5
|
||||||
|
;
|
||||||
%cmp = icmp eq i32 %i, %j
|
%cmp = icmp eq i32 %i, %j
|
||||||
br i1 %cmp, label %cond_true, label %ret
|
br i1 %cmp, label %cond_true, label %ret
|
||||||
|
|
||||||
cond_true:
|
cond_true:
|
||||||
%diff = sub i32 %i, %j
|
%diff = sub i32 %i, %j
|
||||||
ret i32 %diff
|
ret i32 %diff
|
||||||
; CHECK: ret i32 0
|
|
||||||
|
|
||||||
ret:
|
ret:
|
||||||
ret i32 5
|
ret i32 5
|
||||||
; CHECK: ret i32 5
|
|
||||||
}
|
}
|
||||||
|
|
||||||
; PR1768
|
; PR1768
|
||||||
; CHECK-LABEL: @test10(
|
|
||||||
define i32 @test10(i32 %j, i32 %i) {
|
define i32 @test10(i32 %j, i32 %i) {
|
||||||
|
; CHECK-LABEL: @test10(
|
||||||
|
; CHECK-NEXT: [[CMP:%.*]] = icmp eq i32 [[I:%.*]], [[J:%.*]]
|
||||||
|
; CHECK-NEXT: br i1 [[CMP]], label [[COND_TRUE:%.*]], label [[RET:%.*]]
|
||||||
|
; CHECK: cond_true:
|
||||||
|
; CHECK-NEXT: ret i32 0
|
||||||
|
; CHECK: ret:
|
||||||
|
; CHECK-NEXT: ret i32 5
|
||||||
|
;
|
||||||
%cmp = icmp eq i32 %i, %j
|
%cmp = icmp eq i32 %i, %j
|
||||||
br i1 %cmp, label %cond_true, label %ret
|
br i1 %cmp, label %cond_true, label %ret
|
||||||
|
|
||||||
cond_true:
|
cond_true:
|
||||||
%diff = sub i32 %i, %j
|
%diff = sub i32 %i, %j
|
||||||
ret i32 %diff
|
ret i32 %diff
|
||||||
; CHECK: ret i32 0
|
|
||||||
|
|
||||||
ret:
|
ret:
|
||||||
ret i32 5
|
ret i32 5
|
||||||
; CHECK: ret i32 5
|
|
||||||
}
|
}
|
||||||
|
|
||||||
declare i32 @yogibar()
|
declare i32 @yogibar()
|
||||||
|
|
||||||
; CHECK-LABEL: @test11(
|
|
||||||
define i32 @test11(i32 %x) {
|
define i32 @test11(i32 %x) {
|
||||||
|
; CHECK-LABEL: @test11(
|
||||||
|
; CHECK-NEXT: [[V0:%.*]] = call i32 @yogibar()
|
||||||
|
; CHECK-NEXT: [[V1:%.*]] = call i32 @yogibar()
|
||||||
|
; CHECK-NEXT: [[CMP:%.*]] = icmp eq i32 [[V0]], [[V1]]
|
||||||
|
; CHECK-NEXT: br i1 [[CMP]], label [[COND_TRUE:%.*]], label [[NEXT:%.*]]
|
||||||
|
; CHECK: cond_true:
|
||||||
|
; CHECK-NEXT: ret i32 [[V0]]
|
||||||
|
; CHECK: next:
|
||||||
|
; CHECK-NEXT: [[CMP2:%.*]] = icmp eq i32 [[X:%.*]], [[V0]]
|
||||||
|
; CHECK-NEXT: br i1 [[CMP2]], label [[COND_TRUE2:%.*]], label [[NEXT2:%.*]]
|
||||||
|
; CHECK: cond_true2:
|
||||||
|
; CHECK-NEXT: ret i32 [[X]]
|
||||||
|
; CHECK: next2:
|
||||||
|
; CHECK-NEXT: ret i32 0
|
||||||
|
;
|
||||||
%v0 = call i32 @yogibar()
|
%v0 = call i32 @yogibar()
|
||||||
%v1 = call i32 @yogibar()
|
%v1 = call i32 @yogibar()
|
||||||
%cmp = icmp eq i32 %v0, %v1
|
%cmp = icmp eq i32 %v0, %v1
|
||||||
|
@ -118,7 +213,6 @@ define i32 @test11(i32 %x) {
|
||||||
|
|
||||||
cond_true:
|
cond_true:
|
||||||
ret i32 %v1
|
ret i32 %v1
|
||||||
; CHECK: ret i32 %v0
|
|
||||||
|
|
||||||
next:
|
next:
|
||||||
%cmp2 = icmp eq i32 %x, %v0
|
%cmp2 = icmp eq i32 %x, %v0
|
||||||
|
@ -126,14 +220,23 @@ next:
|
||||||
|
|
||||||
cond_true2:
|
cond_true2:
|
||||||
ret i32 %v0
|
ret i32 %v0
|
||||||
; CHECK: ret i32 %x
|
|
||||||
|
|
||||||
next2:
|
next2:
|
||||||
ret i32 0
|
ret i32 0
|
||||||
}
|
}
|
||||||
|
|
||||||
; CHECK-LABEL: @test12(
|
|
||||||
define i32 @test12(i32 %x) {
|
define i32 @test12(i32 %x) {
|
||||||
|
; CHECK-LABEL: @test12(
|
||||||
|
; CHECK-NEXT: [[CMP:%.*]] = icmp eq i32 [[X:%.*]], 0
|
||||||
|
; CHECK-NEXT: br i1 [[CMP]], label [[COND_TRUE:%.*]], label [[COND_FALSE:%.*]]
|
||||||
|
; CHECK: cond_true:
|
||||||
|
; CHECK-NEXT: br label [[RET:%.*]]
|
||||||
|
; CHECK: cond_false:
|
||||||
|
; CHECK-NEXT: br label [[RET]]
|
||||||
|
; CHECK: ret:
|
||||||
|
; CHECK-NEXT: [[RES:%.*]] = phi i32 [ 0, [[COND_TRUE]] ], [ [[X]], [[COND_FALSE]] ]
|
||||||
|
; CHECK-NEXT: ret i32 [[RES]]
|
||||||
|
;
|
||||||
%cmp = icmp eq i32 %x, 0
|
%cmp = icmp eq i32 %x, 0
|
||||||
br i1 %cmp, label %cond_true, label %cond_false
|
br i1 %cmp, label %cond_true, label %cond_false
|
||||||
|
|
||||||
|
@ -145,6 +248,5 @@ cond_false:
|
||||||
|
|
||||||
ret:
|
ret:
|
||||||
%res = phi i32 [ %x, %cond_true ], [ %x, %cond_false ]
|
%res = phi i32 [ %x, %cond_true ], [ %x, %cond_false ]
|
||||||
; CHECK: %res = phi i32 [ 0, %cond_true ], [ %x, %cond_false ]
|
|
||||||
ret i32 %res
|
ret i32 %res
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,4 +1,3 @@
|
||||||
; XFAIL: *
|
|
||||||
; RUN: opt -newgvn -S < %s | FileCheck %s
|
; RUN: opt -newgvn -S < %s | FileCheck %s
|
||||||
|
|
||||||
define i32 @f1(i32 %x) {
|
define i32 @f1(i32 %x) {
|
||||||
|
|
|
@ -133,11 +133,18 @@ define void @test4(i1 %b, i32 %x) {
|
||||||
; CHECK-LABEL: @test4(
|
; CHECK-LABEL: @test4(
|
||||||
; CHECK-NEXT: br i1 [[B:%.*]], label [[SW:%.*]], label [[CASE3:%.*]]
|
; CHECK-NEXT: br i1 [[B:%.*]], label [[SW:%.*]], label [[CASE3:%.*]]
|
||||||
; CHECK: sw:
|
; CHECK: sw:
|
||||||
; CHECK-NEXT: switch i32 [[X:%.*]], label [[DEFAULT:%.*]] [
|
; CHECK: i32 0, label [[CASE0:%.*]]
|
||||||
; CHECK-NEXT: i32 0, label [[CASE0:%.*]]
|
|
||||||
; CHECK-NEXT: i32 1, label [[CASE1:%.*]]
|
; CHECK-NEXT: i32 1, label [[CASE1:%.*]]
|
||||||
; CHECK-NEXT: i32 2, label [[CASE0]]
|
; CHECK-NEXT: i32 2, label [[CASE0]]
|
||||||
; CHECK-NEXT: i32 3, label [[CASE3]]
|
; CHECK-NEXT: i32 3, label [[CASE3]]
|
||||||
|
; CHECK-NEXT: i32 4, label [[DEFAULT:%.*]]
|
||||||
|
; CHECK-NEXT: ] Edge: [label [[SW]],label %case1] }
|
||||||
|
; CHECK-NEXT: [[X_0:%.*]] = call i32 @llvm.ssa.copy.i32(i32 [[X:%.*]])
|
||||||
|
; CHECK-NEXT: switch i32 [[X]], label [[DEFAULT]] [
|
||||||
|
; CHECK-NEXT: i32 0, label [[CASE0]]
|
||||||
|
; CHECK-NEXT: i32 1, label [[CASE1]]
|
||||||
|
; CHECK-NEXT: i32 2, label [[CASE0]]
|
||||||
|
; CHECK-NEXT: i32 3, label [[CASE3]]
|
||||||
; CHECK-NEXT: i32 4, label [[DEFAULT]]
|
; CHECK-NEXT: i32 4, label [[DEFAULT]]
|
||||||
; CHECK-NEXT: ]
|
; CHECK-NEXT: ]
|
||||||
; CHECK: default:
|
; CHECK: default:
|
||||||
|
@ -147,7 +154,7 @@ define void @test4(i1 %b, i32 %x) {
|
||||||
; CHECK-NEXT: call void @bar(i32 [[X]])
|
; CHECK-NEXT: call void @bar(i32 [[X]])
|
||||||
; CHECK-NEXT: ret void
|
; CHECK-NEXT: ret void
|
||||||
; CHECK: case1:
|
; CHECK: case1:
|
||||||
; CHECK-NEXT: call void @bar(i32 [[X]])
|
; CHECK-NEXT: call void @bar(i32 [[X_0]])
|
||||||
; CHECK-NEXT: ret void
|
; CHECK-NEXT: ret void
|
||||||
; CHECK: case3:
|
; CHECK: case3:
|
||||||
; CHECK-NEXT: call void @bar(i32 [[X]])
|
; CHECK-NEXT: call void @bar(i32 [[X]])
|
||||||
|
|
|
@ -52,13 +52,14 @@ bb2:
|
||||||
define i32 @f3(i32 %x) {
|
define i32 @f3(i32 %x) {
|
||||||
; CHECK-LABEL: @f3(
|
; CHECK-LABEL: @f3(
|
||||||
; CHECK-NEXT: bb0:
|
; CHECK-NEXT: bb0:
|
||||||
; CHECK-NEXT: switch i32 [[X:%.*]], label [[BB1:%.*]] [
|
; CHECK: [[X_0:%.*]] = call i32 @llvm.ssa.copy.i32(i32 [[X:%.*]])
|
||||||
|
; CHECK-NEXT: switch i32 [[X]], label [[BB1:%.*]] [
|
||||||
; CHECK-NEXT: i32 0, label [[BB2:%.*]]
|
; CHECK-NEXT: i32 0, label [[BB2:%.*]]
|
||||||
; CHECK-NEXT: ]
|
; CHECK-NEXT: ]
|
||||||
; CHECK: bb1:
|
; CHECK: bb1:
|
||||||
; CHECK-NEXT: br label [[BB2]]
|
; CHECK-NEXT: br label [[BB2]]
|
||||||
; CHECK: bb2:
|
; CHECK: bb2:
|
||||||
; CHECK-NEXT: [[COND:%.*]] = phi i32 [ [[X]], [[BB0:%.*]] ], [ 0, [[BB1]] ]
|
; CHECK-NEXT: [[COND:%.*]] = phi i32 [ [[X_0]], [[BB0:%.*]] ], [ 0, [[BB1]] ]
|
||||||
; CHECK-NEXT: [[FOO:%.*]] = add i32 [[COND]], [[X]]
|
; CHECK-NEXT: [[FOO:%.*]] = add i32 [[COND]], [[X]]
|
||||||
; CHECK-NEXT: ret i32 [[FOO]]
|
; CHECK-NEXT: ret i32 [[FOO]]
|
||||||
;
|
;
|
||||||
|
|
Loading…
Reference in New Issue