/////////////////////////////////////////////////////////////// // 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_ #ifndef BOOST_MATH_RATIONAL_ADAPTER_HPP #define BOOST_MATH_RATIONAL_ADAPTER_HPP #include #include #include #include #include #ifdef BOOST_MSVC # pragma warning(push) # pragma warning(disable:4512 4127) #endif #include #ifdef BOOST_MSVC # pragma warning(pop) #endif namespace boost{ namespace multiprecision{ namespace backends{ template struct rational_adaptor { typedef number integer_type; typedef boost::rational rational_type; typedef typename IntBackend::signed_types signed_types; typedef typename IntBackend::unsigned_types unsigned_types; typedef typename IntBackend::float_types float_types; rational_adaptor(){} rational_adaptor(const rational_adaptor& o) { m_value = o.m_value; } rational_adaptor(const IntBackend& o) : m_value(o) {} template rational_adaptor(const U& u, typename enable_if_c::value>::type* = 0) : m_value(static_cast(u)){} template explicit rational_adaptor(const U& u, typename enable_if_c< boost::multiprecision::detail::is_explicitly_convertible::value && !is_convertible::value >::type* = 0) : m_value(IntBackend(u)){} template typename enable_if_c<(boost::multiprecision::detail::is_explicitly_convertible::value && !is_arithmetic::value), rational_adaptor&>::type operator = (const U& u) { m_value = IntBackend(u); } #ifndef BOOST_NO_CXX11_RVALUE_REFERENCES rational_adaptor(rational_adaptor&& o) : m_value(o.m_value) {} rational_adaptor(IntBackend&& o) : m_value(o) {} rational_adaptor& operator = (rational_adaptor&& o) { m_value = static_cast(o.m_value); return *this; } #endif rational_adaptor& operator = (const rational_adaptor& o) { m_value = o.m_value; return *this; } rational_adaptor& operator = (const IntBackend& o) { m_value = o; return *this; } template typename enable_if, rational_adaptor&>::type operator = (Int i) { m_value = i; return *this; } template typename enable_if, rational_adaptor&>::type operator = (Float i) { int e; Float f = std::frexp(i, &e); f = std::ldexp(f, std::numeric_limits::digits); e -= std::numeric_limits::digits; integer_type num(f); integer_type denom(1u); if(e > 0) { num <<= e; } else if(e < 0) { denom <<= -e; } m_value.assign(num, denom); return *this; } rational_adaptor& operator = (const char* s) { std::string s1; multiprecision::number v1, v2; char c; bool have_hex = false; const char* p = s; // saved for later while((0 != (c = *s)) && (c == 'x' || c == 'X' || c == '-' || c == '+' || (c >= '0' && c <= '9') || (have_hex && (c >= 'a' && c <= 'f')) || (have_hex && (c >= 'A' && c <= 'F')))) { if(c == 'x' || c == 'X') have_hex = true; s1.append(1, c); ++s; } v1.assign(s1); s1.erase(); if(c == '/') { ++s; while((0 != (c = *s)) && (c == 'x' || c == 'X' || c == '-' || c == '+' || (c >= '0' && c <= '9') || (have_hex && (c >= 'a' && c <= 'f')) || (have_hex && (c >= 'A' && c <= 'F')))) { if(c == 'x' || c == 'X') have_hex = true; s1.append(1, c); ++s; } v2.assign(s1); } else v2 = 1; if(*s) { BOOST_THROW_EXCEPTION(std::runtime_error(std::string("Could parse the string \"") + p + std::string("\" as a valid rational number."))); } data().assign(v1, v2); return *this; } void swap(rational_adaptor& o) { std::swap(m_value, o.m_value); } std::string str(std::streamsize digits, std::ios_base::fmtflags f)const { // // We format the string ourselves so we can match what GMP's mpq type does: // std::string result = data().numerator().str(digits, f); if(data().denominator() != 1) { result.append(1, '/'); result.append(data().denominator().str(digits, f)); } return result; } void negate() { m_value = -m_value; } int compare(const rational_adaptor& o)const { return m_value > o.m_value ? 1 : (m_value < o.m_value ? -1 : 0); } template typename enable_if_c::value && !is_floating_point::value, int>::type compare(Arithmatic i)const { return m_value > i ? 1 : (m_value < i ? -1 : 0); } template typename enable_if_c::value, int>::type compare(Arithmatic i)const { rational_adaptor r; r = i; return this->compare(r); } rational_type& data() { return m_value; } const rational_type& data()const { return m_value; } template void serialize(Archive& ar, const mpl::true_&) { // Saving integer_type n(m_value.numerator()), d(m_value.denominator()); ar & n; ar & d; } template void serialize(Archive& ar, const mpl::false_&) { // Loading integer_type n, d; ar & n; ar & d; m_value.assign(n, d); } template void serialize(Archive& ar, const unsigned int /*version*/) { typedef typename Archive::is_saving tag; serialize(ar, tag()); } private: rational_type m_value; }; template inline void eval_add(rational_adaptor& result, const rational_adaptor& o) { result.data() += o.data(); } template inline void eval_subtract(rational_adaptor& result, const rational_adaptor& o) { result.data() -= o.data(); } template inline void eval_multiply(rational_adaptor& result, const rational_adaptor& o) { result.data() *= o.data(); } template inline void eval_divide(rational_adaptor& result, const rational_adaptor& o) { using default_ops::eval_is_zero; if(eval_is_zero(o)) { BOOST_THROW_EXCEPTION(std::overflow_error("Divide by zero.")); } result.data() /= o.data(); } template inline typename enable_if_c::value == number_kind_floating_point>::type eval_convert_to(R* result, const rational_adaptor& backend) { // // The generic conversion is as good as anything we can write here: // ::boost::multiprecision::detail::generic_convert_rational_to_float(*result, backend); } template inline typename enable_if_c<(number_category::value != number_kind_integer) && (number_category::value != number_kind_floating_point)>::type eval_convert_to(R* result, const rational_adaptor& backend) { typedef typename component_type > >::type comp_t; comp_t num(backend.data().numerator()); comp_t denom(backend.data().denominator()); *result = num.template convert_to(); *result /= denom.template convert_to(); } template inline typename enable_if_c::value == number_kind_integer>::type eval_convert_to(R* result, const rational_adaptor& backend) { typedef typename component_type > >::type comp_t; comp_t t = backend.data().numerator(); t /= backend.data().denominator(); *result = t.template convert_to(); } template inline bool eval_is_zero(const rational_adaptor& val) { return eval_is_zero(val.data().numerator().backend()); } template inline int eval_get_sign(const rational_adaptor& val) { return eval_get_sign(val.data().numerator().backend()); } template inline void assign_components(rational_adaptor& result, const V& v1, const V& v2) { result.data().assign(v1, v2); } } // namespace backends template struct expression_template_default > : public expression_template_default {}; template struct number_category > : public mpl::int_{}; using boost::multiprecision::backends::rational_adaptor; template struct component_type > { typedef number type; }; template inline number numerator(const number, ET>& val) { return val.backend().data().numerator(); } template inline number denominator(const number, ET>& val) { return val.backend().data().denominator(); } #ifdef BOOST_NO_SFINAE_EXPR namespace detail{ template struct is_explicitly_convertible > : public is_explicitly_convertible {}; } #endif }} // namespaces namespace std{ template class numeric_limits, ExpressionTemplates> > : public std::numeric_limits > { typedef std::numeric_limits > base_type; typedef boost::multiprecision::number > number_type; public: BOOST_STATIC_CONSTEXPR bool is_integer = false; BOOST_STATIC_CONSTEXPR bool is_exact = true; BOOST_STATIC_CONSTEXPR number_type (min)() { return (base_type::min)(); } BOOST_STATIC_CONSTEXPR number_type (max)() { return (base_type::max)(); } BOOST_STATIC_CONSTEXPR number_type lowest() { return -(max)(); } BOOST_STATIC_CONSTEXPR number_type epsilon() { return base_type::epsilon(); } BOOST_STATIC_CONSTEXPR number_type round_error() { return epsilon() / 2; } BOOST_STATIC_CONSTEXPR number_type infinity() { return base_type::infinity(); } BOOST_STATIC_CONSTEXPR number_type quiet_NaN() { return base_type::quiet_NaN(); } BOOST_STATIC_CONSTEXPR number_type signaling_NaN() { return base_type::signaling_NaN(); } BOOST_STATIC_CONSTEXPR number_type denorm_min() { return base_type::denorm_min(); } }; #ifndef BOOST_NO_INCLASS_MEMBER_INITIALIZATION template BOOST_CONSTEXPR_OR_CONST bool numeric_limits, ExpressionTemplates> >::is_integer; template BOOST_CONSTEXPR_OR_CONST bool numeric_limits, ExpressionTemplates> >::is_exact; #endif } #endif