/////////////////////////////////////////////////////////////// // Copyright 2013 John Maddock. Distributed under the Boost // Software License, Version 1.0. (See accompanying file // LICENSE_1_0.txt or copy at https://www.boost.org/LICENSE_1_0.txt #ifndef BOOST_MP_CPP_INT_LITERALS_HPP #define BOOST_MP_CPP_INT_LITERALS_HPP #include namespace boost { namespace multiprecision { namespace literals { namespace detail { template struct hex_value; template <> struct hex_value<'0'> { static constexpr limb_type value = 0; }; template <> struct hex_value<'1'> { static constexpr limb_type value = 1; }; template <> struct hex_value<'2'> { static constexpr limb_type value = 2; }; template <> struct hex_value<'3'> { static constexpr limb_type value = 3; }; template <> struct hex_value<'4'> { static constexpr limb_type value = 4; }; template <> struct hex_value<'5'> { static constexpr limb_type value = 5; }; template <> struct hex_value<'6'> { static constexpr limb_type value = 6; }; template <> struct hex_value<'7'> { static constexpr limb_type value = 7; }; template <> struct hex_value<'8'> { static constexpr limb_type value = 8; }; template <> struct hex_value<'9'> { static constexpr limb_type value = 9; }; template <> struct hex_value<'a'> { static constexpr limb_type value = 10; }; template <> struct hex_value<'b'> { static constexpr limb_type value = 11; }; template <> struct hex_value<'c'> { static constexpr limb_type value = 12; }; template <> struct hex_value<'d'> { static constexpr limb_type value = 13; }; template <> struct hex_value<'e'> { static constexpr limb_type value = 14; }; template <> struct hex_value<'f'> { static constexpr limb_type value = 15; }; template <> struct hex_value<'A'> { static constexpr limb_type value = 10; }; template <> struct hex_value<'B'> { static constexpr limb_type value = 11; }; template <> struct hex_value<'C'> { static constexpr limb_type value = 12; }; template <> struct hex_value<'D'> { static constexpr limb_type value = 13; }; template <> struct hex_value<'E'> { static constexpr limb_type value = 14; }; template <> struct hex_value<'F'> { static constexpr limb_type value = 15; }; template struct combine_value_to_pack; template struct combine_value_to_pack, value> { using type = value_pack; }; template struct pack_values { static constexpr std::size_t chars_per_limb = sizeof(limb_type) * CHAR_BIT / 4; static constexpr std::size_t shift = ((sizeof...(CHARS)) % chars_per_limb) * 4; static constexpr limb_type value_to_add = shift ? hex_value::value << shift : hex_value::value; using recursive_packed_type = typename pack_values::type ; using pack_type = typename std::conditional::type; using type = typename combine_value_to_pack::type; }; template struct pack_values { static constexpr limb_type value_to_add = hex_value::value; using type = value_pack; }; template struct strip_leading_zeros_from_pack; template struct strip_leading_zeros_from_pack > { using type = value_pack; }; template struct strip_leading_zeros_from_pack > { using type = typename strip_leading_zeros_from_pack >::type; }; template struct append_value_to_pack; template struct append_value_to_pack > { using type = value_pack; }; template struct reverse_value_pack; template struct reverse_value_pack > { using lead_values = typename reverse_value_pack >::type; using type = typename append_value_to_pack::type ; }; template struct reverse_value_pack > { using type = value_pack; }; template <> struct reverse_value_pack > { using type = value_pack<>; }; template struct make_packed_value_from_str { static_assert(l1 == '0', "Multi-precision integer literals must be in hexadecimal notation."); static_assert((l2 == 'X') || (l2 == 'x'), "Multi-precision integer literals must be in hexadecimal notation."); using packed_type = typename pack_values::type ; using stripped_type = typename strip_leading_zeros_from_pack::type; using type = typename reverse_value_pack::type ; }; template struct make_backend_from_pack { static constexpr Pack p = {}; static constexpr B value = p; }; template constexpr B make_backend_from_pack::value; template struct signed_cpp_int_literal_result_type { static constexpr unsigned bits = Digits * 4; using backend_type = boost::multiprecision::backends::cpp_int_backend; using number_type = number ; }; template struct unsigned_cpp_int_literal_result_type { static constexpr unsigned bits = Digits * 4; using backend_type = boost::multiprecision::backends::cpp_int_backend; using number_type = number ; }; } // namespace detail template constexpr typename boost::multiprecision::literals::detail::signed_cpp_int_literal_result_type((sizeof...(STR)) - 2u)>::number_type operator"" _cppi() { using pt = typename boost::multiprecision::literals::detail::make_packed_value_from_str::type; return boost::multiprecision::literals::detail::make_backend_from_pack((sizeof...(STR)) - 2u)>::backend_type>::value; } template constexpr typename boost::multiprecision::literals::detail::unsigned_cpp_int_literal_result_type((sizeof...(STR)) - 2u)>::number_type operator"" _cppui() { using pt = typename boost::multiprecision::literals::detail::make_packed_value_from_str::type; return boost::multiprecision::literals::detail::make_backend_from_pack((sizeof...(STR)) - 2u)>::backend_type>::value; } #define BOOST_MP_DEFINE_SIZED_CPP_INT_LITERAL(Bits) \ template \ constexpr boost::multiprecision::number > operator"" BOOST_JOIN(_cppi, Bits)() \ { \ using pt = typename boost::multiprecision::literals::detail::make_packed_value_from_str::type; \ return boost::multiprecision::literals::detail::make_backend_from_pack< \ pt, \ boost::multiprecision::backends::cpp_int_backend >::value; \ } \ template \ constexpr boost::multiprecision::number > operator"" BOOST_JOIN(_cppui, Bits)() \ { \ using pt = typename boost::multiprecision::literals::detail::make_packed_value_from_str::type; \ return boost::multiprecision::literals::detail::make_backend_from_pack< \ pt, \ boost::multiprecision::backends::cpp_int_backend >::value; \ } BOOST_MP_DEFINE_SIZED_CPP_INT_LITERAL(128) BOOST_MP_DEFINE_SIZED_CPP_INT_LITERAL(256) BOOST_MP_DEFINE_SIZED_CPP_INT_LITERAL(512) BOOST_MP_DEFINE_SIZED_CPP_INT_LITERAL(1024) } // namespace literals // // Overload unary minus operator for constexpr use: // template constexpr number, et_off> operator-(const number, et_off>& a) { return cpp_int_backend(a.backend(), boost::multiprecision::literals::detail::make_negate_tag()); } template constexpr number, et_off> operator-(number, et_off>&& a) { return cpp_int_backend(static_cast, et_off>&>(a).backend(), boost::multiprecision::literals::detail::make_negate_tag()); } }} // namespace boost::multiprecision #endif // BOOST_MP_CPP_INT_CORE_HPP