/////////////////////////////////////////////////////////////////////////////// // Copyright 2011 John Maddock. Distributed under the Boost // Software License, Version 1.0. (See accompanying file // LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt) #ifndef BOOST_MP_GENERIC_INTERCONVERT_HPP #define BOOST_MP_GENERIC_INTERCONVERT_HPP #include #ifdef BOOST_MSVC #pragma warning(push) #pragma warning(disable : 4127 6326) #endif namespace boost { namespace multiprecision { namespace detail { template inline To do_cast(const From& from) { return static_cast(from); } template inline To do_cast(const number& from) { return from.template convert_to(); } template void generic_interconvert(To& to, const From& from, const std::integral_constant& /*to_type*/, const std::integral_constant& /*from_type*/) { using default_ops::eval_add; using default_ops::eval_bitwise_and; using default_ops::eval_convert_to; using default_ops::eval_get_sign; using default_ops::eval_is_zero; using default_ops::eval_ldexp; using default_ops::eval_right_shift; // smallest unsigned type handled natively by "From" is likely to be it's limb_type: using l_limb_type = typename canonical::type; // get the corresponding type that we can assign to "To": using to_type = typename canonical::type; From t(from); bool is_neg = eval_get_sign(t) < 0; if (is_neg) t.negate(); // Pick off the first limb: l_limb_type limb; l_limb_type mask = static_cast(~static_cast(0)); From fl; eval_bitwise_and(fl, t, mask); eval_convert_to(&limb, fl); to = static_cast(limb); eval_right_shift(t, std::numeric_limits::digits); // // Then keep picking off more limbs until "t" is zero: // To l; unsigned shift = std::numeric_limits::digits; while (!eval_is_zero(t)) { eval_bitwise_and(fl, t, mask); eval_convert_to(&limb, fl); l = static_cast(limb); eval_right_shift(t, std::numeric_limits::digits); eval_ldexp(l, l, shift); eval_add(to, l); shift += std::numeric_limits::digits; } // // Finish off by setting the sign: // if (is_neg) to.negate(); } template void generic_interconvert(To& to, const From& from, const std::integral_constant& /*to_type*/, const std::integral_constant& /*from_type*/) { using default_ops::eval_bitwise_and; using default_ops::eval_bitwise_or; using default_ops::eval_convert_to; using default_ops::eval_get_sign; using default_ops::eval_is_zero; using default_ops::eval_left_shift; using default_ops::eval_right_shift; // smallest unsigned type handled natively by "From" is likely to be it's limb_type: using limb_type = typename canonical::type; // get the corresponding type that we can assign to "To": using to_type = typename canonical::type; From t(from); bool is_neg = eval_get_sign(t) < 0; if (is_neg) t.negate(); // Pick off the first limb: limb_type limb; limb_type mask = static_cast(~static_cast(0)); From fl; eval_bitwise_and(fl, t, mask); eval_convert_to(&limb, fl); to = static_cast(limb); eval_right_shift(t, std::numeric_limits::digits); // // Then keep picking off more limbs until "t" is zero: // To l; unsigned shift = std::numeric_limits::digits; while (!eval_is_zero(t)) { eval_bitwise_and(fl, t, mask); eval_convert_to(&limb, fl); l = static_cast(limb); eval_right_shift(t, std::numeric_limits::digits); eval_left_shift(l, shift); eval_bitwise_or(to, l); shift += std::numeric_limits::digits; } // // Finish off by setting the sign: // if (is_neg) to.negate(); } template void generic_interconvert(To& to, const From& from, const std::integral_constant& /*to_type*/, const std::integral_constant& /*from_type*/) { #ifdef BOOST_MSVC #pragma warning(push) //#pragma warning(disable : 4127) #endif // // The code here only works when the radix of "From" is 2, we could try shifting by other // radixes but it would complicate things.... use a string conversion when the radix is other // than 2: // BOOST_IF_CONSTEXPR(std::numeric_limits >::radix != 2) { to = from.str(0, std::ios_base::fmtflags()).c_str(); return; } else { using ui_type = typename canonical::type; using default_ops::eval_add; using default_ops::eval_convert_to; using default_ops::eval_fpclassify; using default_ops::eval_get_sign; using default_ops::eval_is_zero; using default_ops::eval_subtract; // // First classify the input, then handle the special cases: // int c = eval_fpclassify(from); if (c == (int)FP_ZERO) { to = ui_type(0); return; } else if (c == (int)FP_NAN) { to = static_cast("nan"); return; } else if (c == (int)FP_INFINITE) { to = static_cast("inf"); if (eval_get_sign(from) < 0) to.negate(); return; } typename From::exponent_type e; From f, term; to = ui_type(0); eval_frexp(f, from, &e); constexpr const int shift = std::numeric_limits::digits - 1; while (!eval_is_zero(f)) { // extract int sized bits from f: eval_ldexp(f, f, shift); eval_floor(term, f); e -= shift; eval_ldexp(to, to, shift); typename boost::multiprecision::detail::canonical::type ll; eval_convert_to(&ll, term); eval_add(to, ll); eval_subtract(f, term); } using to_exponent = typename To::exponent_type; if (e > (std::numeric_limits::max)()) { to = static_cast("inf"); if (eval_get_sign(from) < 0) to.negate(); return; } if (e < (std::numeric_limits::min)()) { to = ui_type(0); if (eval_get_sign(from) < 0) to.negate(); return; } eval_ldexp(to, to, static_cast(e)); } #ifdef BOOST_MSVC #pragma warning(pop) #endif } template void generic_interconvert(To& to, const From& from, const std::integral_constant& /*to_type*/, const std::integral_constant& /*from_type*/) { using to_component_type = typename component_type >::type; number t(from); to_component_type n(numerator(t)), d(denominator(t)); using default_ops::assign_components; assign_components(to, n.backend(), d.backend()); } template void generic_interconvert(To& to, const From& from, const std::integral_constant& /*to_type*/, const std::integral_constant& /*from_type*/) { using to_component_type = typename component_type >::type; number t(from); to_component_type n(t), d(1); using default_ops::assign_components; assign_components(to, n.backend(), d.backend()); } template R safe_convert_to_float(const LargeInteger& i) { using std::ldexp; if (!i) return R(0); BOOST_IF_CONSTEXPR(std::numeric_limits::is_specialized && std::numeric_limits::max_exponent) { LargeInteger val(i); if (val.sign() < 0) val = -val; unsigned mb = msb(val); if (mb >= std::numeric_limits::max_exponent) { int scale_factor = (int)mb + 1 - std::numeric_limits::max_exponent; BOOST_ASSERT(scale_factor >= 1); val >>= scale_factor; R result = val.template convert_to(); BOOST_IF_CONSTEXPR(std::numeric_limits::digits == 0 || std::numeric_limits::digits >= std::numeric_limits::max_exponent) { // // Calculate and add on the remainder, only if there are more // digits in the mantissa that the size of the exponent, in // other words if we are dropping digits in the conversion // otherwise: // LargeInteger remainder(i); remainder &= (LargeInteger(1) << scale_factor) - 1; result += ldexp(safe_convert_to_float(remainder), -scale_factor); } return i.sign() < 0 ? static_cast(-result) : result; } } return i.template convert_to(); } template inline typename std::enable_if::value || std::is_floating_point::value)>::type generic_convert_rational_to_float_imp(To& result, const Integer& n, const Integer& d, const std::integral_constant&) { // // If we get here, then there's something about one type or the other // that prevents an exactly rounded result from being calculated // (or at least it's not clear how to implement such a thing). // using default_ops::eval_divide; number fn(safe_convert_to_float >(n)), fd(safe_convert_to_float >(d)); eval_divide(result, fn.backend(), fd.backend()); } template inline typename std::enable_if::value || std::is_floating_point::value>::type generic_convert_rational_to_float_imp(To& result, const Integer& n, const Integer& d, const std::integral_constant&) { // // If we get here, then there's something about one type or the other // that prevents an exactly rounded result from being calculated // (or at least it's not clear how to implement such a thing). // To fd(safe_convert_to_float(d)); result = safe_convert_to_float(n); result /= fd; } template typename std::enable_if::value || std::is_floating_point::value>::type generic_convert_rational_to_float_imp(To& result, Integer& num, Integer& denom, const std::integral_constant&) { // // If we get here, then the precision of type To is known, and the integer type is unbounded // so we can use integer division plus manipulation of the remainder to get an exactly // rounded result. // if (num == 0) { result = 0; return; } bool s = false; if (num < 0) { s = true; num = -num; } int denom_bits = msb(denom); int shift = std::numeric_limits::digits + denom_bits - msb(num); if (shift > 0) num <<= shift; else if (shift < 0) denom <<= boost::multiprecision::detail::unsigned_abs(shift); Integer q, r; divide_qr(num, denom, q, r); int q_bits = msb(q); if (q_bits == std::numeric_limits::digits - 1) { // // Round up if 2 * r > denom: // r <<= 1; int c = r.compare(denom); if (c > 0) ++q; else if ((c == 0) && (q & 1u)) { ++q; } } else { BOOST_ASSERT(q_bits == std::numeric_limits::digits); // // We basically already have the rounding info: // if (q & 1u) { if (r || (q & 2u)) ++q; } } using std::ldexp; result = do_cast(q); result = ldexp(result, -shift); if (s) result = -result; } template inline typename std::enable_if::value || std::is_floating_point::value)>::type generic_convert_rational_to_float_imp(To& result, Integer& num, Integer& denom, const std::integral_constant& tag) { number t; generic_convert_rational_to_float_imp(t, num, denom, tag); result = t.backend(); } template inline void generic_convert_rational_to_float(To& result, const From& f) { // // Type From is always a Backend to number<>, or an // instance of number<>, but we allow // To to be either a Backend type, or a real number type, // that way we can call this from generic conversions, and // from specific conversions to built in types. // using actual_from_type = typename std::conditional::value, From, number >::type ; using actual_to_type = typename std::conditional::value || std::is_floating_point::value, To, number >::type ; using integer_type = typename component_type::type ; using dispatch_tag = std::integral_constant::is_specialized || std::numeric_limits::is_bounded || !std::numeric_limits::is_specialized || !std::numeric_limits::is_bounded || (std::numeric_limits::radix != 2)>; integer_type n(numerator(static_cast(f))), d(denominator(static_cast(f))); generic_convert_rational_to_float_imp(result, n, d, dispatch_tag()); } template inline void generic_interconvert(To& to, const From& from, const std::integral_constant& /*to_type*/, const std::integral_constant& /*from_type*/) { generic_convert_rational_to_float(to, from); } template void generic_interconvert_float2rational(To& to, const From& from, const std::integral_constant& /*radix*/) { using ui_type = typename std::tuple_element<0, typename To::unsigned_types>::type; constexpr const int shift = std::numeric_limits::digits; typename From::exponent_type e; typename component_type >::type num, denom; number val(from); val = frexp(val, &e); while (val) { val = ldexp(val, shift); e -= shift; boost::long_long_type ll = boost::math::lltrunc(val); val -= ll; num <<= shift; num += ll; } denom = ui_type(1u); if (e < 0) denom <<= -e; else if (e > 0) num <<= e; assign_components(to, num.backend(), denom.backend()); } template void generic_interconvert_float2rational(To& to, const From& from, const std::integral_constant& /*radix*/) { // // This is almost the same as the binary case above, but we have to use // scalbn and ilogb rather than ldexp and frexp, we also only extract // one Radix digit at a time which is terribly inefficient! // using ui_type = typename std::tuple_element<0, typename To::unsigned_types>::type; typename From::exponent_type e; typename component_type >::type num, denom; number val(from); if (!val) { to = ui_type(0u); return; } e = ilogb(val); val = scalbn(val, -e); while (val) { boost::long_long_type ll = boost::math::lltrunc(val); val -= ll; val = scalbn(val, 1); num *= Radix; num += ll; --e; } ++e; denom = ui_type(Radix); denom = pow(denom, abs(e)); if (e > 0) { num *= denom; denom = 1; } assign_components(to, num.backend(), denom.backend()); } template void generic_interconvert(To& to, const From& from, const std::integral_constant& /*to_type*/, const std::integral_constant& /*from_type*/) { generic_interconvert_float2rational(to, from, std::integral_constant >::radix>()); } template void generic_interconvert(To& to, const From& from, const std::integral_constant& /*to_type*/, const std::integral_constant& /*from_type*/) { number t(from); number result(numerator(t) / denominator(t)); to = result.backend(); } template void generic_interconvert_float2int(To& to, const From& from, const std::integral_constant& /*radix*/) { using exponent_type = typename From::exponent_type; constexpr const exponent_type shift = std::numeric_limits::digits; exponent_type e; number num(0u); number val(from); val = frexp(val, &e); bool neg = false; if (val.sign() < 0) { val.backend().negate(); neg = true; } while (e > 0) { exponent_type s = (std::min)(e, shift); val = ldexp(val, s); e -= s; boost::long_long_type ll = boost::math::lltrunc(val); val -= ll; num <<= s; num += ll; } to = num.backend(); if (neg) to.negate(); } template void generic_interconvert_float2int(To& to, const From& from, const std::integral_constant& /*radix*/) { // // This is almost the same as the binary case above, but we have to use // scalbn and ilogb rather than ldexp and frexp, we also only extract // one Radix digit at a time which is terribly inefficient! // typename From::exponent_type e; number num(0u); number val(from); e = ilogb(val); val = scalbn(val, -e); while (e >= 0) { boost::long_long_type ll = boost::math::lltrunc(val); val -= ll; val = scalbn(val, 1); num *= Radix; num += ll; --e; } to = num.backend(); } template void generic_interconvert(To& to, const From& from, const std::integral_constant& /*to_type*/, const std::integral_constant& /*from_type*/) { generic_interconvert_float2int(to, from, std::integral_constant >::radix>()); } template void generic_interconvert_complex_to_scalar(To& to, const From& from, const std::integral_constant&, const tag&) { // We just want the real part, and "to" is the correct type already: eval_real(to, from); To im; eval_imag(im, from); if (!eval_is_zero(im)) BOOST_THROW_EXCEPTION(std::runtime_error("Could not convert imaginary number to scalar.")); } template void generic_interconvert_complex_to_scalar(To& to, const From& from, const std::integral_constant&, const std::integral_constant&) { using component_number = typename component_type >::type; using component_backend = typename component_number::backend_type ; // // Get the real part and copy-construct the result from it: // component_backend r; generic_interconvert_complex_to_scalar(r, from, std::integral_constant(), std::integral_constant()); to = r; } template void generic_interconvert_complex_to_scalar(To& to, const From& from, const std::integral_constant&, const std::integral_constant&) { using component_number = typename component_type >::type; using component_backend = typename component_number::backend_type ; // // Get the real part and use a generic_interconvert to type To: // component_backend r; generic_interconvert_complex_to_scalar(r, from, std::integral_constant(), std::integral_constant()); generic_interconvert(to, r, std::integral_constant::value>(), std::integral_constant::value>()); } template void generic_interconvert(To& to, const From& from, const std::integral_constant& /*to_type*/, const std::integral_constant& /*from_type*/) { using component_number = typename component_type >::type; using component_backend = typename component_number::backend_type ; generic_interconvert_complex_to_scalar(to, from, std::integral_constant::value>(), std::integral_constant::value>()); } template void generic_interconvert(To& to, const From& from, const std::integral_constant& /*to_type*/, const std::integral_constant& /*from_type*/) { using component_number = typename component_type >::type; using component_backend = typename component_number::backend_type ; generic_interconvert_complex_to_scalar(to, from, std::integral_constant::value>(), std::integral_constant::value>()); } } } } // namespace boost::multiprecision::detail #ifdef BOOST_MSVC #pragma warning(pop) #endif #endif // BOOST_MP_GENERIC_INTERCONVERT_HPP