forked from OSchip/llvm-project
rearrange some code, no functionality change.
llvm-svn: 123136
This commit is contained in:
parent
d011d5317c
commit
dff679f4b6
|
@ -4807,47 +4807,14 @@ ScalarEvolution::HowFarToZero(const SCEV *V, const Loop *L) {
|
||||||
if (!AddRec || AddRec->getLoop() != L)
|
if (!AddRec || AddRec->getLoop() != L)
|
||||||
return getCouldNotCompute();
|
return getCouldNotCompute();
|
||||||
|
|
||||||
if (AddRec->isAffine()) {
|
// If this is a quadratic (3-term) AddRec {L,+,M,+,N}, find the roots of
|
||||||
// If this is an affine expression, the execution count of this branch is
|
// the quadratic equation to solve it.
|
||||||
// the minimum unsigned root of the following equation:
|
if (AddRec->isQuadratic() && AddRec->getType()->isIntegerTy()) {
|
||||||
//
|
std::pair<const SCEV *,const SCEV *> Roots =
|
||||||
// Start + Step*N = 0 (mod 2^BW)
|
SolveQuadraticEquation(AddRec, *this);
|
||||||
//
|
|
||||||
// equivalent to:
|
|
||||||
//
|
|
||||||
// Step*N = -Start (mod 2^BW)
|
|
||||||
//
|
|
||||||
// where BW is the common bit width of Start and Step.
|
|
||||||
|
|
||||||
// Get the initial value for the loop.
|
|
||||||
const SCEV *Start = getSCEVAtScope(AddRec->getStart(),
|
|
||||||
L->getParentLoop());
|
|
||||||
const SCEV *Step = getSCEVAtScope(AddRec->getOperand(1),
|
|
||||||
L->getParentLoop());
|
|
||||||
|
|
||||||
if (const SCEVConstant *StepC = dyn_cast<SCEVConstant>(Step)) {
|
|
||||||
// For now we handle only constant steps.
|
|
||||||
|
|
||||||
// First, handle unitary steps.
|
|
||||||
if (StepC->getValue()->equalsInt(1)) // 1*N = -Start (mod 2^BW), so:
|
|
||||||
return getNegativeSCEV(Start); // N = -Start (as unsigned)
|
|
||||||
if (StepC->getValue()->isAllOnesValue()) // -1*N = -Start (mod 2^BW), so:
|
|
||||||
return Start; // N = Start (as unsigned)
|
|
||||||
|
|
||||||
// Then, try to solve the above equation provided that Start is constant.
|
|
||||||
if (const SCEVConstant *StartC = dyn_cast<SCEVConstant>(Start))
|
|
||||||
return SolveLinEquationWithOverflow(StepC->getValue()->getValue(),
|
|
||||||
-StartC->getValue()->getValue(),
|
|
||||||
*this);
|
|
||||||
}
|
|
||||||
} else if (AddRec->isQuadratic() && AddRec->getType()->isIntegerTy()) {
|
|
||||||
// If this is a quadratic (3-term) AddRec {L,+,M,+,N}, find the roots of
|
|
||||||
// the quadratic equation to solve it.
|
|
||||||
std::pair<const SCEV *,const SCEV *> Roots = SolveQuadraticEquation(AddRec,
|
|
||||||
*this);
|
|
||||||
const SCEVConstant *R1 = dyn_cast<SCEVConstant>(Roots.first);
|
const SCEVConstant *R1 = dyn_cast<SCEVConstant>(Roots.first);
|
||||||
const SCEVConstant *R2 = dyn_cast<SCEVConstant>(Roots.second);
|
const SCEVConstant *R2 = dyn_cast<SCEVConstant>(Roots.second);
|
||||||
if (R1) {
|
if (R1 && R2) {
|
||||||
#if 0
|
#if 0
|
||||||
dbgs() << "HFTZ: " << *V << " - sol#1: " << *R1
|
dbgs() << "HFTZ: " << *V << " - sol#1: " << *R1
|
||||||
<< " sol#2: " << *R2 << "\n";
|
<< " sol#2: " << *R2 << "\n";
|
||||||
|
@ -4855,10 +4822,10 @@ ScalarEvolution::HowFarToZero(const SCEV *V, const Loop *L) {
|
||||||
// Pick the smallest positive root value.
|
// Pick the smallest positive root value.
|
||||||
if (ConstantInt *CB =
|
if (ConstantInt *CB =
|
||||||
dyn_cast<ConstantInt>(ConstantExpr::getICmp(ICmpInst::ICMP_ULT,
|
dyn_cast<ConstantInt>(ConstantExpr::getICmp(ICmpInst::ICMP_ULT,
|
||||||
R1->getValue(), R2->getValue()))) {
|
R1->getValue(), R2->getValue()))) {
|
||||||
if (CB->getZExtValue() == false)
|
if (CB->getZExtValue() == false)
|
||||||
std::swap(R1, R2); // R1 is the minimum root now.
|
std::swap(R1, R2); // R1 is the minimum root now.
|
||||||
|
|
||||||
// We can only use this value if the chrec ends up with an exact zero
|
// We can only use this value if the chrec ends up with an exact zero
|
||||||
// value at this index. When solving for "X*X != 5", for example, we
|
// value at this index. When solving for "X*X != 5", for example, we
|
||||||
// should not accept a root of 2.
|
// should not accept a root of 2.
|
||||||
|
@ -4867,8 +4834,45 @@ ScalarEvolution::HowFarToZero(const SCEV *V, const Loop *L) {
|
||||||
return R1; // We found a quadratic root!
|
return R1; // We found a quadratic root!
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
return getCouldNotCompute();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Otherwise we can only handle this if it is affine.
|
||||||
|
if (!AddRec->isAffine())
|
||||||
|
return getCouldNotCompute();
|
||||||
|
|
||||||
|
// If this is an affine expression, the execution count of this branch is
|
||||||
|
// the minimum unsigned root of the following equation:
|
||||||
|
//
|
||||||
|
// Start + Step*N = 0 (mod 2^BW)
|
||||||
|
//
|
||||||
|
// equivalent to:
|
||||||
|
//
|
||||||
|
// Step*N = -Start (mod 2^BW)
|
||||||
|
//
|
||||||
|
// where BW is the common bit width of Start and Step.
|
||||||
|
|
||||||
|
// Get the initial value for the loop.
|
||||||
|
const SCEV *Start = getSCEVAtScope(AddRec->getStart(), L->getParentLoop());
|
||||||
|
const SCEV *Step = getSCEVAtScope(AddRec->getOperand(1), L->getParentLoop());
|
||||||
|
|
||||||
|
// For now we handle only constant steps.
|
||||||
|
const SCEVConstant *StepC = dyn_cast<SCEVConstant>(Step);
|
||||||
|
if (StepC == 0)
|
||||||
|
return getCouldNotCompute();
|
||||||
|
|
||||||
|
// First, handle unitary steps.
|
||||||
|
if (StepC->getValue()->equalsInt(1)) // 1*N = -Start (mod 2^BW), so:
|
||||||
|
return getNegativeSCEV(Start); // N = -Start (as unsigned)
|
||||||
|
|
||||||
|
if (StepC->getValue()->isAllOnesValue()) // -1*N = -Start (mod 2^BW), so:
|
||||||
|
return Start; // N = Start (as unsigned)
|
||||||
|
|
||||||
|
// Then, try to solve the above equation provided that Start is constant.
|
||||||
|
if (const SCEVConstant *StartC = dyn_cast<SCEVConstant>(Start))
|
||||||
|
return SolveLinEquationWithOverflow(StepC->getValue()->getValue(),
|
||||||
|
-StartC->getValue()->getValue(),
|
||||||
|
*this);
|
||||||
return getCouldNotCompute();
|
return getCouldNotCompute();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue