serialize.hpp 5.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199
  1. ///////////////////////////////////////////////////////////////
  2. // Copyright 2013 John Maddock. Distributed under the Boost
  3. // Software License, Version 1.0. (See accompanying file
  4. // LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_
  5. #ifndef BOOST_MP_CPP_INT_SERIALIZE_HPP
  6. #define BOOST_MP_CPP_INT_SERIALIZE_HPP
  7. namespace boost {
  8. namespace archive{
  9. class binary_oarchive;
  10. class binary_iarchive;
  11. }
  12. namespace serialization {
  13. namespace mp = boost::multiprecision;
  14. namespace cpp_int_detail{
  15. using namespace boost::multiprecision;
  16. using namespace boost::multiprecision::backends;
  17. template <class T>
  18. struct is_binary_archive : public mpl::false_ {};
  19. template <>
  20. struct is_binary_archive<boost::archive::binary_oarchive> : public mpl::true_ {};
  21. template <>
  22. struct is_binary_archive<boost::archive::binary_iarchive> : public mpl::true_ {};
  23. //
  24. // We have 8 serialization methods to fill out (and test), they are all permutations of:
  25. // Load vs Store.
  26. // Trivial or non-trivial cpp_int type.
  27. // Binary or not archive.
  28. //
  29. template <class Archive, class Int>
  30. void do_serialize(Archive& ar, Int& val, mpl::false_ const&, mpl::false_ const&, mpl::false_ const&)
  31. {
  32. // Load.
  33. // Non-trivial.
  34. // Non binary.
  35. bool s;
  36. ar & s;
  37. std::size_t limb_count;
  38. std::size_t byte_count;
  39. ar & byte_count;
  40. limb_count = byte_count / sizeof(limb_type) + ((byte_count % sizeof(limb_type)) ? 1 : 0);
  41. val.resize(limb_count, limb_count);
  42. limb_type* pl = val.limbs();
  43. for(std::size_t i = 0; i < limb_count; ++i)
  44. {
  45. pl[i] = 0;
  46. for(std::size_t j = 0; (j < sizeof(limb_type)) && byte_count; ++j)
  47. {
  48. unsigned char byte;
  49. ar & byte;
  50. pl[i] |= static_cast<limb_type>(byte) << (j * CHAR_BIT);
  51. --byte_count;
  52. }
  53. }
  54. if(s != val.sign())
  55. val.negate();
  56. val.normalize();
  57. }
  58. template <class Archive, class Int>
  59. void do_serialize(Archive& ar, Int& val, mpl::true_ const&, mpl::false_ const&, mpl::false_ const&)
  60. {
  61. // Store.
  62. // Non-trivial.
  63. // Non binary.
  64. bool s = val.sign();
  65. ar & s;
  66. limb_type* pl = val.limbs();
  67. std::size_t limb_count = val.size();
  68. std::size_t byte_count = limb_count * sizeof(limb_type);
  69. ar & byte_count;
  70. for(std::size_t i = 0; i < limb_count; ++i)
  71. {
  72. limb_type l = pl[i];
  73. for(std::size_t j = 0; j < sizeof(limb_type); ++j)
  74. {
  75. unsigned char byte = static_cast<unsigned char>((l >> (j * CHAR_BIT)) & ((1u << CHAR_BIT) - 1));
  76. ar & byte;
  77. }
  78. }
  79. }
  80. template <class Archive, class Int>
  81. void do_serialize(Archive& ar, Int& val, mpl::false_ const&, mpl::true_ const&, mpl::false_ const&)
  82. {
  83. // Load.
  84. // Trivial.
  85. // Non binary.
  86. bool s;
  87. typename Int::local_limb_type l = 0;
  88. ar & s;
  89. std::size_t byte_count;
  90. ar & byte_count;
  91. for(std::size_t i = 0; i < byte_count; ++i)
  92. {
  93. unsigned char b;
  94. ar & b;
  95. l |= static_cast<typename Int::local_limb_type>(b) << (i * CHAR_BIT);
  96. }
  97. *val.limbs() = l;
  98. if(s != val.sign())
  99. val.negate();
  100. }
  101. template <class Archive, class Int>
  102. void do_serialize(Archive& ar, Int& val, mpl::true_ const&, mpl::true_ const&, mpl::false_ const&)
  103. {
  104. // Store.
  105. // Trivial.
  106. // Non binary.
  107. bool s = val.sign();
  108. typename Int::local_limb_type l = *val.limbs();
  109. ar & s;
  110. std::size_t limb_count = sizeof(l);
  111. ar & limb_count;
  112. for(std::size_t i = 0; i < limb_count; ++i)
  113. {
  114. unsigned char b = static_cast<unsigned char>(static_cast<typename Int::local_limb_type>(l >> (i * CHAR_BIT)) & static_cast<typename Int::local_limb_type>((1u << CHAR_BIT) - 1));
  115. ar & b;
  116. }
  117. }
  118. template <class Archive, class Int>
  119. void do_serialize(Archive& ar, Int& val, mpl::false_ const&, mpl::false_ const&, mpl::true_ const&)
  120. {
  121. // Load.
  122. // Non-trivial.
  123. // Binary.
  124. bool s;
  125. std::size_t c;
  126. ar & s;
  127. ar & c;
  128. val.resize(c, c);
  129. ar.load_binary(val.limbs(), c * sizeof(limb_type));
  130. if(s != val.sign())
  131. val.negate();
  132. val.normalize();
  133. }
  134. template <class Archive, class Int>
  135. void do_serialize(Archive& ar, Int& val, mpl::true_ const&, mpl::false_ const&, mpl::true_ const&)
  136. {
  137. // Store.
  138. // Non-trivial.
  139. // Binary.
  140. bool s = val.sign();
  141. std::size_t c = val.size();
  142. ar & s;
  143. ar & c;
  144. ar.save_binary(val.limbs(), c * sizeof(limb_type));
  145. }
  146. template <class Archive, class Int>
  147. void do_serialize(Archive& ar, Int& val, mpl::false_ const&, mpl::true_ const&, mpl::true_ const&)
  148. {
  149. // Load.
  150. // Trivial.
  151. // Binary.
  152. bool s;
  153. ar & s;
  154. ar.load_binary(val.limbs(), sizeof(*val.limbs()));
  155. if(s != val.sign())
  156. val.negate();
  157. }
  158. template <class Archive, class Int>
  159. void do_serialize(Archive& ar, Int& val, mpl::true_ const&, mpl::true_ const&, mpl::true_ const&)
  160. {
  161. // Store.
  162. // Trivial.
  163. // Binary.
  164. bool s = val.sign();
  165. ar & s;
  166. ar.save_binary(val.limbs(), sizeof(*val.limbs()));
  167. }
  168. }
  169. template<class Archive, unsigned MinBits, unsigned MaxBits, mp::cpp_integer_type SignType, mp::cpp_int_check_type Checked, class Allocator>
  170. void serialize(Archive & ar, mp::cpp_int_backend<MinBits, MaxBits, SignType, Checked, Allocator>& val, const unsigned int /*version*/)
  171. {
  172. typedef typename Archive::is_saving save_tag;
  173. typedef mpl::bool_<mp::backends::is_trivial_cpp_int<mp::cpp_int_backend<MinBits, MaxBits, SignType, Checked, Allocator> >::value> trivial_tag;
  174. typedef typename cpp_int_detail::is_binary_archive<Archive>::type binary_tag;
  175. // Just dispatch to the correct method:
  176. cpp_int_detail::do_serialize(ar, val, save_tag(), trivial_tag(), binary_tag());
  177. }
  178. }} // namespaces
  179. #endif // BOOST_MP_CPP_INT_SERIALIZE_HPP