maths: split normalise functions
This commit is contained in:
parent
3991848726
commit
c2770a266b
131
maths.hpp
131
maths.hpp
@ -376,33 +376,120 @@ smoothstep [[gnu::pure]] (T a, T b, T x)
|
||||
|
||||
#include "types/string.hpp"
|
||||
|
||||
//-----------------------------------------------------------------------------
|
||||
|
||||
///////////////////////////////////////////////////////////////////////////////
|
||||
// renormalisation of unit floating point and/or normalised integers
|
||||
|
||||
// int -> float
|
||||
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)
|
||||
{
|
||||
static const T T_max = 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)
|
||||
return U (t / (sizeof (T) / sizeof (U)));
|
||||
else
|
||||
return U (t) * (sizeof (U) / sizeof (T));
|
||||
return t / static_cast<U> (std::numeric_limits<T>::max ());
|
||||
}
|
||||
|
||||
|
||||
// float -> int
|
||||
template <typename T, typename 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)
|
||||
{
|
||||
// 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"
|
||||
|
||||
#endif // __MATHS_HPP
|
||||
|
@ -77,24 +77,53 @@ test_normalisations (util::TAP::logger &tap)
|
||||
{
|
||||
bool success = true;
|
||||
|
||||
struct {
|
||||
static const struct {
|
||||
float a;
|
||||
uint8_t b;
|
||||
} TESTS[] = {
|
||||
{ 1.f, 255 },
|
||||
{ 0.f, 0 },
|
||||
{ 2.f, 255 },
|
||||
{ -1.f, 0 }
|
||||
{ 1.f, 255 },
|
||||
{ 0.5f, 127 },
|
||||
{ 2.f, 255 },
|
||||
{ -1.f, 0 }
|
||||
};
|
||||
|
||||
for (auto i: TESTS) {
|
||||
std::cerr << "x";
|
||||
auto v = renormalise<decltype(i.a),decltype(i.b)> (i.a);
|
||||
success = success && almost_equal (unsigned{v}, unsigned{i.b});
|
||||
}
|
||||
|
||||
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");
|
||||
}
|
||||
|
||||
|
||||
|
Loading…
Reference in New Issue
Block a user