serialization.hpp 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581
  1. // Boost.Geometry Index
  2. //
  3. // Copyright (c) 2011-2013 Adam Wulkiewicz, Lodz, Poland.
  4. //
  5. // Use, modification and distribution is subject to the Boost Software License,
  6. // Version 1.0. (See accompanying file LICENSE_1_0.txt or copy at
  7. // http://www.boost.org/LICENSE_1_0.txt)
  8. #ifndef BOOST_GEOMETRY_INDEX_DETAIL_SERIALIZATION_HPP
  9. #define BOOST_GEOMETRY_INDEX_DETAIL_SERIALIZATION_HPP
  10. //#include <boost/serialization/serialization.hpp>
  11. #include <boost/serialization/split_member.hpp>
  12. #include <boost/serialization/version.hpp>
  13. //#include <boost/serialization/nvp.hpp>
  14. // TODO
  15. // how about using the unsigned type capable of storing Max in compile-time versions?
  16. // TODO
  17. // - add wrappers for Point and Box and implement serialize for those wrappers instead of
  18. // raw geometries
  19. // PROBLEM: after implementing this, how Values would be set?
  20. // - store the name of the parameters to know how to load and detect errors
  21. // - in the header, once store info about the Indexable and Bounds types (geometry type, point CS, Dim, etc.)
  22. // each geometry save without this info
  23. // TODO - move to index/detail/serialization.hpp
  24. namespace boost { namespace geometry { namespace index { namespace detail {
  25. // TODO - use boost::move?
  26. template<typename T>
  27. class serialization_storage
  28. {
  29. public:
  30. template <typename Archive>
  31. serialization_storage(Archive & ar, unsigned int version)
  32. {
  33. boost::serialization::load_construct_data_adl(ar, this->address(), version);
  34. }
  35. ~serialization_storage()
  36. {
  37. this->address()->~T();
  38. }
  39. T * address()
  40. {
  41. return static_cast<T*>(m_storage.address());
  42. }
  43. private:
  44. boost::aligned_storage<sizeof(T), boost::alignment_of<T>::value> m_storage;
  45. };
  46. // TODO - save and load item_version? see: collections_load_imp and collections_save_imp
  47. // this should be done once for the whole container
  48. // versions of all used types should be stored
  49. template <typename T, typename Archive> inline
  50. T serialization_load(const char * name, Archive & ar)
  51. {
  52. namespace bs = boost::serialization;
  53. serialization_storage<T> storage(ar, bs::version<T>::value); // load_construct_data
  54. ar >> boost::serialization::make_nvp(name, *storage.address()); // serialize
  55. //ar >> *storage.address(); // serialize
  56. return *storage.address();
  57. }
  58. template <typename T, typename Archive> inline
  59. void serialization_save(T const& t, const char * name, Archive & ar)
  60. {
  61. namespace bs = boost::serialization;
  62. bs::save_construct_data_adl(ar, boost::addressof(t), bs::version<T>::value); // save_construct_data
  63. ar << boost::serialization::make_nvp(name, t); // serialize
  64. //ar << t; // serialize
  65. }
  66. }}}}
  67. // TODO - move to index/serialization/rtree.hpp
  68. namespace boost { namespace serialization {
  69. // boost::geometry::index::linear
  70. template<class Archive, size_t Max, size_t Min>
  71. void save_construct_data(Archive & ar, const boost::geometry::index::linear<Max, Min> * params, unsigned int )
  72. {
  73. size_t max = params->get_max_elements(), min = params->get_min_elements();
  74. ar << boost::serialization::make_nvp("max", max);
  75. ar << boost::serialization::make_nvp("min", min);
  76. }
  77. template<class Archive, size_t Max, size_t Min>
  78. void load_construct_data(Archive & ar, boost::geometry::index::linear<Max, Min> * params, unsigned int )
  79. {
  80. size_t max, min;
  81. ar >> boost::serialization::make_nvp("max", max);
  82. ar >> boost::serialization::make_nvp("min", min);
  83. if ( max != params->get_max_elements() || min != params->get_min_elements() )
  84. // TODO change exception type
  85. BOOST_THROW_EXCEPTION(std::runtime_error("parameters not compatible"));
  86. // the constructor musn't be called for this type
  87. //::new(params)boost::geometry::index::linear<Max, Min>();
  88. }
  89. template<class Archive, size_t Max, size_t Min> void serialize(Archive &, boost::geometry::index::linear<Max, Min> &, unsigned int) {}
  90. // boost::geometry::index::quadratic
  91. template<class Archive, size_t Max, size_t Min>
  92. void save_construct_data(Archive & ar, const boost::geometry::index::quadratic<Max, Min> * params, unsigned int )
  93. {
  94. size_t max = params->get_max_elements(), min = params->get_min_elements();
  95. ar << boost::serialization::make_nvp("max", max);
  96. ar << boost::serialization::make_nvp("min", min);
  97. }
  98. template<class Archive, size_t Max, size_t Min>
  99. void load_construct_data(Archive & ar, boost::geometry::index::quadratic<Max, Min> * params, unsigned int )
  100. {
  101. size_t max, min;
  102. ar >> boost::serialization::make_nvp("max", max);
  103. ar >> boost::serialization::make_nvp("min", min);
  104. if ( max != params->get_max_elements() || min != params->get_min_elements() )
  105. // TODO change exception type
  106. BOOST_THROW_EXCEPTION(std::runtime_error("parameters not compatible"));
  107. // the constructor musn't be called for this type
  108. //::new(params)boost::geometry::index::quadratic<Max, Min>();
  109. }
  110. template<class Archive, size_t Max, size_t Min> void serialize(Archive &, boost::geometry::index::quadratic<Max, Min> &, unsigned int) {}
  111. // boost::geometry::index::rstar
  112. template<class Archive, size_t Max, size_t Min, size_t RE, size_t OCT>
  113. void save_construct_data(Archive & ar, const boost::geometry::index::rstar<Max, Min, RE, OCT> * params, unsigned int )
  114. {
  115. size_t max = params->get_max_elements()
  116. , min = params->get_min_elements()
  117. , re = params->get_reinserted_elements()
  118. , oct = params->get_overlap_cost_threshold();
  119. ar << boost::serialization::make_nvp("max", max);
  120. ar << boost::serialization::make_nvp("min", min);
  121. ar << boost::serialization::make_nvp("re", re);
  122. ar << boost::serialization::make_nvp("oct", oct);
  123. }
  124. template<class Archive, size_t Max, size_t Min, size_t RE, size_t OCT>
  125. void load_construct_data(Archive & ar, boost::geometry::index::rstar<Max, Min, RE, OCT> * params, unsigned int )
  126. {
  127. size_t max, min, re, oct;
  128. ar >> boost::serialization::make_nvp("max", max);
  129. ar >> boost::serialization::make_nvp("min", min);
  130. ar >> boost::serialization::make_nvp("re", re);
  131. ar >> boost::serialization::make_nvp("oct", oct);
  132. if ( max != params->get_max_elements() || min != params->get_min_elements() ||
  133. re != params->get_reinserted_elements() || oct != params->get_overlap_cost_threshold() )
  134. // TODO change exception type
  135. BOOST_THROW_EXCEPTION(std::runtime_error("parameters not compatible"));
  136. // the constructor musn't be called for this type
  137. //::new(params)boost::geometry::index::rstar<Max, Min, RE, OCT>();
  138. }
  139. template<class Archive, size_t Max, size_t Min, size_t RE, size_t OCT>
  140. void serialize(Archive &, boost::geometry::index::rstar<Max, Min, RE, OCT> &, unsigned int) {}
  141. // boost::geometry::index::dynamic_linear
  142. template<class Archive>
  143. inline void save_construct_data(Archive & ar, const boost::geometry::index::dynamic_linear * params, unsigned int )
  144. {
  145. size_t max = params->get_max_elements(), min = params->get_min_elements();
  146. ar << boost::serialization::make_nvp("max", max);
  147. ar << boost::serialization::make_nvp("min", min);
  148. }
  149. template<class Archive>
  150. inline void load_construct_data(Archive & ar, boost::geometry::index::dynamic_linear * params, unsigned int )
  151. {
  152. size_t max, min;
  153. ar >> boost::serialization::make_nvp("max", max);
  154. ar >> boost::serialization::make_nvp("min", min);
  155. ::new(params)boost::geometry::index::dynamic_linear(max, min);
  156. }
  157. template<class Archive> void serialize(Archive &, boost::geometry::index::dynamic_linear &, unsigned int) {}
  158. // boost::geometry::index::dynamic_quadratic
  159. template<class Archive>
  160. inline void save_construct_data(Archive & ar, const boost::geometry::index::dynamic_quadratic * params, unsigned int )
  161. {
  162. size_t max = params->get_max_elements(), min = params->get_min_elements();
  163. ar << boost::serialization::make_nvp("max", max);
  164. ar << boost::serialization::make_nvp("min", min);
  165. }
  166. template<class Archive>
  167. inline void load_construct_data(Archive & ar, boost::geometry::index::dynamic_quadratic * params, unsigned int )
  168. {
  169. size_t max, min;
  170. ar >> boost::serialization::make_nvp("max", max);
  171. ar >> boost::serialization::make_nvp("min", min);
  172. ::new(params)boost::geometry::index::dynamic_quadratic(max, min);
  173. }
  174. template<class Archive> void serialize(Archive &, boost::geometry::index::dynamic_quadratic &, unsigned int) {}
  175. // boost::geometry::index::dynamic_rstar
  176. template<class Archive>
  177. inline void save_construct_data(Archive & ar, const boost::geometry::index::dynamic_rstar * params, unsigned int )
  178. {
  179. size_t max = params->get_max_elements()
  180. , min = params->get_min_elements()
  181. , re = params->get_reinserted_elements()
  182. , oct = params->get_overlap_cost_threshold();
  183. ar << boost::serialization::make_nvp("max", max);
  184. ar << boost::serialization::make_nvp("min", min);
  185. ar << boost::serialization::make_nvp("re", re);
  186. ar << boost::serialization::make_nvp("oct", oct);
  187. }
  188. template<class Archive>
  189. inline void load_construct_data(Archive & ar, boost::geometry::index::dynamic_rstar * params, unsigned int )
  190. {
  191. size_t max, min, re, oct;
  192. ar >> boost::serialization::make_nvp("max", max);
  193. ar >> boost::serialization::make_nvp("min", min);
  194. ar >> boost::serialization::make_nvp("re", re);
  195. ar >> boost::serialization::make_nvp("oct", oct);
  196. ::new(params)boost::geometry::index::dynamic_rstar(max, min, re, oct);
  197. }
  198. template<class Archive> void serialize(Archive &, boost::geometry::index::dynamic_rstar &, unsigned int) {}
  199. }} // boost::serialization
  200. // TODO - move to index/detail/serialization.hpp or maybe geometry/serialization.hpp
  201. namespace boost { namespace geometry { namespace index { namespace detail {
  202. template <typename P, size_t I = 0, size_t D = traits::dimension<P>::value>
  203. struct serialize_point
  204. {
  205. template <typename Archive>
  206. static inline void save(Archive & ar, P const& p, unsigned int version)
  207. {
  208. typename coordinate_type<P>::type c = get<I>(p);
  209. ar << boost::serialization::make_nvp("c", c);
  210. serialize_point<P, I+1, D>::save(ar, p, version);
  211. }
  212. template <typename Archive>
  213. static inline void load(Archive & ar, P & p, unsigned int version)
  214. {
  215. typename traits::coordinate_type<P>::type c;
  216. ar >> boost::serialization::make_nvp("c", c);
  217. set<I>(p, c);
  218. serialize_point<P, I+1, D>::load(ar, p, version);
  219. }
  220. };
  221. template <typename P, size_t D>
  222. struct serialize_point<P, D, D>
  223. {
  224. template <typename Archive> static inline void save(Archive &, P const&, unsigned int) {}
  225. template <typename Archive> static inline void load(Archive &, P &, unsigned int) {}
  226. };
  227. }}}}
  228. // TODO - move to index/detail/serialization.hpp or maybe geometry/serialization.hpp
  229. namespace boost { namespace serialization {
  230. template<class Archive, typename T, size_t D, typename C>
  231. void save(Archive & ar, boost::geometry::model::point<T, D, C> const& p, unsigned int version)
  232. {
  233. boost::geometry::index::detail::serialize_point< boost::geometry::model::point<T, D, C> >::save(ar, p, version);
  234. }
  235. template<class Archive, typename T, size_t D, typename C>
  236. void load(Archive & ar, boost::geometry::model::point<T, D, C> & p, unsigned int version)
  237. {
  238. boost::geometry::index::detail::serialize_point< boost::geometry::model::point<T, D, C> >::load(ar, p, version);
  239. }
  240. template<class Archive, typename T, size_t D, typename C>
  241. inline void serialize(Archive & ar, boost::geometry::model::point<T, D, C> & o, const unsigned int version) { split_free(ar, o, version); }
  242. template<class Archive, typename P>
  243. inline void serialize(Archive & ar, boost::geometry::model::box<P> & b, const unsigned int)
  244. {
  245. ar & boost::serialization::make_nvp("min", b.min_corner());
  246. ar & boost::serialization::make_nvp("max", b.max_corner());
  247. }
  248. }} // boost::serialization
  249. // TODO - move to index/detail/rtree/visitors/save.hpp
  250. namespace boost { namespace geometry { namespace index { namespace detail { namespace rtree { namespace visitors {
  251. // TODO move saving and loading of the rtree outside the rtree, this will require adding some kind of members_view
  252. template <typename Archive, typename Value, typename Options, typename Translator, typename Box, typename Allocators>
  253. class save
  254. : public rtree::visitor<Value, typename Options::parameters_type, Box, Allocators, typename Options::node_tag, true>::type
  255. {
  256. public:
  257. typedef typename rtree::node<Value, typename Options::parameters_type, Box, Allocators, typename Options::node_tag>::type node;
  258. typedef typename rtree::internal_node<Value, typename Options::parameters_type, Box, Allocators, typename Options::node_tag>::type internal_node;
  259. typedef typename rtree::leaf<Value, typename Options::parameters_type, Box, Allocators, typename Options::node_tag>::type leaf;
  260. save(Archive & archive, unsigned int version)
  261. : m_archive(archive), m_version(version)
  262. {}
  263. inline void operator()(internal_node const& n)
  264. {
  265. typedef typename rtree::elements_type<internal_node>::type elements_type;
  266. elements_type const& elements = rtree::elements(n);
  267. // change to elements_type::size_type or size_type?
  268. size_t s = elements.size();
  269. m_archive << boost::serialization::make_nvp("s", s);
  270. for (typename elements_type::const_iterator it = elements.begin() ; it != elements.end() ; ++it)
  271. {
  272. serialization_save(it->first, "b", m_archive);
  273. rtree::apply_visitor(*this, *it->second);
  274. }
  275. }
  276. inline void operator()(leaf const& l)
  277. {
  278. typedef typename rtree::elements_type<leaf>::type elements_type;
  279. typedef typename elements_type::size_type elements_size;
  280. elements_type const& elements = rtree::elements(l);
  281. // change to elements_type::size_type or size_type?
  282. size_t s = elements.size();
  283. m_archive << boost::serialization::make_nvp("s", s);
  284. for (typename elements_type::const_iterator it = elements.begin() ; it != elements.end() ; ++it)
  285. {
  286. serialization_save(*it, "v", m_archive);
  287. }
  288. }
  289. private:
  290. Archive & m_archive;
  291. unsigned int m_version;
  292. };
  293. }}}}}} // boost::geometry::index::detail::rtree::visitors
  294. // TODO - move to index/detail/rtree/load.hpp
  295. namespace boost { namespace geometry { namespace index { namespace detail { namespace rtree {
  296. template <typename Value, typename Options, typename Translator, typename Box, typename Allocators>
  297. class load
  298. {
  299. typedef typename rtree::node<Value, typename Options::parameters_type, Box, Allocators, typename Options::node_tag>::type node;
  300. typedef typename rtree::internal_node<Value, typename Options::parameters_type, Box, Allocators, typename Options::node_tag>::type internal_node;
  301. typedef typename rtree::leaf<Value, typename Options::parameters_type, Box, Allocators, typename Options::node_tag>::type leaf;
  302. typedef typename Options::parameters_type parameters_type;
  303. typedef typename Allocators::node_pointer node_pointer;
  304. typedef rtree::node_auto_ptr<Value, Options, Translator, Box, Allocators> node_auto_ptr;
  305. typedef typename Allocators::size_type size_type;
  306. public:
  307. template <typename Archive> inline static
  308. node_pointer apply(Archive & ar, unsigned int version, size_type leafs_level, size_type & values_count, parameters_type const& parameters, Translator const& translator, Allocators & allocators)
  309. {
  310. values_count = 0;
  311. return raw_apply(ar, version, leafs_level, values_count, parameters, translator, allocators);
  312. }
  313. private:
  314. template <typename Archive> inline static
  315. node_pointer raw_apply(Archive & ar, unsigned int version, size_type leafs_level, size_type & values_count, parameters_type const& parameters, Translator const& translator, Allocators & allocators, size_type current_level = 0)
  316. {
  317. //BOOST_GEOMETRY_INDEX_ASSERT(current_level <= leafs_level, "invalid parameter");
  318. // change to elements_type::size_type or size_type?
  319. size_t elements_count;
  320. ar >> boost::serialization::make_nvp("s", elements_count);
  321. if ( elements_count < parameters.get_min_elements() || parameters.get_max_elements() < elements_count )
  322. BOOST_THROW_EXCEPTION(std::runtime_error("rtree loading error"));
  323. if ( current_level < leafs_level )
  324. {
  325. node_pointer n = rtree::create_node<Allocators, internal_node>::apply(allocators); // MAY THROW (A)
  326. node_auto_ptr auto_remover(n, allocators);
  327. internal_node & in = rtree::get<internal_node>(*n);
  328. typedef typename rtree::elements_type<internal_node>::type elements_type;
  329. typedef typename elements_type::value_type element_type;
  330. typedef typename elements_type::size_type elements_size;
  331. elements_type & elements = rtree::elements(in);
  332. elements.reserve(elements_count); // MAY THROW (A)
  333. for ( size_t i = 0 ; i < elements_count ; ++i )
  334. {
  335. typedef typename elements_type::value_type::first_type box_type;
  336. box_type b = serialization_load<box_type>("b", ar);
  337. node_pointer n = raw_apply(ar, version, leafs_level, values_count, parameters, translator, allocators, current_level+1); // recursive call
  338. elements.push_back(element_type(b, n));
  339. }
  340. auto_remover.release();
  341. return n;
  342. }
  343. else
  344. {
  345. BOOST_GEOMETRY_INDEX_ASSERT(current_level == leafs_level, "unexpected value");
  346. node_pointer n = rtree::create_node<Allocators, leaf>::apply(allocators); // MAY THROW (A)
  347. node_auto_ptr auto_remover(n, allocators);
  348. leaf & l = rtree::get<leaf>(*n);
  349. typedef typename rtree::elements_type<leaf>::type elements_type;
  350. typedef typename elements_type::value_type element_type;
  351. elements_type & elements = rtree::elements(l);
  352. values_count += elements_count;
  353. elements.reserve(elements_count); // MAY THROW (A)
  354. for ( size_t i = 0 ; i < elements_count ; ++i )
  355. {
  356. element_type el = serialization_load<element_type>("v", ar); // MAY THROW (C)
  357. elements.push_back(el); // MAY THROW (C)
  358. }
  359. auto_remover.release();
  360. return n;
  361. }
  362. }
  363. };
  364. }}}}} // boost::geometry::index::detail::rtree
  365. // TODO - move to index/detail/rtree/private_view.hpp
  366. namespace boost { namespace geometry { namespace index { namespace detail { namespace rtree {
  367. template <typename Rtree>
  368. class const_private_view
  369. {
  370. public:
  371. typedef typename Rtree::size_type size_type;
  372. typedef typename Rtree::translator_type translator_type;
  373. typedef typename Rtree::value_type value_type;
  374. typedef typename Rtree::options_type options_type;
  375. typedef typename Rtree::box_type box_type;
  376. typedef typename Rtree::allocators_type allocators_type;
  377. const_private_view(Rtree const& rt) : m_rtree(rt) {}
  378. typedef typename Rtree::members_holder members_holder;
  379. members_holder const& members() const { return m_rtree.m_members; }
  380. private:
  381. const_private_view(const_private_view const&);
  382. const_private_view & operator=(const_private_view const&);
  383. Rtree const& m_rtree;
  384. };
  385. template <typename Rtree>
  386. class private_view
  387. {
  388. public:
  389. typedef typename Rtree::size_type size_type;
  390. typedef typename Rtree::translator_type translator_type;
  391. typedef typename Rtree::value_type value_type;
  392. typedef typename Rtree::options_type options_type;
  393. typedef typename Rtree::box_type box_type;
  394. typedef typename Rtree::allocators_type allocators_type;
  395. private_view(Rtree & rt) : m_rtree(rt) {}
  396. typedef typename Rtree::members_holder members_holder;
  397. members_holder & members() { return m_rtree.m_members; }
  398. members_holder const& members() const { return m_rtree.m_members; }
  399. private:
  400. private_view(private_view const&);
  401. private_view & operator=(private_view const&);
  402. Rtree & m_rtree;
  403. };
  404. }}}}} // namespace boost::geometry::index::detail::rtree
  405. // TODO - move to index/serialization/rtree.hpp
  406. namespace boost { namespace serialization {
  407. template<class Archive, typename V, typename P, typename I, typename E, typename A>
  408. void save(Archive & ar, boost::geometry::index::rtree<V, P, I, E, A> const& rt, unsigned int version)
  409. {
  410. namespace detail = boost::geometry::index::detail;
  411. typedef boost::geometry::index::rtree<V, P, I, E, A> rtree;
  412. typedef detail::rtree::const_private_view<rtree> view;
  413. typedef typename view::translator_type translator_type;
  414. typedef typename view::value_type value_type;
  415. typedef typename view::options_type options_type;
  416. typedef typename view::box_type box_type;
  417. typedef typename view::allocators_type allocators_type;
  418. view tree(rt);
  419. detail::serialization_save(tree.members().parameters(), "parameters", ar);
  420. ar << boost::serialization::make_nvp("values_count", tree.members().values_count);
  421. ar << boost::serialization::make_nvp("leafs_level", tree.members().leafs_level);
  422. if ( tree.members().values_count )
  423. {
  424. BOOST_GEOMETRY_INDEX_ASSERT(tree.members().root, "root shouldn't be null_ptr");
  425. detail::rtree::visitors::save<Archive, value_type, options_type, translator_type, box_type, allocators_type> save_v(ar, version);
  426. detail::rtree::apply_visitor(save_v, *tree.members().root);
  427. }
  428. }
  429. template<class Archive, typename V, typename P, typename I, typename E, typename A>
  430. void load(Archive & ar, boost::geometry::index::rtree<V, P, I, E, A> & rt, unsigned int version)
  431. {
  432. namespace detail = boost::geometry::index::detail;
  433. typedef boost::geometry::index::rtree<V, P, I, E, A> rtree;
  434. typedef detail::rtree::private_view<rtree> view;
  435. typedef typename view::size_type size_type;
  436. typedef typename view::translator_type translator_type;
  437. typedef typename view::value_type value_type;
  438. typedef typename view::options_type options_type;
  439. typedef typename view::box_type box_type;
  440. typedef typename view::allocators_type allocators_type;
  441. typedef typename options_type::parameters_type parameters_type;
  442. typedef typename allocators_type::node_pointer node_pointer;
  443. typedef detail::rtree::node_auto_ptr<value_type, options_type, translator_type, box_type, allocators_type> node_auto_ptr;
  444. view tree(rt);
  445. parameters_type params = detail::serialization_load<parameters_type>("parameters", ar);
  446. size_type values_count, leafs_level;
  447. ar >> boost::serialization::make_nvp("values_count", values_count);
  448. ar >> boost::serialization::make_nvp("leafs_level", leafs_level);
  449. node_pointer n(0);
  450. if ( 0 < values_count )
  451. {
  452. size_type loaded_values_count = 0;
  453. n = detail::rtree::load<value_type, options_type, translator_type, box_type, allocators_type>
  454. ::apply(ar, version, leafs_level, loaded_values_count, params, tree.members().translator(), tree.members().allocators()); // MAY THROW
  455. node_auto_ptr remover(n, tree.members().allocators());
  456. if ( loaded_values_count != values_count )
  457. BOOST_THROW_EXCEPTION(std::runtime_error("unexpected number of values")); // TODO change exception type
  458. remover.release();
  459. }
  460. tree.members().parameters() = params;
  461. tree.members().values_count = values_count;
  462. tree.members().leafs_level = leafs_level;
  463. node_auto_ptr remover(tree.members().root, tree.members().allocators());
  464. tree.members().root = n;
  465. }
  466. template<class Archive, typename V, typename P, typename I, typename E, typename A> inline
  467. void serialize(Archive & ar, boost::geometry::index::rtree<V, P, I, E, A> & rt, unsigned int version)
  468. {
  469. split_free(ar, rt, version);
  470. }
  471. template<class Archive, typename V, typename P, typename I, typename E, typename A> inline
  472. void serialize(Archive & ar, boost::geometry::index::rtree<V, P, I, E, A> const& rt, unsigned int version)
  473. {
  474. split_free(ar, rt, version);
  475. }
  476. }} // boost::serialization
  477. #endif // BOOST_GEOMETRY_INDEX_DETAIL_SERIALIZATION_HPP