#ifndef SPROUT_MATH_REM_QUO_HPP #define SPROUT_MATH_REM_QUO_HPP #include #include #include #include #include #include #include #include #include #include #include #include #include namespace sprout { namespace math { namespace detail { template inline SPROUT_CONSTEXPR sprout::pair rem_quo_ret(Pair const& p) { typedef sprout::pair type; return type(static_cast(p.first), p.second); } template inline SPROUT_CONSTEXPR sprout::pair rem_quo_impl_7(T x, T y, int k, R iquo, T x1, T z) { typedef sprout::pair type; return (z > y) || ((z == y) && ((iquo & 1) != 0)) ? x < 0 ? type(-(x1 - y), ((iquo + 1) & 0x7) * k) : type(x1 - y, ((iquo + 1) & 0x7) * k) : x < 0 ? type(-x1, (iquo & 0x7) * k) : type(x1, (iquo & 0x7) * k) ; } template inline SPROUT_CONSTEXPR sprout::pair rem_quo_impl_6(T x, T y, int k, R iquo, T x1) { return x1 >= y ? sprout::math::detail::rem_quo_impl_7(x, y, k, iquo + 1, x1 - y, (x1 - y) + (x1 - y)) : sprout::math::detail::rem_quo_impl_7(x, y, k, iquo, x1, x1 + x1) ; } template inline SPROUT_CONSTEXPR sprout::pair rem_quo_impl_5(T x, T y, int k, R iquo, T x1, T y1, T z, int iscy, int idiff, int i) { return i != idiff ? z >= 0 ? sprout::math::detail::rem_quo_impl_5(x, y, k, iquo + iquo + 2, z + z, y1, z + z - y1, iscy, idiff, i + 1) : sprout::math::detail::rem_quo_impl_5(x, y, k, iquo + iquo, x1 + x1, y1, x1 + x1 - y1, iscy, idiff, i + 1) : sprout::math::detail::rem_quo_impl_6(x, y, k, iquo, sprout::math::scalbn(x1, iscy)) ; } template inline SPROUT_CONSTEXPR sprout::pair rem_quo_impl_4(T x, T y, int k, R iquo, T x1, T y1, int iscy, int idiff, int i) { return sprout::math::detail::rem_quo_impl_5(x, y, k, iquo, x1, y1, x1 - y1, iscy, idiff, i); } template inline SPROUT_CONSTEXPR sprout::pair rem_quo_impl_3(T x, T y, int k, R iquo, T x1, int iscx, int iscy, int idiff) { return idiff < 0 ? sprout::math::detail::rem_quo_impl_7(x, y, k, iquo, x1, x1 + x1) : idiff ? sprout::math::detail::rem_quo_impl_4(x, y, k, iquo, sprout::math::scalbn(x1, -iscx), sprout::math::scalbn(y, -iscy), iscy, idiff, 0) : sprout::math::detail::rem_quo_impl_6(x, y, k, iquo, x1) ; } template inline SPROUT_CONSTEXPR sprout::pair rem_quo_impl_2(T x, T y, int k, R iquo, T x1, int iscx, int iscy) { return sprout::math::detail::rem_quo_impl_3(x, y, k, iquo, x1, iscx, iscy, iscx - iscy); } template inline SPROUT_CONSTEXPR sprout::pair rem_quo_impl_1(T x, T y, int k, R iquo, T x1) { return sprout::math::detail::rem_quo_impl_2(x, y, k, iquo, x1, sprout::math::ilogb(x1), sprout::math::ilogb(y)); } template inline SPROUT_CONSTEXPR sprout::pair rem_quo_impl(T x, T y, R iquo) { return sprout::math::detail::rem_quo_impl_1(x, sprout::math::fabs(y), (sprout::math::signbit(x) ^ sprout::math::signbit(y) ? -1 : 1), iquo, sprout::math::fabs(x)); } template< typename R = int, typename FloatType, typename sprout::enabler_if::value && std::is_integral::value>::type = sprout::enabler > inline SPROUT_CONSTEXPR sprout::pair rem_quo(FloatType x, FloatType y) { typedef sprout::pair type; return y == 0 ? type(-std::numeric_limits::quiet_NaN(), R(0)) : sprout::math::isnan(y) ? sprout::math::isnan(x) ? type(sprout::math::signbit(y) && sprout::math::signbit(x) ? -std::numeric_limits::quiet_NaN() : std::numeric_limits::quiet_NaN(), R(0)) : type(y, R(0)) : sprout::math::isnan(x) ? type(x, R(0)) : x == std::numeric_limits::infinity() || x == -std::numeric_limits::infinity() ? type(-std::numeric_limits::quiet_NaN(), R(0)) : x == 0 ? type(x, R(0)) : y == std::numeric_limits::infinity() || y == -std::numeric_limits::infinity() ? type(x, R(0)) : sprout::math::detail::rem_quo_ret(sprout::math::detail::rem_quo_impl( static_cast::type>(x), static_cast::type>(y), R(0) )) ; } template< typename R = int, typename ArithmeticType1, typename ArithmeticType2, typename sprout::enabler_if< std::is_arithmetic::value && std::is_arithmetic::value && std::is_integral::value >::type = sprout::enabler > inline SPROUT_CONSTEXPR sprout::pair< typename sprout::float_promote::type, R > rem_quo(ArithmeticType1 x, ArithmeticType2 y) { typedef typename sprout::float_promote::type type; return sprout::math::detail::rem_quo(static_cast(x), static_cast(y)); } } // namespace detail // // issue: // rem_quo(-NaN, -NaN) returns {-NaN, _} . // # returns {+NaN, _} . ( same as rem_quo(+NaN, +NaN) ) // using sprout::math::detail::rem_quo; } // namespace math using sprout::math::rem_quo; } // namespace sprout #endif // #ifndef SPROUT_MATH_REM_QUO_HPP