maths: split normalise functions

This commit is contained in:
Danny Robson 2015-10-20 16:50:44 +11:00
parent 3991848726
commit c2770a266b
2 changed files with 144 additions and 28 deletions

125
maths.hpp
View File

@ -376,33 +376,120 @@ smoothstep [[gnu::pure]] (T a, T b, T x)
#include "types/string.hpp" #include "types/string.hpp"
//-----------------------------------------------------------------------------
///////////////////////////////////////////////////////////////////////////////
// renormalisation of unit floating point and/or normalised integers
// int -> float
template <typename T, typename U> template <typename T, typename U>
U constexpr
typename std::enable_if<
!std::is_floating_point<T>::value && std::is_floating_point<U>::value, U
>::type
renormalise [[gnu::pure]] (T t) renormalise [[gnu::pure]] (T t)
{ {
static const T T_max = std::numeric_limits<T>::max (); return t / static_cast<U> (std::numeric_limits<T>::max ());
static const U U_max = std::numeric_limits<U>::max ();
static const bool shrinking = sizeof (U) < sizeof (T);
static const bool T_float = std::is_floating_point<T>::value;
static const bool U_float = std::is_floating_point<U>::value;
if (T_float && U_float)
return U (t);
if (T_float) {
return U (limit (t, 0, 1) * U_max);
} }
if (U_float)
return U(U (t) / T_max);
if (shrinking) // float -> int
return U (t / (sizeof (T) / sizeof (U))); template <typename T, typename U>
else constexpr
return U (t) * (sizeof (U) / sizeof (T)); typename std::enable_if<
std::is_floating_point<T>::value && !std::is_floating_point<U>::value, U
>::type
renormalise [[gnu::pure]] (T t)
{
// Ideally std::ldexp would be involved but it complicates handing
// integers with greater precision than our floating point type. Also it
// would prohibit constexpr and involve errno.
size_t usable = std::numeric_limits<T>::digits;
size_t available = sizeof (U) * 8;
size_t shift = std::max (available, usable) - usable;
t = limit (t, 0, 1);
// construct an integer of the float's mantissa size, multiply it by our
// parameter, then shift it back into the full range of the integer type.
U in = std::numeric_limits<U>::max () >> shift;
U mid = static_cast<U> (t * in);
U out = mid << shift;
// use the top bits of the output to fill the bottom bits which through
// shifting would otherwise be zero. this gives us the full extent of the
// integer range, while varying predictably through the entire output
// space.
return out | out >> (available - shift);
} }
// float -> float, avoid identity conversion as we don't want to create
// ambiguous overloads
template <typename T, typename U>
constexpr
typename std::enable_if<
std::is_floating_point<T>::value &&
std::is_floating_point<U>::value &&
!std::is_same<T,U>::value, U
>::type
renormalise [[gnu::pure]] (T t)
{
return static_cast<U> (t);
}
// hi_int -> lo_int
template <typename T, typename U>
constexpr
typename std::enable_if<
std::is_integral<T>::value &&
std::is_integral<U>::value &&
(sizeof (T) > sizeof (U)), U
>::type
renormalise [[gnu::pure]] (T t)
{
// we have excess bits ,just shift and return
constexpr auto shift = 8 * (sizeof (T) - sizeof (U));
return t >> shift;
}
// lo_int -> hi_int
template <typename T, typename U>
constexpr
typename std::enable_if<
std::is_integral<T>::value &&
std::is_integral<U>::value &&
sizeof (T) < sizeof (U), U
>::type
renormalise [[gnu::pure]] (T t)
{
// we need to create bits. fill the output integer with copies of ourself.
// this is approximately correct in the general case (introducing a small
// linear positive bias), but allows us to fill the output space in the
// case of input maximum.
static_assert (sizeof (U) % sizeof (T) == 0,
"assumes integer multiple of sizes");
U out = 0;
for (size_t i = 0; i < sizeof (U) / sizeof (T); ++i)
out |= t << sizeof (U) * 8 * i;
return out;
}
template <typename T, typename U>
constexpr
typename std::enable_if<
std::is_same<T,U>::value, U
>::type
renormalise [[gnu::pure]] (T t)
{ return t; }
#include "maths.ipp" #include "maths.ipp"
#endif // __MATHS_HPP #endif // __MATHS_HPP

View File

@ -77,24 +77,53 @@ test_normalisations (util::TAP::logger &tap)
{ {
bool success = true; bool success = true;
struct { static const struct {
float a; float a;
uint8_t b; uint8_t b;
} TESTS[] = { } TESTS[] = {
{ 1.f, 255 }, { 1.f, 255 },
{ 0.f, 0 }, { 0.5f, 127 },
{ 2.f, 255 }, { 2.f, 255 },
{ -1.f, 0 } { -1.f, 0 }
}; };
for (auto i: TESTS) { for (auto i: TESTS) {
std::cerr << "x";
auto v = renormalise<decltype(i.a),decltype(i.b)> (i.a); auto v = renormalise<decltype(i.a),decltype(i.b)> (i.a);
success = success && almost_equal (unsigned{v}, unsigned{i.b}); success = success && almost_equal (unsigned{v}, unsigned{i.b});
} }
tap.expect (success, "float-u8 normalisation"); tap.expect (success, "float-u8 normalisation");
} }
// float to uint32
// exercises an integer type that has more precision than float
{
bool success = true;
static const struct {
float a;
uint32_t b;
} TESTS[] {
{ 0.f, 0x00000000 }, // lo range
{ 1.f, 0xffffffff }, // hi range
{ 0.5f, 0x7fffff7f }, // 31 bits
{ 0.001953125f, 0x007fff00 }, // 23 bits
};
for (auto t: TESTS) {
auto v = renormalise<float,uint32_t> (t.a);
success = success && almost_equal (t.b, v);
}
tap.expect (success, "float-u32 normalisation");
}
std::cerr << renormalise<uint8_t,uint32_t> (0xff) << '\n';
tap.expect_eq (renormalise<uint8_t,uint32_t> (0xff), 0xffffffff, "normalise hi u8-to-u32");
tap.expect_eq (renormalise<uint8_t,uint32_t> (0x00), 0x00000000, "normalise lo u8-to-u32");
tap.expect_eq (renormalise<uint32_t,uint8_t> (0xffffffff), 0xff, "normalise hi u32-to-u8");
} }