mirror of
https://github.com/c64scene-ar/llvm-6502.git
synced 2024-12-14 11:32:34 +00:00
Clean up some of LVI:
* mergeIn now uses constant folding for constants that are provably not-equal. * sink some sanity checks from the get*() methods into the mark*() methods, to ensure that we never have a constant/notconstant ConstantInt * some textual cleanups, whitespace changes, removing "else" after return, that sort of thing. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@121877 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
parent
41b1d4e472
commit
69bfdf5a24
@ -52,18 +52,18 @@ namespace llvm {
|
|||||||
namespace {
|
namespace {
|
||||||
class LVILatticeVal {
|
class LVILatticeVal {
|
||||||
enum LatticeValueTy {
|
enum LatticeValueTy {
|
||||||
/// undefined - This LLVM Value has no known value yet.
|
/// undefined - This Value has no known value yet.
|
||||||
undefined,
|
undefined,
|
||||||
|
|
||||||
/// constant - This LLVM Value has a specific constant value.
|
/// constant - This Value has a specific constant value.
|
||||||
constant,
|
constant,
|
||||||
/// notconstant - This LLVM value is known to not have the specified value.
|
/// notconstant - This Value is known to not have the specified value.
|
||||||
notconstant,
|
notconstant,
|
||||||
|
|
||||||
/// constantrange
|
/// constantrange - The Value falls within this range.
|
||||||
constantrange,
|
constantrange,
|
||||||
|
|
||||||
/// overdefined - This instruction is not known to be constant, and we know
|
/// overdefined - This value is not known to be constant, and we know that
|
||||||
/// it has a value.
|
/// it has a value.
|
||||||
overdefined
|
overdefined
|
||||||
};
|
};
|
||||||
@ -79,17 +79,13 @@ public:
|
|||||||
|
|
||||||
static LVILatticeVal get(Constant *C) {
|
static LVILatticeVal get(Constant *C) {
|
||||||
LVILatticeVal Res;
|
LVILatticeVal Res;
|
||||||
if (ConstantInt *CI = dyn_cast<ConstantInt>(C))
|
if (!isa<UndefValue>(C))
|
||||||
Res.markConstantRange(ConstantRange(CI->getValue(), CI->getValue()+1));
|
|
||||||
else if (!isa<UndefValue>(C))
|
|
||||||
Res.markConstant(C);
|
Res.markConstant(C);
|
||||||
return Res;
|
return Res;
|
||||||
}
|
}
|
||||||
static LVILatticeVal getNot(Constant *C) {
|
static LVILatticeVal getNot(Constant *C) {
|
||||||
LVILatticeVal Res;
|
LVILatticeVal Res;
|
||||||
if (ConstantInt *CI = dyn_cast<ConstantInt>(C))
|
if (!isa<UndefValue>(C))
|
||||||
Res.markConstantRange(ConstantRange(CI->getValue()+1, CI->getValue()));
|
|
||||||
else
|
|
||||||
Res.markNotConstant(C);
|
Res.markNotConstant(C);
|
||||||
return Res;
|
return Res;
|
||||||
}
|
}
|
||||||
@ -131,32 +127,34 @@ public:
|
|||||||
|
|
||||||
/// markConstant - Return true if this is a change in status.
|
/// markConstant - Return true if this is a change in status.
|
||||||
bool markConstant(Constant *V) {
|
bool markConstant(Constant *V) {
|
||||||
if (isConstant()) {
|
assert(V && "Marking constant with NULL");
|
||||||
assert(getConstant() == V && "Marking constant with different value");
|
if (ConstantInt *CI = dyn_cast<ConstantInt>(V))
|
||||||
|
return markConstantRange(ConstantRange(CI->getValue()));
|
||||||
|
if (isa<UndefValue>(V))
|
||||||
return false;
|
return false;
|
||||||
}
|
|
||||||
|
assert((!isConstant() || getConstant() == V) &&
|
||||||
|
"Marking constant with different value");
|
||||||
assert(isUndefined());
|
assert(isUndefined());
|
||||||
Tag = constant;
|
Tag = constant;
|
||||||
assert(V && "Marking constant with NULL");
|
|
||||||
Val = V;
|
Val = V;
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// markNotConstant - Return true if this is a change in status.
|
/// markNotConstant - Return true if this is a change in status.
|
||||||
bool markNotConstant(Constant *V) {
|
bool markNotConstant(Constant *V) {
|
||||||
if (isNotConstant()) {
|
|
||||||
assert(getNotConstant() == V && "Marking !constant with different value");
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (isConstant())
|
|
||||||
assert(getConstant() != V && "Marking not constant with different value");
|
|
||||||
else
|
|
||||||
assert(isUndefined());
|
|
||||||
|
|
||||||
Tag = notconstant;
|
|
||||||
assert(V && "Marking constant with NULL");
|
assert(V && "Marking constant with NULL");
|
||||||
|
if (ConstantInt *CI = dyn_cast<ConstantInt>(V))
|
||||||
|
return markConstantRange(ConstantRange(CI->getValue()+1, CI->getValue()));
|
||||||
|
if (isa<UndefValue>(V))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
assert((!isConstant() || getConstant() != V) &&
|
||||||
|
"Marking constant !constant with same value");
|
||||||
|
assert((!isNotConstant() || getNotConstant() == V) &&
|
||||||
|
"Marking !constant with different value");
|
||||||
|
assert(isUndefined() || isConstant());
|
||||||
|
Tag = notconstant;
|
||||||
Val = V;
|
Val = V;
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
@ -187,67 +185,81 @@ public:
|
|||||||
if (RHS.isUndefined() || isOverdefined()) return false;
|
if (RHS.isUndefined() || isOverdefined()) return false;
|
||||||
if (RHS.isOverdefined()) return markOverdefined();
|
if (RHS.isOverdefined()) return markOverdefined();
|
||||||
|
|
||||||
if (RHS.isNotConstant()) {
|
if (isUndefined()) {
|
||||||
if (isNotConstant()) {
|
Tag = RHS.Tag;
|
||||||
if (getNotConstant() != RHS.getNotConstant() ||
|
Val = RHS.Val;
|
||||||
isa<ConstantExpr>(getNotConstant()) ||
|
Range = RHS.Range;
|
||||||
isa<ConstantExpr>(RHS.getNotConstant()))
|
return true;
|
||||||
return markOverdefined();
|
|
||||||
return false;
|
|
||||||
} else if (isConstant()) {
|
|
||||||
if (getConstant() == RHS.getNotConstant() ||
|
|
||||||
isa<ConstantExpr>(RHS.getNotConstant()) ||
|
|
||||||
isa<ConstantExpr>(getConstant()))
|
|
||||||
return markOverdefined();
|
|
||||||
return markNotConstant(RHS.getNotConstant());
|
|
||||||
} else if (isConstantRange()) {
|
|
||||||
// FIXME: This could be made more precise.
|
|
||||||
return markOverdefined();
|
|
||||||
}
|
|
||||||
|
|
||||||
assert(isUndefined() && "Unexpected lattice");
|
|
||||||
return markNotConstant(RHS.getNotConstant());
|
|
||||||
}
|
|
||||||
|
|
||||||
if (RHS.isConstantRange()) {
|
|
||||||
if (isConstantRange()) {
|
|
||||||
ConstantRange NewR = Range.unionWith(RHS.getConstantRange());
|
|
||||||
if (NewR.isFullSet())
|
|
||||||
return markOverdefined();
|
|
||||||
else
|
|
||||||
return markConstantRange(NewR);
|
|
||||||
} else if (!isUndefined()) {
|
|
||||||
return markOverdefined();
|
|
||||||
}
|
|
||||||
|
|
||||||
assert(isUndefined() && "Unexpected lattice");
|
|
||||||
return markConstantRange(RHS.getConstantRange());
|
|
||||||
}
|
|
||||||
|
|
||||||
// RHS must be a constant, we must be constantrange,
|
|
||||||
// undef, constant, or notconstant.
|
|
||||||
if (isConstantRange()) {
|
|
||||||
// FIXME: This could be made more precise.
|
|
||||||
return markOverdefined();
|
|
||||||
}
|
|
||||||
|
|
||||||
if (isUndefined())
|
|
||||||
return markConstant(RHS.getConstant());
|
|
||||||
|
|
||||||
if (isConstant()) {
|
|
||||||
if (getConstant() != RHS.getConstant())
|
|
||||||
return markOverdefined();
|
|
||||||
return false;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// If we are known "!=4" and RHS is "==5", stay at "!=4".
|
if (isConstant()) {
|
||||||
if (getNotConstant() == RHS.getConstant() ||
|
if (RHS.isConstant()) {
|
||||||
isa<ConstantExpr>(getNotConstant()) ||
|
if (Val == RHS.Val)
|
||||||
isa<ConstantExpr>(RHS.getConstant()))
|
return false;
|
||||||
|
return markOverdefined();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (RHS.isNotConstant()) {
|
||||||
|
if (Val == RHS.Val)
|
||||||
|
return markOverdefined();
|
||||||
|
|
||||||
|
// Unless we can prove that the two Constants are different, we must
|
||||||
|
// move to overdefined.
|
||||||
|
// FIXME: use TargetData for smarter constant folding.
|
||||||
|
if (ConstantInt *Res = dyn_cast<ConstantInt>(
|
||||||
|
ConstantFoldCompareInstOperands(CmpInst::ICMP_NE,
|
||||||
|
getConstant(),
|
||||||
|
RHS.getNotConstant())))
|
||||||
|
if (Res->isOne())
|
||||||
|
return markNotConstant(RHS.getNotConstant());
|
||||||
|
|
||||||
|
return markOverdefined();
|
||||||
|
}
|
||||||
|
|
||||||
|
// RHS is a ConstantRange, LHS is a non-integer Constant.
|
||||||
|
|
||||||
|
// FIXME: consider the case where RHS is a range [1, 0) and LHS is
|
||||||
|
// a function. The correct result is to pick up RHS.
|
||||||
|
|
||||||
return markOverdefined();
|
return markOverdefined();
|
||||||
return false;
|
}
|
||||||
|
|
||||||
|
if (isNotConstant()) {
|
||||||
|
if (RHS.isConstant()) {
|
||||||
|
if (Val == RHS.Val)
|
||||||
|
return markOverdefined();
|
||||||
|
|
||||||
|
// Unless we can prove that the two Constants are different, we must
|
||||||
|
// move to overdefined.
|
||||||
|
// FIXME: use TargetData for smarter constant folding.
|
||||||
|
if (ConstantInt *Res = dyn_cast<ConstantInt>(
|
||||||
|
ConstantFoldCompareInstOperands(CmpInst::ICMP_NE,
|
||||||
|
getNotConstant(),
|
||||||
|
RHS.getConstant())))
|
||||||
|
if (Res->isOne())
|
||||||
|
return false;
|
||||||
|
|
||||||
|
return markOverdefined();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (RHS.isNotConstant()) {
|
||||||
|
if (Val == RHS.Val)
|
||||||
|
return false;
|
||||||
|
return markOverdefined();
|
||||||
|
}
|
||||||
|
|
||||||
|
return markOverdefined();
|
||||||
|
}
|
||||||
|
|
||||||
|
assert(isConstantRange() && "New LVILattice type?");
|
||||||
|
if (!RHS.isConstantRange())
|
||||||
|
return markOverdefined();
|
||||||
|
|
||||||
|
ConstantRange NewR = Range.unionWith(RHS.getConstantRange());
|
||||||
|
if (NewR.isFullSet())
|
||||||
|
return markOverdefined();
|
||||||
|
return markConstantRange(NewR);
|
||||||
}
|
}
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
} // end anonymous namespace.
|
} // end anonymous namespace.
|
||||||
@ -292,7 +304,7 @@ namespace {
|
|||||||
: CallbackVH(V), Parent(P) { }
|
: CallbackVH(V), Parent(P) { }
|
||||||
|
|
||||||
void deleted();
|
void deleted();
|
||||||
void allUsesReplacedWith(Value* V) {
|
void allUsesReplacedWith(Value *V) {
|
||||||
deleted();
|
deleted();
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
@ -310,7 +322,7 @@ namespace {
|
|||||||
LVILatticeVal getBlockValue(Value *Val, BasicBlock *BB);
|
LVILatticeVal getBlockValue(Value *Val, BasicBlock *BB);
|
||||||
LVILatticeVal getEdgeValue(Value *V, BasicBlock *F, BasicBlock *T);
|
LVILatticeVal getEdgeValue(Value *V, BasicBlock *F, BasicBlock *T);
|
||||||
|
|
||||||
ValueCacheEntryTy &lookup(Value *V) {
|
ValueCacheEntryTy &lookup(Value *V) {
|
||||||
return ValueCache[LVIValueHandle(V, this)];
|
return ValueCache[LVIValueHandle(V, this)];
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -321,7 +333,6 @@ namespace {
|
|||||||
}
|
}
|
||||||
|
|
||||||
public:
|
public:
|
||||||
|
|
||||||
/// getValueInBlock - This is the query interface to determine the lattice
|
/// getValueInBlock - This is the query interface to determine the lattice
|
||||||
/// value for the specified Value* at the end of the specified block.
|
/// value for the specified Value* at the end of the specified block.
|
||||||
LVILatticeVal getValueInBlock(Value *V, BasicBlock *BB);
|
LVILatticeVal getValueInBlock(Value *V, BasicBlock *BB);
|
||||||
@ -456,7 +467,7 @@ LVILatticeVal LazyValueInfoCache::getBlockValue(Value *Val, BasicBlock *BB) {
|
|||||||
// Loop over all of our predecessors, merging what we know from them into
|
// Loop over all of our predecessors, merging what we know from them into
|
||||||
// result.
|
// result.
|
||||||
for (pred_iterator PI = pred_begin(BB), E = pred_end(BB); PI != E; ++PI) {
|
for (pred_iterator PI = pred_begin(BB), E = pred_end(BB); PI != E; ++PI) {
|
||||||
Value* PhiVal = PN->getIncomingValueForBlock(*PI);
|
Value *PhiVal = PN->getIncomingValueForBlock(*PI);
|
||||||
Result.mergeIn(getValueOnEdge(PhiVal, *PI, BB));
|
Result.mergeIn(getValueOnEdge(PhiVal, *PI, BB));
|
||||||
|
|
||||||
// If we hit overdefined, exit early. The BlockVals entry is already set
|
// If we hit overdefined, exit early. The BlockVals entry is already set
|
||||||
@ -574,8 +585,8 @@ LVILatticeVal LazyValueInfoCache::getBlockValue(Value *Val, BasicBlock *BB) {
|
|||||||
|
|
||||||
/// getEdgeValue - This method attempts to infer more complex
|
/// getEdgeValue - This method attempts to infer more complex
|
||||||
LVILatticeVal LazyValueInfoCache::getEdgeValue(Value *Val,
|
LVILatticeVal LazyValueInfoCache::getEdgeValue(Value *Val,
|
||||||
BasicBlock *BBFrom,
|
BasicBlock *BBFrom,
|
||||||
BasicBlock *BBTo) {
|
BasicBlock *BBTo) {
|
||||||
// TODO: Handle more complex conditionals. If (v == 0 || v2 < 1) is false, we
|
// TODO: Handle more complex conditionals. If (v == 0 || v2 < 1) is false, we
|
||||||
// know that v != 0.
|
// know that v != 0.
|
||||||
if (BranchInst *BI = dyn_cast<BranchInst>(BBFrom->getTerminator())) {
|
if (BranchInst *BI = dyn_cast<BranchInst>(BBFrom->getTerminator())) {
|
||||||
@ -725,7 +736,7 @@ void LazyValueInfoCache::threadEdge(BasicBlock *PredBB, BasicBlock *OldSucc,
|
|||||||
if (ToUpdate == NewSucc) continue;
|
if (ToUpdate == NewSucc) continue;
|
||||||
|
|
||||||
bool changed = false;
|
bool changed = false;
|
||||||
for (DenseSet<Value*>::iterator I = ClearSet.begin(),E = ClearSet.end();
|
for (DenseSet<Value*>::iterator I = ClearSet.begin(), E = ClearSet.end();
|
||||||
I != E; ++I) {
|
I != E; ++I) {
|
||||||
// If a value was marked overdefined in OldSucc, and is here too...
|
// If a value was marked overdefined in OldSucc, and is here too...
|
||||||
std::set<std::pair<AssertingVH<BasicBlock>, Value*> >::iterator OI =
|
std::set<std::pair<AssertingVH<BasicBlock>, Value*> >::iterator OI =
|
||||||
@ -784,7 +795,7 @@ Constant *LazyValueInfo::getConstant(Value *V, BasicBlock *BB) {
|
|||||||
|
|
||||||
if (Result.isConstant())
|
if (Result.isConstant())
|
||||||
return Result.getConstant();
|
return Result.getConstant();
|
||||||
else if (Result.isConstantRange()) {
|
if (Result.isConstantRange()) {
|
||||||
ConstantRange CR = Result.getConstantRange();
|
ConstantRange CR = Result.getConstantRange();
|
||||||
if (const APInt *SingleVal = CR.getSingleElement())
|
if (const APInt *SingleVal = CR.getSingleElement())
|
||||||
return ConstantInt::get(V->getContext(), *SingleVal);
|
return ConstantInt::get(V->getContext(), *SingleVal);
|
||||||
@ -800,7 +811,7 @@ Constant *LazyValueInfo::getConstantOnEdge(Value *V, BasicBlock *FromBB,
|
|||||||
|
|
||||||
if (Result.isConstant())
|
if (Result.isConstant())
|
||||||
return Result.getConstant();
|
return Result.getConstant();
|
||||||
else if (Result.isConstantRange()) {
|
if (Result.isConstantRange()) {
|
||||||
ConstantRange CR = Result.getConstantRange();
|
ConstantRange CR = Result.getConstantRange();
|
||||||
if (const APInt *SingleVal = CR.getSingleElement())
|
if (const APInt *SingleVal = CR.getSingleElement())
|
||||||
return ConstantInt::get(V->getContext(), *SingleVal);
|
return ConstantInt::get(V->getContext(), *SingleVal);
|
||||||
@ -820,7 +831,7 @@ LazyValueInfo::getPredicateOnEdge(unsigned Pred, Value *V, Constant *C,
|
|||||||
Constant *Res = 0;
|
Constant *Res = 0;
|
||||||
if (Result.isConstant()) {
|
if (Result.isConstant()) {
|
||||||
Res = ConstantFoldCompareInstOperands(Pred, Result.getConstant(), C, TD);
|
Res = ConstantFoldCompareInstOperands(Pred, Result.getConstant(), C, TD);
|
||||||
if (ConstantInt *ResCI = dyn_cast_or_null<ConstantInt>(Res))
|
if (ConstantInt *ResCI = dyn_cast<ConstantInt>(Res))
|
||||||
return ResCI->isZero() ? False : True;
|
return ResCI->isZero() ? False : True;
|
||||||
return Unknown;
|
return Unknown;
|
||||||
}
|
}
|
||||||
@ -845,13 +856,12 @@ LazyValueInfo::getPredicateOnEdge(unsigned Pred, Value *V, Constant *C,
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Handle more complex predicates.
|
// Handle more complex predicates.
|
||||||
ConstantRange RHS(CI->getValue(), CI->getValue()+1);
|
ConstantRange TrueValues =
|
||||||
ConstantRange TrueValues = ConstantRange::makeICmpRegion(Pred, RHS);
|
ICmpInst::makeConstantRange((ICmpInst::Predicate)Pred, CI->getValue());
|
||||||
if (CR.intersectWith(TrueValues).isEmptySet())
|
if (TrueValues.contains(CR))
|
||||||
return False;
|
|
||||||
else if (TrueValues.contains(CR))
|
|
||||||
return True;
|
return True;
|
||||||
|
if (TrueValues.inverse().contains(CR))
|
||||||
|
return False;
|
||||||
return Unknown;
|
return Unknown;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -878,7 +888,7 @@ LazyValueInfo::getPredicateOnEdge(unsigned Pred, Value *V, Constant *C,
|
|||||||
}
|
}
|
||||||
|
|
||||||
void LazyValueInfo::threadEdge(BasicBlock *PredBB, BasicBlock *OldSucc,
|
void LazyValueInfo::threadEdge(BasicBlock *PredBB, BasicBlock *OldSucc,
|
||||||
BasicBlock* NewSucc) {
|
BasicBlock *NewSucc) {
|
||||||
if (PImpl) getCache(PImpl).threadEdge(PredBB, OldSucc, NewSucc);
|
if (PImpl) getCache(PImpl).threadEdge(PredBB, OldSucc, NewSucc);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user