2014-06-20 01:30:43 +00:00
|
|
|
//===- llvm/Support/ScaledNumber.h - Support for scaled numbers -*- C++ -*-===//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// This file contains functions (and a class) useful for working with scaled
|
|
|
|
// numbers -- in particular, pairs of integers where one represents digits and
|
|
|
|
// another represents a scale. The functions are helpers and live in the
|
|
|
|
// namespace ScaledNumbers. The class ScaledNumber is useful for modelling
|
|
|
|
// certain cost metrics that need simple, integer-like semantics that are easy
|
|
|
|
// to reason about.
|
|
|
|
//
|
|
|
|
// These might remind you of soft-floats. If you want one of those, you're in
|
|
|
|
// the wrong place. Look at include/llvm/ADT/APFloat.h instead.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#ifndef LLVM_SUPPORT_SCALEDNUMBER_H
|
|
|
|
#define LLVM_SUPPORT_SCALEDNUMBER_H
|
|
|
|
|
2014-06-20 02:31:03 +00:00
|
|
|
#include "llvm/Support/MathExtras.h"
|
|
|
|
|
2014-06-20 01:30:43 +00:00
|
|
|
#include <cstdint>
|
|
|
|
#include <limits>
|
|
|
|
#include <utility>
|
|
|
|
|
|
|
|
namespace llvm {
|
|
|
|
namespace ScaledNumbers {
|
|
|
|
|
|
|
|
/// \brief Get the width of a number.
|
|
|
|
template <class DigitsT> inline int getWidth() { return sizeof(DigitsT) * 8; }
|
|
|
|
|
|
|
|
/// \brief Conditionally round up a scaled number.
|
|
|
|
///
|
|
|
|
/// Given \c Digits and \c Scale, round up iff \c ShouldRound is \c true.
|
|
|
|
/// Always returns \c Scale unless there's an overflow, in which case it
|
|
|
|
/// returns \c 1+Scale.
|
|
|
|
///
|
|
|
|
/// \pre adding 1 to \c Scale will not overflow INT16_MAX.
|
|
|
|
template <class DigitsT>
|
|
|
|
inline std::pair<DigitsT, int16_t> getRounded(DigitsT Digits, int16_t Scale,
|
|
|
|
bool ShouldRound) {
|
|
|
|
static_assert(!std::numeric_limits<DigitsT>::is_signed, "expected unsigned");
|
|
|
|
|
|
|
|
if (ShouldRound)
|
|
|
|
if (!++Digits)
|
|
|
|
// Overflow.
|
|
|
|
return std::make_pair(DigitsT(1) << (getWidth<DigitsT>() - 1), Scale + 1);
|
|
|
|
return std::make_pair(Digits, Scale);
|
|
|
|
}
|
2014-06-20 02:31:03 +00:00
|
|
|
|
2014-06-20 02:31:07 +00:00
|
|
|
/// \brief Convenience helper for 32-bit rounding.
|
|
|
|
inline std::pair<uint32_t, int16_t> getRounded32(uint32_t Digits, int16_t Scale,
|
|
|
|
bool ShouldRound) {
|
|
|
|
return getRounded(Digits, Scale, ShouldRound);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// \brief Convenience helper for 64-bit rounding.
|
|
|
|
inline std::pair<uint64_t, int16_t> getRounded64(uint64_t Digits, int16_t Scale,
|
|
|
|
bool ShouldRound) {
|
|
|
|
return getRounded(Digits, Scale, ShouldRound);
|
|
|
|
}
|
|
|
|
|
2014-06-20 02:31:03 +00:00
|
|
|
/// \brief Adjust a 64-bit scaled number down to the appropriate width.
|
|
|
|
///
|
2014-06-20 21:44:36 +00:00
|
|
|
/// \pre Adding 64 to \c Scale will not overflow INT16_MAX.
|
2014-06-20 02:31:03 +00:00
|
|
|
template <class DigitsT>
|
|
|
|
inline std::pair<DigitsT, int16_t> getAdjusted(uint64_t Digits,
|
|
|
|
int16_t Scale = 0) {
|
|
|
|
static_assert(!std::numeric_limits<DigitsT>::is_signed, "expected unsigned");
|
|
|
|
|
|
|
|
const int Width = getWidth<DigitsT>();
|
|
|
|
if (Width == 64 || Digits <= std::numeric_limits<DigitsT>::max())
|
|
|
|
return std::make_pair(Digits, Scale);
|
|
|
|
|
|
|
|
// Shift right and round.
|
|
|
|
int Shift = 64 - Width - countLeadingZeros(Digits);
|
|
|
|
return getRounded<DigitsT>(Digits >> Shift, Scale + Shift,
|
|
|
|
Digits & (UINT64_C(1) << (Shift - 1)));
|
|
|
|
}
|
|
|
|
|
|
|
|
/// \brief Convenience helper for adjusting to 32 bits.
|
|
|
|
inline std::pair<uint32_t, int16_t> getAdjusted32(uint64_t Digits,
|
|
|
|
int16_t Scale = 0) {
|
|
|
|
return getAdjusted<uint32_t>(Digits, Scale);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// \brief Convenience helper for adjusting to 64 bits.
|
|
|
|
inline std::pair<uint64_t, int16_t> getAdjusted64(uint64_t Digits,
|
|
|
|
int16_t Scale = 0) {
|
|
|
|
return getAdjusted<uint64_t>(Digits, Scale);
|
|
|
|
}
|
2014-06-20 21:43:20 +00:00
|
|
|
|
2014-06-20 21:47:47 +00:00
|
|
|
/// \brief Multiply two 64-bit integers to create a 64-bit scaled number.
|
|
|
|
///
|
|
|
|
/// Implemented with four 64-bit integer multiplies.
|
|
|
|
std::pair<uint64_t, int16_t> multiply64(uint64_t LHS, uint64_t RHS);
|
|
|
|
|
|
|
|
/// \brief Multiply two 32-bit integers to create a 32-bit scaled number.
|
|
|
|
///
|
|
|
|
/// Implemented with one 64-bit integer multiply.
|
|
|
|
template <class DigitsT>
|
|
|
|
inline std::pair<DigitsT, int16_t> getProduct(DigitsT LHS, DigitsT RHS) {
|
|
|
|
static_assert(!std::numeric_limits<DigitsT>::is_signed, "expected unsigned");
|
|
|
|
|
|
|
|
if (getWidth<DigitsT>() <= 32 || (LHS <= UINT32_MAX && RHS <= UINT32_MAX))
|
|
|
|
return getAdjusted<DigitsT>(uint64_t(LHS) * RHS);
|
|
|
|
|
|
|
|
return multiply64(LHS, RHS);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// \brief Convenience helper for 32-bit product.
|
|
|
|
inline std::pair<uint32_t, int16_t> getProduct32(uint32_t LHS, uint32_t RHS) {
|
|
|
|
return getProduct(LHS, RHS);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// \brief Convenience helper for 64-bit product.
|
|
|
|
inline std::pair<uint64_t, int16_t> getProduct64(uint64_t LHS, uint64_t RHS) {
|
|
|
|
return getProduct(LHS, RHS);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// \brief Divide two 64-bit integers to create a 64-bit scaled number.
|
|
|
|
///
|
|
|
|
/// Implemented with long division.
|
|
|
|
///
|
|
|
|
/// \pre \c Dividend and \c Divisor are non-zero.
|
|
|
|
std::pair<uint64_t, int16_t> divide64(uint64_t Dividend, uint64_t Divisor);
|
|
|
|
|
|
|
|
/// \brief Divide two 32-bit integers to create a 32-bit scaled number.
|
|
|
|
///
|
|
|
|
/// Implemented with one 64-bit integer divide/remainder pair.
|
|
|
|
///
|
|
|
|
/// \pre \c Dividend and \c Divisor are non-zero.
|
|
|
|
std::pair<uint32_t, int16_t> divide32(uint32_t Dividend, uint32_t Divisor);
|
|
|
|
|
|
|
|
/// \brief Divide two 32-bit numbers to create a 32-bit scaled number.
|
|
|
|
///
|
|
|
|
/// Implemented with one 64-bit integer divide/remainder pair.
|
|
|
|
///
|
|
|
|
/// Returns \c (DigitsT_MAX, INT16_MAX) for divide-by-zero (0 for 0/0).
|
|
|
|
template <class DigitsT>
|
|
|
|
std::pair<DigitsT, int16_t> getQuotient(DigitsT Dividend, DigitsT Divisor) {
|
|
|
|
static_assert(!std::numeric_limits<DigitsT>::is_signed, "expected unsigned");
|
|
|
|
static_assert(sizeof(DigitsT) == 4 || sizeof(DigitsT) == 8,
|
|
|
|
"expected 32-bit or 64-bit digits");
|
|
|
|
|
|
|
|
// Check for zero.
|
|
|
|
if (!Dividend)
|
|
|
|
return std::make_pair(0, 0);
|
|
|
|
if (!Divisor)
|
|
|
|
return std::make_pair(std::numeric_limits<DigitsT>::max(), INT16_MAX);
|
|
|
|
|
|
|
|
if (getWidth<DigitsT>() == 64)
|
|
|
|
return divide64(Dividend, Divisor);
|
|
|
|
return divide32(Dividend, Divisor);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// \brief Convenience helper for 32-bit quotient.
|
|
|
|
inline std::pair<uint32_t, int16_t> getQuotient32(uint32_t Dividend,
|
|
|
|
uint32_t Divisor) {
|
|
|
|
return getQuotient(Dividend, Divisor);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// \brief Convenience helper for 64-bit quotient.
|
|
|
|
inline std::pair<uint64_t, int16_t> getQuotient64(uint64_t Dividend,
|
|
|
|
uint64_t Divisor) {
|
|
|
|
return getQuotient(Dividend, Divisor);
|
|
|
|
}
|
|
|
|
|
2014-06-20 21:43:20 +00:00
|
|
|
} // end namespace ScaledNumbers
|
|
|
|
} // end namespace llvm
|
2014-06-20 01:30:43 +00:00
|
|
|
|
|
|
|
#endif
|
|
|
|
|