BFI: Rename UnsignedFloat => ScaledNumber

A lot of the docs and API are out of date, but I'll leave that for a
separate commit.

git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@211555 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
Duncan P. N. Exon Smith 2014-06-23 23:36:17 +00:00
parent 67886a98a2
commit 7c21d709a3
2 changed files with 111 additions and 107 deletions

View File

@ -33,14 +33,14 @@
//===----------------------------------------------------------------------===// //===----------------------------------------------------------------------===//
// //
// UnsignedFloat definition. // ScaledNumber definition.
// //
// TODO: Make this private to BlockFrequencyInfoImpl or delete. // TODO: Move to include/llvm/Support/ScaledNumber.h
// //
//===----------------------------------------------------------------------===// //===----------------------------------------------------------------------===//
namespace llvm { namespace llvm {
class UnsignedFloatBase { class ScaledNumberBase {
public: public:
static const int32_t MaxExponent = 16383; static const int32_t MaxExponent = 16383;
static const int32_t MinExponent = -16382; static const int32_t MinExponent = -16382;
@ -70,7 +70,7 @@ public:
/// \brief Simple representation of an unsigned floating point. /// \brief Simple representation of an unsigned floating point.
/// ///
/// UnsignedFloat is a unsigned floating point number. It uses simple /// ScaledNumber is a unsigned floating point number. It uses simple
/// saturation arithmetic, and every operation is well-defined for every value. /// saturation arithmetic, and every operation is well-defined for every value.
/// ///
/// The number is split into a signed exponent and unsigned digits. The number /// The number is split into a signed exponent and unsigned digits. The number
@ -79,23 +79,23 @@ public:
/// form, so the same number can be represented by many bit representations /// form, so the same number can be represented by many bit representations
/// (it's always in "denormal" mode). /// (it's always in "denormal" mode).
/// ///
/// UnsignedFloat is templated on the underlying integer type for digits, which /// ScaledNumber is templated on the underlying integer type for digits, which
/// is expected to be one of uint64_t, uint32_t, uint16_t or uint8_t. /// is expected to be one of uint64_t, uint32_t, uint16_t or uint8_t.
/// ///
/// Unlike builtin floating point types, UnsignedFloat is portable. /// Unlike builtin floating point types, ScaledNumber is portable.
/// ///
/// Unlike APFloat, UnsignedFloat does not model architecture floating point /// Unlike APFloat, ScaledNumber does not model architecture floating point
/// behaviour (this should make it a little faster), and implements most /// behaviour (this should make it a little faster), and implements most
/// operators (this makes it usable). /// operators (this makes it usable).
/// ///
/// UnsignedFloat is totally ordered. However, there is no canonical form, so /// ScaledNumber is totally ordered. However, there is no canonical form, so
/// there are multiple representations of most scalars. E.g.: /// there are multiple representations of most scalars. E.g.:
/// ///
/// UnsignedFloat(8u, 0) == UnsignedFloat(4u, 1) /// ScaledNumber(8u, 0) == ScaledNumber(4u, 1)
/// UnsignedFloat(4u, 1) == UnsignedFloat(2u, 2) /// ScaledNumber(4u, 1) == ScaledNumber(2u, 2)
/// UnsignedFloat(2u, 2) == UnsignedFloat(1u, 3) /// ScaledNumber(2u, 2) == ScaledNumber(1u, 3)
/// ///
/// UnsignedFloat implements most arithmetic operations. Precision is kept /// ScaledNumber implements most arithmetic operations. Precision is kept
/// where possible. Uses simple saturation arithmetic, so that operations /// where possible. Uses simple saturation arithmetic, so that operations
/// saturate to 0.0 or getLargest() rather than under or overflowing. It has /// saturate to 0.0 or getLargest() rather than under or overflowing. It has
/// some extra arithmetic for unit inversion. 0.0/0.0 is defined to be 0.0. /// some extra arithmetic for unit inversion. 0.0/0.0 is defined to be 0.0.
@ -111,7 +111,7 @@ public:
/// ///
/// The current plan is to gut this and make the necessary parts of it (even /// The current plan is to gut this and make the necessary parts of it (even
/// more) private to BlockFrequencyInfo. /// more) private to BlockFrequencyInfo.
template <class DigitsT> class UnsignedFloat : UnsignedFloatBase { template <class DigitsT> class ScaledNumber : ScaledNumberBase {
public: public:
static_assert(!std::numeric_limits<DigitsT>::is_signed, static_assert(!std::numeric_limits<DigitsT>::is_signed,
"only unsigned floats supported"); "only unsigned floats supported");
@ -129,26 +129,26 @@ private:
int16_t Exponent; int16_t Exponent;
public: public:
UnsignedFloat() : Digits(0), Exponent(0) {} ScaledNumber() : Digits(0), Exponent(0) {}
UnsignedFloat(DigitsType Digits, int16_t Exponent) ScaledNumber(DigitsType Digits, int16_t Exponent)
: Digits(Digits), Exponent(Exponent) {} : Digits(Digits), Exponent(Exponent) {}
private: private:
UnsignedFloat(const std::pair<uint64_t, int16_t> &X) ScaledNumber(const std::pair<uint64_t, int16_t> &X)
: Digits(X.first), Exponent(X.second) {} : Digits(X.first), Exponent(X.second) {}
public: public:
static UnsignedFloat getZero() { return UnsignedFloat(0, 0); } static ScaledNumber getZero() { return ScaledNumber(0, 0); }
static UnsignedFloat getOne() { return UnsignedFloat(1, 0); } static ScaledNumber getOne() { return ScaledNumber(1, 0); }
static UnsignedFloat getLargest() { static ScaledNumber getLargest() {
return UnsignedFloat(DigitsLimits::max(), MaxExponent); return ScaledNumber(DigitsLimits::max(), MaxExponent);
} }
static UnsignedFloat getFloat(uint64_t N) { return adjustToWidth(N, 0); } static ScaledNumber getFloat(uint64_t N) { return adjustToWidth(N, 0); }
static UnsignedFloat getInverseFloat(uint64_t N) { static ScaledNumber getInverseFloat(uint64_t N) {
return getFloat(N).invert(); return getFloat(N).invert();
} }
static UnsignedFloat getFraction(DigitsType N, DigitsType D) { static ScaledNumber getFraction(DigitsType N, DigitsType D) {
return getQuotient(N, D); return getQuotient(N, D);
} }
@ -188,12 +188,12 @@ public:
return ScaledNumbers::getLgCeiling(Digits, Exponent); return ScaledNumbers::getLgCeiling(Digits, Exponent);
} }
bool operator==(const UnsignedFloat &X) const { return compare(X) == 0; } bool operator==(const ScaledNumber &X) const { return compare(X) == 0; }
bool operator<(const UnsignedFloat &X) const { return compare(X) < 0; } bool operator<(const ScaledNumber &X) const { return compare(X) < 0; }
bool operator!=(const UnsignedFloat &X) const { return compare(X) != 0; } bool operator!=(const ScaledNumber &X) const { return compare(X) != 0; }
bool operator>(const UnsignedFloat &X) const { return compare(X) > 0; } bool operator>(const ScaledNumber &X) const { return compare(X) > 0; }
bool operator<=(const UnsignedFloat &X) const { return compare(X) <= 0; } bool operator<=(const ScaledNumber &X) const { return compare(X) <= 0; }
bool operator>=(const UnsignedFloat &X) const { return compare(X) >= 0; } bool operator>=(const ScaledNumber &X) const { return compare(X) >= 0; }
bool operator!() const { return isZero(); } bool operator!() const { return isZero(); }
@ -217,7 +217,7 @@ public:
/// 65432198.7654... => 65432198.77 /// 65432198.7654... => 65432198.77
/// 5432198.7654... => 5432198.765 /// 5432198.7654... => 5432198.765
std::string toString(unsigned Precision = DefaultPrecision) { std::string toString(unsigned Precision = DefaultPrecision) {
return UnsignedFloatBase::toString(Digits, Exponent, Width, Precision); return ScaledNumberBase::toString(Digits, Exponent, Width, Precision);
} }
/// \brief Print a decimal representation. /// \brief Print a decimal representation.
@ -225,11 +225,11 @@ public:
/// Print a string. See toString for documentation. /// Print a string. See toString for documentation.
raw_ostream &print(raw_ostream &OS, raw_ostream &print(raw_ostream &OS,
unsigned Precision = DefaultPrecision) const { unsigned Precision = DefaultPrecision) const {
return UnsignedFloatBase::print(OS, Digits, Exponent, Width, Precision); return ScaledNumberBase::print(OS, Digits, Exponent, Width, Precision);
} }
void dump() const { return UnsignedFloatBase::dump(Digits, Exponent, Width); } void dump() const { return ScaledNumberBase::dump(Digits, Exponent, Width); }
UnsignedFloat &operator+=(const UnsignedFloat &X) { ScaledNumber &operator+=(const ScaledNumber &X) {
std::tie(Digits, Exponent) = std::tie(Digits, Exponent) =
ScaledNumbers::getSum(Digits, Exponent, X.Digits, X.Exponent); ScaledNumbers::getSum(Digits, Exponent, X.Digits, X.Exponent);
// Check for exponent past MaxExponent. // Check for exponent past MaxExponent.
@ -237,15 +237,21 @@ public:
*this = getLargest(); *this = getLargest();
return *this; return *this;
} }
UnsignedFloat &operator-=(const UnsignedFloat &X) { ScaledNumber &operator-=(const ScaledNumber &X) {
std::tie(Digits, Exponent) = std::tie(Digits, Exponent) =
ScaledNumbers::getDifference(Digits, Exponent, X.Digits, X.Exponent); ScaledNumbers::getDifference(Digits, Exponent, X.Digits, X.Exponent);
return *this; return *this;
} }
UnsignedFloat &operator*=(const UnsignedFloat &X); ScaledNumber &operator*=(const ScaledNumber &X);
UnsignedFloat &operator/=(const UnsignedFloat &X); ScaledNumber &operator/=(const ScaledNumber &X);
UnsignedFloat &operator<<=(int16_t Shift) { shiftLeft(Shift); return *this; } ScaledNumber &operator<<=(int16_t Shift) {
UnsignedFloat &operator>>=(int16_t Shift) { shiftRight(Shift); return *this; } shiftLeft(Shift);
return *this;
}
ScaledNumber &operator>>=(int16_t Shift) {
shiftRight(Shift);
return *this;
}
private: private:
void shiftLeft(int32_t Shift); void shiftLeft(int32_t Shift);
@ -258,7 +264,7 @@ private:
/// ///
/// The value that compares smaller will lose precision, and possibly become /// The value that compares smaller will lose precision, and possibly become
/// \a isZero(). /// \a isZero().
UnsignedFloat matchExponents(UnsignedFloat X) { ScaledNumber matchExponents(ScaledNumber X) {
ScaledNumbers::matchScales(Digits, Exponent, X.Digits, X.Exponent); ScaledNumbers::matchScales(Digits, Exponent, X.Digits, X.Exponent);
return X; return X;
} }
@ -283,11 +289,11 @@ public:
return joinSigned(scaleByInverse(Unsigned.first), Unsigned.second); return joinSigned(scaleByInverse(Unsigned.first), Unsigned.second);
} }
int compare(const UnsignedFloat &X) const { int compare(const ScaledNumber &X) const {
return ScaledNumbers::compare(Digits, Exponent, X.Digits, X.Exponent); return ScaledNumbers::compare(Digits, Exponent, X.Digits, X.Exponent);
} }
int compareTo(uint64_t N) const { int compareTo(uint64_t N) const {
UnsignedFloat Float = getFloat(N); ScaledNumber Float = getFloat(N);
int Compare = compare(Float); int Compare = compare(Float);
if (Width == 64 || Compare != 0) if (Width == 64 || Compare != 0)
return Compare; return Compare;
@ -298,14 +304,14 @@ public:
} }
int compareTo(int64_t N) const { return N < 0 ? 1 : compareTo(uint64_t(N)); } int compareTo(int64_t N) const { return N < 0 ? 1 : compareTo(uint64_t(N)); }
UnsignedFloat &invert() { return *this = UnsignedFloat::getFloat(1) / *this; } ScaledNumber &invert() { return *this = ScaledNumber::getFloat(1) / *this; }
UnsignedFloat inverse() const { return UnsignedFloat(*this).invert(); } ScaledNumber inverse() const { return ScaledNumber(*this).invert(); }
private: private:
static UnsignedFloat getProduct(DigitsType LHS, DigitsType RHS) { static ScaledNumber getProduct(DigitsType LHS, DigitsType RHS) {
return ScaledNumbers::getProduct(LHS, RHS); return ScaledNumbers::getProduct(LHS, RHS);
} }
static UnsignedFloat getQuotient(DigitsType Dividend, DigitsType Divisor) { static ScaledNumber getQuotient(DigitsType Dividend, DigitsType Divisor) {
return ScaledNumbers::getQuotient(Dividend, Divisor); return ScaledNumbers::getQuotient(Dividend, Divisor);
} }
@ -322,14 +328,14 @@ private:
/// Should only be called for \c Shift close to zero. /// Should only be called for \c Shift close to zero.
/// ///
/// \pre Shift >= MinExponent && Shift + 64 <= MaxExponent. /// \pre Shift >= MinExponent && Shift + 64 <= MaxExponent.
static UnsignedFloat adjustToWidth(uint64_t N, int32_t Shift) { static ScaledNumber adjustToWidth(uint64_t N, int32_t Shift) {
assert(Shift >= MinExponent && "Shift should be close to 0"); assert(Shift >= MinExponent && "Shift should be close to 0");
assert(Shift <= MaxExponent - 64 && "Shift should be close to 0"); assert(Shift <= MaxExponent - 64 && "Shift should be close to 0");
auto Adjusted = ScaledNumbers::getAdjusted<DigitsT>(N, Shift); auto Adjusted = ScaledNumbers::getAdjusted<DigitsT>(N, Shift);
return Adjusted; return Adjusted;
} }
static UnsignedFloat getRounded(UnsignedFloat P, bool Round) { static ScaledNumber getRounded(ScaledNumber P, bool Round) {
// Saturate. // Saturate.
if (P.isLargest()) if (P.isLargest())
return P; return P;
@ -338,60 +344,60 @@ private:
} }
}; };
#define UNSIGNED_FLOAT_BOP(op, base) \ #define SCALED_NUMBER_BOP(op, base) \
template <class DigitsT> \ template <class DigitsT> \
UnsignedFloat<DigitsT> operator op(const UnsignedFloat<DigitsT> &L, \ ScaledNumber<DigitsT> operator op(const ScaledNumber<DigitsT> &L, \
const UnsignedFloat<DigitsT> &R) { \ const ScaledNumber<DigitsT> &R) { \
return UnsignedFloat<DigitsT>(L) base R; \ return ScaledNumber<DigitsT>(L) base R; \
} }
UNSIGNED_FLOAT_BOP(+, += ) SCALED_NUMBER_BOP(+, += )
UNSIGNED_FLOAT_BOP(-, -= ) SCALED_NUMBER_BOP(-, -= )
UNSIGNED_FLOAT_BOP(*, *= ) SCALED_NUMBER_BOP(*, *= )
UNSIGNED_FLOAT_BOP(/, /= ) SCALED_NUMBER_BOP(/, /= )
UNSIGNED_FLOAT_BOP(<<, <<= ) SCALED_NUMBER_BOP(<<, <<= )
UNSIGNED_FLOAT_BOP(>>, >>= ) SCALED_NUMBER_BOP(>>, >>= )
#undef UNSIGNED_FLOAT_BOP #undef SCALED_NUMBER_BOP
template <class DigitsT> template <class DigitsT>
raw_ostream &operator<<(raw_ostream &OS, const UnsignedFloat<DigitsT> &X) { raw_ostream &operator<<(raw_ostream &OS, const ScaledNumber<DigitsT> &X) {
return X.print(OS, 10); return X.print(OS, 10);
} }
#define UNSIGNED_FLOAT_COMPARE_TO_TYPE(op, T1, T2) \ #define SCALED_NUMBER_COMPARE_TO_TYPE(op, T1, T2) \
template <class DigitsT> \ template <class DigitsT> \
bool operator op(const UnsignedFloat<DigitsT> &L, T1 R) { \ bool operator op(const ScaledNumber<DigitsT> &L, T1 R) { \
return L.compareTo(T2(R)) op 0; \ return L.compareTo(T2(R)) op 0; \
} \ } \
template <class DigitsT> \ template <class DigitsT> \
bool operator op(T1 L, const UnsignedFloat<DigitsT> &R) { \ bool operator op(T1 L, const ScaledNumber<DigitsT> &R) { \
return 0 op R.compareTo(T2(L)); \ return 0 op R.compareTo(T2(L)); \
} }
#define UNSIGNED_FLOAT_COMPARE_TO(op) \ #define SCALED_NUMBER_COMPARE_TO(op) \
UNSIGNED_FLOAT_COMPARE_TO_TYPE(op, uint64_t, uint64_t) \ SCALED_NUMBER_COMPARE_TO_TYPE(op, uint64_t, uint64_t) \
UNSIGNED_FLOAT_COMPARE_TO_TYPE(op, uint32_t, uint64_t) \ SCALED_NUMBER_COMPARE_TO_TYPE(op, uint32_t, uint64_t) \
UNSIGNED_FLOAT_COMPARE_TO_TYPE(op, int64_t, int64_t) \ SCALED_NUMBER_COMPARE_TO_TYPE(op, int64_t, int64_t) \
UNSIGNED_FLOAT_COMPARE_TO_TYPE(op, int32_t, int64_t) SCALED_NUMBER_COMPARE_TO_TYPE(op, int32_t, int64_t)
UNSIGNED_FLOAT_COMPARE_TO(< ) SCALED_NUMBER_COMPARE_TO(< )
UNSIGNED_FLOAT_COMPARE_TO(> ) SCALED_NUMBER_COMPARE_TO(> )
UNSIGNED_FLOAT_COMPARE_TO(== ) SCALED_NUMBER_COMPARE_TO(== )
UNSIGNED_FLOAT_COMPARE_TO(!= ) SCALED_NUMBER_COMPARE_TO(!= )
UNSIGNED_FLOAT_COMPARE_TO(<= ) SCALED_NUMBER_COMPARE_TO(<= )
UNSIGNED_FLOAT_COMPARE_TO(>= ) SCALED_NUMBER_COMPARE_TO(>= )
#undef UNSIGNED_FLOAT_COMPARE_TO #undef SCALED_NUMBER_COMPARE_TO
#undef UNSIGNED_FLOAT_COMPARE_TO_TYPE #undef SCALED_NUMBER_COMPARE_TO_TYPE
template <class DigitsT> template <class DigitsT>
uint64_t UnsignedFloat<DigitsT>::scale(uint64_t N) const { uint64_t ScaledNumber<DigitsT>::scale(uint64_t N) const {
if (Width == 64 || N <= DigitsLimits::max()) if (Width == 64 || N <= DigitsLimits::max())
return (getFloat(N) * *this).template toInt<uint64_t>(); return (getFloat(N) * *this).template toInt<uint64_t>();
// Defer to the 64-bit version. // Defer to the 64-bit version.
return UnsignedFloat<uint64_t>(Digits, Exponent).scale(N); return ScaledNumber<uint64_t>(Digits, Exponent).scale(N);
} }
template <class DigitsT> template <class DigitsT>
template <class IntT> template <class IntT>
IntT UnsignedFloat<DigitsT>::toInt() const { IntT ScaledNumber<DigitsT>::toInt() const {
typedef std::numeric_limits<IntT> Limits; typedef std::numeric_limits<IntT> Limits;
if (*this < 1) if (*this < 1)
return 0; return 0;
@ -411,8 +417,8 @@ IntT UnsignedFloat<DigitsT>::toInt() const {
} }
template <class DigitsT> template <class DigitsT>
UnsignedFloat<DigitsT> &UnsignedFloat<DigitsT>:: ScaledNumber<DigitsT> &ScaledNumber<DigitsT>::
operator*=(const UnsignedFloat &X) { operator*=(const ScaledNumber &X) {
if (isZero()) if (isZero())
return *this; return *this;
if (X.isZero()) if (X.isZero())
@ -428,8 +434,8 @@ operator*=(const UnsignedFloat &X) {
return *this <<= Exponents; return *this <<= Exponents;
} }
template <class DigitsT> template <class DigitsT>
UnsignedFloat<DigitsT> &UnsignedFloat<DigitsT>:: ScaledNumber<DigitsT> &ScaledNumber<DigitsT>::
operator/=(const UnsignedFloat &X) { operator/=(const ScaledNumber &X) {
if (isZero()) if (isZero())
return *this; return *this;
if (X.isZero()) if (X.isZero())
@ -444,8 +450,7 @@ operator/=(const UnsignedFloat &X) {
// Combine with exponents. // Combine with exponents.
return *this <<= Exponents; return *this <<= Exponents;
} }
template <class DigitsT> template <class DigitsT> void ScaledNumber<DigitsT>::shiftLeft(int32_t Shift) {
void UnsignedFloat<DigitsT>::shiftLeft(int32_t Shift) {
if (!Shift || isZero()) if (!Shift || isZero())
return; return;
assert(Shift != INT32_MIN); assert(Shift != INT32_MIN);
@ -476,8 +481,7 @@ void UnsignedFloat<DigitsT>::shiftLeft(int32_t Shift) {
return; return;
} }
template <class DigitsT> template <class DigitsT> void ScaledNumber<DigitsT>::shiftRight(int32_t Shift) {
void UnsignedFloat<DigitsT>::shiftRight(int32_t Shift) {
if (!Shift || isZero()) if (!Shift || isZero())
return; return;
assert(Shift != INT32_MIN); assert(Shift != INT32_MIN);
@ -504,7 +508,7 @@ void UnsignedFloat<DigitsT>::shiftRight(int32_t Shift) {
return; return;
} }
template <class T> struct isPodLike<UnsignedFloat<T>> { template <class T> struct isPodLike<ScaledNumber<T>> {
static const bool value = true; static const bool value = true;
}; };
} }
@ -583,7 +587,7 @@ public:
/// ///
/// Convert to a float. \a isFull() gives 1.0, while \a isEmpty() gives /// Convert to a float. \a isFull() gives 1.0, while \a isEmpty() gives
/// slightly above 0.0. /// slightly above 0.0.
UnsignedFloat<uint64_t> toFloat() const; ScaledNumber<uint64_t> toFloat() const;
void dump() const; void dump() const;
raw_ostream &print(raw_ostream &OS) const; raw_ostream &print(raw_ostream &OS) const;
@ -646,7 +650,7 @@ template <class BT> struct BlockEdgesAdder;
/// BlockFrequencyInfoImpl. See there for details. /// BlockFrequencyInfoImpl. See there for details.
class BlockFrequencyInfoImplBase { class BlockFrequencyInfoImplBase {
public: public:
typedef UnsignedFloat<uint64_t> Float; typedef ScaledNumber<uint64_t> Float;
/// \brief Representative of a block. /// \brief Representative of a block.
/// ///
@ -1119,7 +1123,7 @@ void IrreducibleGraph::addEdges(const BlockNode &Node,
/// entries point to this block. Its successors are the headers, which split /// entries point to this block. Its successors are the headers, which split
/// the frequency evenly. /// the frequency evenly.
/// ///
/// This algorithm leverages BlockMass and UnsignedFloat to maintain precision, /// This algorithm leverages BlockMass and ScaledNumber to maintain precision,
/// separates mass distribution from loop scaling, and dithers to eliminate /// separates mass distribution from loop scaling, and dithers to eliminate
/// probability mass loss. /// probability mass loss.
/// ///

View File

@ -24,12 +24,12 @@ using namespace llvm::bfi_detail;
//===----------------------------------------------------------------------===// //===----------------------------------------------------------------------===//
// //
// UnsignedFloat implementation. // ScaledNumber implementation.
// //
//===----------------------------------------------------------------------===// //===----------------------------------------------------------------------===//
#ifndef _MSC_VER #ifndef _MSC_VER
const int32_t UnsignedFloatBase::MaxExponent; const int32_t ScaledNumberBase::MaxExponent;
const int32_t UnsignedFloatBase::MinExponent; const int32_t ScaledNumberBase::MinExponent;
#endif #endif
static void appendDigit(std::string &Str, unsigned D) { static void appendDigit(std::string &Str, unsigned D) {
@ -58,22 +58,22 @@ static bool doesRoundUp(char Digit) {
} }
static std::string toStringAPFloat(uint64_t D, int E, unsigned Precision) { static std::string toStringAPFloat(uint64_t D, int E, unsigned Precision) {
assert(E >= UnsignedFloatBase::MinExponent); assert(E >= ScaledNumberBase::MinExponent);
assert(E <= UnsignedFloatBase::MaxExponent); assert(E <= ScaledNumberBase::MaxExponent);
// Find a new E, but don't let it increase past MaxExponent. // Find a new E, but don't let it increase past MaxExponent.
int LeadingZeros = UnsignedFloatBase::countLeadingZeros64(D); int LeadingZeros = ScaledNumberBase::countLeadingZeros64(D);
int NewE = std::min(UnsignedFloatBase::MaxExponent, E + 63 - LeadingZeros); int NewE = std::min(ScaledNumberBase::MaxExponent, E + 63 - LeadingZeros);
int Shift = 63 - (NewE - E); int Shift = 63 - (NewE - E);
assert(Shift <= LeadingZeros); assert(Shift <= LeadingZeros);
assert(Shift == LeadingZeros || NewE == UnsignedFloatBase::MaxExponent); assert(Shift == LeadingZeros || NewE == ScaledNumberBase::MaxExponent);
D <<= Shift; D <<= Shift;
E = NewE; E = NewE;
// Check for a denormal. // Check for a denormal.
unsigned AdjustedE = E + 16383; unsigned AdjustedE = E + 16383;
if (!(D >> 63)) { if (!(D >> 63)) {
assert(E == UnsignedFloatBase::MaxExponent); assert(E == ScaledNumberBase::MaxExponent);
AdjustedE = 0; AdjustedE = 0;
} }
@ -95,8 +95,8 @@ static std::string stripTrailingZeros(const std::string &Float) {
return Float.substr(0, NonZero + 1); return Float.substr(0, NonZero + 1);
} }
std::string UnsignedFloatBase::toString(uint64_t D, int16_t E, int Width, std::string ScaledNumberBase::toString(uint64_t D, int16_t E, int Width,
unsigned Precision) { unsigned Precision) {
if (!D) if (!D)
return "0.0"; return "0.0";
@ -206,12 +206,12 @@ std::string UnsignedFloatBase::toString(uint64_t D, int16_t E, int Width,
return stripTrailingZeros(std::string(Carry, '1') + Str.substr(0, Truncate)); return stripTrailingZeros(std::string(Carry, '1') + Str.substr(0, Truncate));
} }
raw_ostream &UnsignedFloatBase::print(raw_ostream &OS, uint64_t D, int16_t E, raw_ostream &ScaledNumberBase::print(raw_ostream &OS, uint64_t D, int16_t E,
int Width, unsigned Precision) { int Width, unsigned Precision) {
return OS << toString(D, E, Width, Precision); return OS << toString(D, E, Width, Precision);
} }
void UnsignedFloatBase::dump(uint64_t D, int16_t E, int Width) { void ScaledNumberBase::dump(uint64_t D, int16_t E, int Width) {
print(dbgs(), D, E, Width, 0) << "[" << Width << ":" << D << "*2^" << E print(dbgs(), D, E, Width, 0) << "[" << Width << ":" << D << "*2^" << E
<< "]"; << "]";
} }
@ -221,10 +221,10 @@ void UnsignedFloatBase::dump(uint64_t D, int16_t E, int Width) {
// BlockMass implementation. // BlockMass implementation.
// //
//===----------------------------------------------------------------------===// //===----------------------------------------------------------------------===//
UnsignedFloat<uint64_t> BlockMass::toFloat() const { ScaledNumber<uint64_t> BlockMass::toFloat() const {
if (isFull()) if (isFull())
return UnsignedFloat<uint64_t>(1, 0); return ScaledNumber<uint64_t>(1, 0);
return UnsignedFloat<uint64_t>(getMass() + 1, -64); return ScaledNumber<uint64_t>(getMass() + 1, -64);
} }
void BlockMass::dump() const { print(dbgs()); } void BlockMass::dump() const { print(dbgs()); }