mirror of
https://github.com/capstone-engine/llvm-capstone.git
synced 2025-01-14 03:45:33 +00:00
[SCEV] Compute affine range in another way to avoid bitwidth extending.
Summary: This approach has two major advantages over the existing one: 1. We don't need to extend bitwidth in our computations. Extending bitwidth is a big issue for compile time as we often end up working with APInts wider than 64bit, which is a slow case for APInt. 2. When we zero extend a wrapped range, we lose some information (we replace the range with [0, 1 << src bit width)). Thus, avoiding such extensions better preserves information. Correctness testing: I ran 'ninja check' with assertions that the new implementation of getRangeForAffineAR gives the same results as the old one (this functionality is not present in this patch). There were several failures - I inspected them manually and found out that they all are caused by the fact that we're returning more accurate results now (see bullet (2) above). Without such assertions 'ninja check' works just fine, as well as SPEC2006. Compile time testing: CTMark/Os: - mafft/pairlocalalign -16.98% - tramp3d-v4/tramp3d-v4 -12.72% - lencod/lencod -11.51% - Bullet/bullet -4.36% - ClamAV/clamscan -3.66% - 7zip/7zip-benchmark -3.19% - sqlite3/sqlite3 -2.95% - SPASS/SPASS -2.74% - Average -5.81% Performance testing: The changes are expected to be neutral for runtime performance. Reviewers: sanjoy, atrick, pete Subscribers: llvm-commits Differential Revision: https://reviews.llvm.org/D30477 llvm-svn: 297992
This commit is contained in:
parent
b879539aac
commit
99de88d1f3
@ -4713,6 +4713,77 @@ ScalarEvolution::getRange(const SCEV *S,
|
||||
return setRange(S, SignHint, ConservativeResult);
|
||||
}
|
||||
|
||||
// Given a StartRange, Step and MaxBECount for an expression compute a range of
|
||||
// values that the expression can take. Initially, the expression has a value
|
||||
// from StartRange and then is changed by Step up to MaxBECount times. Signed
|
||||
// argument defines if we treat Step as signed or unsigned.
|
||||
static ConstantRange getRangeForAffineARHelper(APInt Step,
|
||||
ConstantRange StartRange,
|
||||
APInt MaxBECount,
|
||||
unsigned BitWidth, bool Signed) {
|
||||
// If either Step or MaxBECount is 0, then the expression won't change, and we
|
||||
// just need to return the initial range.
|
||||
if (Step == 0 || MaxBECount == 0)
|
||||
return StartRange;
|
||||
|
||||
// If we don't know anything about the inital value (i.e. StartRange is
|
||||
// FullRange), then we don't know anything about the final range either.
|
||||
// Return FullRange.
|
||||
if (StartRange.isFullSet())
|
||||
return ConstantRange(BitWidth, /* isFullSet = */ true);
|
||||
|
||||
// If Step is signed and negative, then we use its absolute value, but we also
|
||||
// note that we're moving in the opposite direction.
|
||||
bool Descending = Signed && Step.isNegative();
|
||||
|
||||
if (Signed)
|
||||
// This is correct even for INT_SMIN. Let's look at i8 to illustrate this:
|
||||
// abs(INT_SMIN) = abs(-128) = abs(0x80) = -0x80 = 0x80 = 128.
|
||||
// This equations hold true due to the well-defined wrap-around behavior of
|
||||
// APInt.
|
||||
Step = Step.abs();
|
||||
|
||||
// Check if Offset is more than full span of BitWidth. If it is, the
|
||||
// expression is guaranteed to overflow.
|
||||
if (APInt::getMaxValue(StartRange.getBitWidth()).udiv(Step).ult(MaxBECount))
|
||||
return ConstantRange(BitWidth, /* isFullSet = */ true);
|
||||
|
||||
// Offset is by how much the expression can change. Checks above guarantee no
|
||||
// overflow here.
|
||||
APInt Offset = Step * MaxBECount;
|
||||
|
||||
// Minimum value of the final range will match the minimal value of StartRange
|
||||
// if the expression is increasing and will be decreased by Offset otherwise.
|
||||
// Maximum value of the final range will match the maximal value of StartRange
|
||||
// if the expression is decreasing and will be increased by Offset otherwise.
|
||||
APInt StartLower = StartRange.getLower();
|
||||
APInt StartUpper = StartRange.getUpper() - 1;
|
||||
APInt MovedBoundary =
|
||||
Descending ? (StartLower - Offset) : (StartUpper + Offset);
|
||||
|
||||
// It's possible that the new minimum/maximum value will fall into the initial
|
||||
// range (due to wrap around). This means that the expression can take any
|
||||
// value in this bitwidth, and we have to return full range.
|
||||
if (StartRange.contains(MovedBoundary))
|
||||
return ConstantRange(BitWidth, /* isFullSet = */ true);
|
||||
|
||||
APInt NewLower, NewUpper;
|
||||
if (Descending) {
|
||||
NewLower = MovedBoundary;
|
||||
NewUpper = StartUpper;
|
||||
} else {
|
||||
NewLower = StartLower;
|
||||
NewUpper = MovedBoundary;
|
||||
}
|
||||
|
||||
// If we end up with full range, return a proper full range.
|
||||
if (NewLower == NewUpper + 1)
|
||||
return ConstantRange(BitWidth, /* isFullSet = */ true);
|
||||
|
||||
// No overflow detected, return [StartLower, StartUpper + Offset + 1) range.
|
||||
return ConstantRange(NewLower, NewUpper + 1);
|
||||
}
|
||||
|
||||
ConstantRange ScalarEvolution::getRangeForAffineAR(const SCEV *Start,
|
||||
const SCEV *Step,
|
||||
const SCEV *MaxBECount,
|
||||
@ -4721,60 +4792,30 @@ ConstantRange ScalarEvolution::getRangeForAffineAR(const SCEV *Start,
|
||||
getTypeSizeInBits(MaxBECount->getType()) <= BitWidth &&
|
||||
"Precondition!");
|
||||
|
||||
ConstantRange Result(BitWidth, /* isFullSet = */ true);
|
||||
|
||||
// Check for overflow. This must be done with ConstantRange arithmetic
|
||||
// because we could be called from within the ScalarEvolution overflow
|
||||
// checking code.
|
||||
|
||||
MaxBECount = getNoopOrZeroExtend(MaxBECount, Start->getType());
|
||||
ConstantRange MaxBECountRange = getUnsignedRange(MaxBECount);
|
||||
ConstantRange ZExtMaxBECountRange = MaxBECountRange.zextOrTrunc(BitWidth * 2);
|
||||
|
||||
ConstantRange StepSRange = getSignedRange(Step);
|
||||
ConstantRange SExtStepSRange = StepSRange.sextOrTrunc(BitWidth * 2);
|
||||
|
||||
ConstantRange StartURange = getUnsignedRange(Start);
|
||||
ConstantRange EndURange =
|
||||
StartURange.add(MaxBECountRange.multiply(StepSRange));
|
||||
|
||||
// Check for unsigned overflow.
|
||||
ConstantRange ZExtStartURange = StartURange.zextOrTrunc(BitWidth * 2);
|
||||
ConstantRange ZExtEndURange = EndURange.zextOrTrunc(BitWidth * 2);
|
||||
if (ZExtStartURange.add(ZExtMaxBECountRange.multiply(SExtStepSRange)) ==
|
||||
ZExtEndURange) {
|
||||
APInt Min = APIntOps::umin(StartURange.getUnsignedMin(),
|
||||
EndURange.getUnsignedMin());
|
||||
APInt Max = APIntOps::umax(StartURange.getUnsignedMax(),
|
||||
EndURange.getUnsignedMax());
|
||||
bool IsFullRange = Min.isMinValue() && Max.isMaxValue();
|
||||
if (!IsFullRange)
|
||||
Result =
|
||||
Result.intersectWith(ConstantRange(Min, Max + 1));
|
||||
}
|
||||
APInt MaxBECountValue = MaxBECountRange.getUnsignedMax();
|
||||
|
||||
// First, consider step signed.
|
||||
ConstantRange StartSRange = getSignedRange(Start);
|
||||
ConstantRange EndSRange =
|
||||
StartSRange.add(MaxBECountRange.multiply(StepSRange));
|
||||
ConstantRange StepSRange = getSignedRange(Step);
|
||||
|
||||
// Check for signed overflow. This must be done with ConstantRange
|
||||
// arithmetic because we could be called from within the ScalarEvolution
|
||||
// overflow checking code.
|
||||
ConstantRange SExtStartSRange = StartSRange.sextOrTrunc(BitWidth * 2);
|
||||
ConstantRange SExtEndSRange = EndSRange.sextOrTrunc(BitWidth * 2);
|
||||
if (SExtStartSRange.add(ZExtMaxBECountRange.multiply(SExtStepSRange)) ==
|
||||
SExtEndSRange) {
|
||||
APInt Min =
|
||||
APIntOps::smin(StartSRange.getSignedMin(), EndSRange.getSignedMin());
|
||||
APInt Max =
|
||||
APIntOps::smax(StartSRange.getSignedMax(), EndSRange.getSignedMax());
|
||||
bool IsFullRange = Min.isMinSignedValue() && Max.isMaxSignedValue();
|
||||
if (!IsFullRange)
|
||||
Result =
|
||||
Result.intersectWith(ConstantRange(Min, Max + 1));
|
||||
}
|
||||
// If Step can be both positive and negative, we need to find ranges for the
|
||||
// maximum absolute step values in both directions and union them.
|
||||
ConstantRange SR =
|
||||
getRangeForAffineARHelper(StepSRange.getSignedMin(), StartSRange,
|
||||
MaxBECountValue, BitWidth, /* Signed = */ true);
|
||||
SR = SR.unionWith(getRangeForAffineARHelper(StepSRange.getSignedMax(),
|
||||
StartSRange, MaxBECountValue,
|
||||
BitWidth, /* Signed = */ true));
|
||||
|
||||
return Result;
|
||||
// Next, consider step unsigned.
|
||||
ConstantRange UR = getRangeForAffineARHelper(
|
||||
getUnsignedRange(Step).getUnsignedMax(), getUnsignedRange(Start),
|
||||
MaxBECountValue, BitWidth, /* Signed = */ false);
|
||||
|
||||
// Finally, intersect signed and unsigned ranges.
|
||||
return SR.intersectWith(UR);
|
||||
}
|
||||
|
||||
ConstantRange ScalarEvolution::getRangeViaFactoring(const SCEV *Start,
|
||||
|
@ -6,6 +6,10 @@ entry:
|
||||
br label %bb.i
|
||||
|
||||
bb.i: ; preds = %bb1.i, %bb.nph
|
||||
; We should be able to find the range for this expression.
|
||||
; CHECK: %l_95.0.i1 = phi i8
|
||||
; CHECK: --> {0,+,-1}<%bb.i> U: [2,1) S: [2,1){{ *}}Exits: 2
|
||||
|
||||
%l_95.0.i1 = phi i8 [ %tmp1, %bb.i ], [ 0, %entry ]
|
||||
|
||||
; This cast shouldn't be folded into the addrec.
|
||||
|
Loading…
x
Reference in New Issue
Block a user