// Boost.Geometry Index // // R-tree nodes based on static conversion, storing dynamic-size containers // // Copyright (c) 2011-2023 Adam Wulkiewicz, Lodz, Poland. // // This file was modified by Oracle on 2021-2023. // Modifications copyright (c) 2021-2023 Oracle and/or its affiliates. // Contributed and/or modified by Vissarion Fysikopoulos, on behalf of Oracle // Contributed and/or modified by Adam Wulkiewicz, on behalf of Oracle // // Use, modification and distribution is subject to 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_GEOMETRY_INDEX_DETAIL_RTREE_NODE_WEAK_DYNAMIC_HPP #define BOOST_GEOMETRY_INDEX_DETAIL_RTREE_NODE_WEAK_DYNAMIC_HPP #include #include #include #include #include #include #include #include #include namespace boost { namespace geometry { namespace index { namespace detail { namespace rtree { // TODO: This should be defined in options.hpp // For now it's defined here to satisfy Boost header policy struct node_weak_dynamic_tag {}; struct node_weak_static_tag {}; template struct weak_internal_node : public weak_node { typedef rtree::ptr_pair element_type; typedef typename boost::container::allocator_traits < typename Allocators::internal_node_allocator_type >::template rebind_alloc allocator_type; typedef boost::container::vector elements_type; template inline weak_internal_node(Al const& al) : elements(allocator_type(al)) {} elements_type elements; }; template struct weak_leaf : public weak_node { typedef typename boost::container::allocator_traits < typename Allocators::leaf_allocator_type >::template rebind_alloc allocator_type; typedef boost::container::vector elements_type; template inline weak_leaf(Al const& al) : elements(allocator_type(al)) {} elements_type elements; }; // nodes traits template struct node { typedef weak_node type; }; template struct internal_node { typedef weak_internal_node type; }; template struct leaf { typedef weak_leaf type; }; // visitor traits template struct visitor { typedef weak_visitor type; }; // allocators template struct internal_node_alloc { typedef typename internal_node < Value, Parameters, Box, allocators, Tag >::type node_type; typedef typename boost::container::allocator_traits < Allocator >::template rebind_alloc type; }; template struct leaf_alloc { typedef typename leaf < Value, Parameters, Box, allocators, Tag >::type node_type; typedef typename ::boost::container::allocator_traits < Allocator >::template rebind_alloc type; }; template struct node_alloc { typedef typename weak_node < Value, Parameters, Box, allocators, Tag >::type node_type; typedef typename ::boost::container::allocator_traits < Allocator >::template rebind_alloc type; }; template class allocators : public internal_node_alloc::type , public leaf_alloc::type { typedef detail::rtree::internal_node_alloc < Allocator, Value, Parameters, Box, node_weak_dynamic_tag > internal_node_alloc; typedef detail::rtree::leaf_alloc < Allocator, Value, Parameters, Box, node_weak_dynamic_tag > leaf_alloc; typedef detail::rtree::node_alloc < Allocator, Value, Parameters, Box, node_weak_dynamic_tag > node_alloc; public: typedef typename internal_node_alloc::type internal_node_allocator_type; typedef typename leaf_alloc::type leaf_allocator_type; typedef typename node_alloc::traits::pointer node_pointer; private: typedef typename boost::container::allocator_traits < leaf_allocator_type // leaf_allocator_type for consistency with weak_leaf >::template rebind_alloc value_allocator_type; typedef boost::container::allocator_traits value_allocator_traits; public: typedef Allocator allocator_type; typedef Value value_type; typedef typename value_allocator_traits::reference reference; typedef typename value_allocator_traits::const_reference const_reference; typedef typename value_allocator_traits::size_type size_type; typedef typename value_allocator_traits::difference_type difference_type; typedef typename value_allocator_traits::pointer pointer; typedef typename value_allocator_traits::const_pointer const_pointer; inline allocators() : internal_node_allocator_type() , leaf_allocator_type() {} template inline explicit allocators(Alloc const& alloc) : internal_node_allocator_type(alloc) , leaf_allocator_type(alloc) {} inline allocators(allocators&& a) : internal_node_allocator_type(std::move(a.internal_node_allocator())) , leaf_allocator_type(std::move(a.leaf_allocator())) {} inline allocators & operator=(allocators&& a) { internal_node_allocator() = std::move(a.internal_node_allocator()); leaf_allocator() = std::move(a.leaf_allocator()); return *this; } inline allocators & operator=(allocators const& a) { internal_node_allocator() = a.internal_node_allocator(); leaf_allocator() = a.leaf_allocator(); return *this; } void swap(allocators & a) { boost::core::invoke_swap(internal_node_allocator(), a.internal_node_allocator()); boost::core::invoke_swap(leaf_allocator(), a.leaf_allocator()); } bool operator==(allocators const& a) const { return leaf_allocator() == a.leaf_allocator(); } template bool operator==(Alloc const& a) const { return leaf_allocator() == leaf_allocator_type(a); } Allocator allocator() const { return Allocator(leaf_allocator()); } internal_node_allocator_type & internal_node_allocator() { return *this; } internal_node_allocator_type const& internal_node_allocator() const { return *this; } leaf_allocator_type & leaf_allocator() { return *this; } leaf_allocator_type const& leaf_allocator() const { return *this; } }; // create_node_impl template struct create_weak_node { template static inline BaseNodePtr apply(AllocNode & alloc_node) { typedef boost::container::allocator_traits Al; typedef typename Al::pointer P; P p = Al::allocate(alloc_node, 1); if ( 0 == p ) throw_runtime_error("boost::geometry::index::rtree node creation failed"); scoped_deallocator deallocator(p, alloc_node); Al::construct(alloc_node, boost::to_address(p), alloc_node); deallocator.release(); return p; } }; // destroy_node_impl template struct destroy_weak_node { template static inline void apply(AllocNode & alloc_node, BaseNodePtr n) { typedef boost::container::allocator_traits Al; typedef typename Al::pointer P; P p(&static_cast(rtree::get(*n))); Al::destroy(alloc_node, boost::addressof(*p)); Al::deallocate(alloc_node, p, 1); } }; // create_node template struct create_node< Allocators, weak_internal_node > { static inline typename Allocators::node_pointer apply(Allocators & allocators) { return create_weak_node< typename Allocators::node_pointer, weak_internal_node >::apply(allocators.internal_node_allocator()); } }; template struct create_node< Allocators, weak_leaf > { static inline typename Allocators::node_pointer apply(Allocators & allocators) { return create_weak_node< typename Allocators::node_pointer, weak_leaf >::apply(allocators.leaf_allocator()); } }; // destroy_node template struct destroy_node< Allocators, weak_internal_node > { static inline void apply(Allocators & allocators, typename Allocators::node_pointer n) { destroy_weak_node< weak_internal_node >::apply(allocators.internal_node_allocator(), n); } }; template struct destroy_node< Allocators, weak_leaf > { static inline void apply(Allocators & allocators, typename Allocators::node_pointer n) { destroy_weak_node< weak_leaf >::apply(allocators.leaf_allocator(), n); } }; }} // namespace detail::rtree }}} // namespace boost::geometry::index #endif // BOOST_GEOMETRY_INDEX_DETAIL_RTREE_NODE_WEAK_DYNAMIC_HPP