123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357 |
- // 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 <boost/container/allocator_traits.hpp>
- #include <boost/container/vector.hpp>
- #include <boost/core/pointer_traits.hpp>
- #include <boost/core/invoke_swap.hpp>
- #include <boost/geometry/index/detail/rtree/options.hpp>
- #include <boost/geometry/index/detail/rtree/node/concept.hpp>
- #include <boost/geometry/index/detail/rtree/node/pairs.hpp>
- #include <boost/geometry/index/detail/rtree/node/scoped_deallocator.hpp>
- #include <boost/geometry/index/detail/rtree/node/weak_visitor.hpp>
- 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 <typename Value, typename Parameters, typename Box, typename Allocators, typename Tag>
- struct weak_internal_node
- : public weak_node<Value, Parameters, Box, Allocators, Tag>
- {
- typedef rtree::ptr_pair<Box, typename Allocators::node_pointer> element_type;
- typedef typename boost::container::allocator_traits
- <
- typename Allocators::internal_node_allocator_type
- >::template rebind_alloc<element_type> allocator_type;
- typedef boost::container::vector<element_type, allocator_type> elements_type;
- template <typename Al>
- inline weak_internal_node(Al const& al)
- : elements(allocator_type(al))
- {}
- elements_type elements;
- };
- template <typename Value, typename Parameters, typename Box, typename Allocators, typename Tag>
- struct weak_leaf
- : public weak_node<Value, Parameters, Box, Allocators, Tag>
- {
- typedef typename boost::container::allocator_traits
- <
- typename Allocators::leaf_allocator_type
- >::template rebind_alloc<Value> allocator_type;
- typedef boost::container::vector<Value, allocator_type> elements_type;
- template <typename Al>
- inline weak_leaf(Al const& al)
- : elements(allocator_type(al))
- {}
- elements_type elements;
- };
- // nodes traits
- template <typename Value, typename Parameters, typename Box, typename Allocators>
- struct node<Value, Parameters, Box, Allocators, node_weak_dynamic_tag>
- {
- typedef weak_node<Value, Parameters, Box, Allocators, node_weak_dynamic_tag> type;
- };
- template <typename Value, typename Parameters, typename Box, typename Allocators>
- struct internal_node<Value, Parameters, Box, Allocators, node_weak_dynamic_tag>
- {
- typedef weak_internal_node<Value, Parameters, Box, Allocators, node_weak_dynamic_tag> type;
- };
- template <typename Value, typename Parameters, typename Box, typename Allocators>
- struct leaf<Value, Parameters, Box, Allocators, node_weak_dynamic_tag>
- {
- typedef weak_leaf<Value, Parameters, Box, Allocators, node_weak_dynamic_tag> type;
- };
- // visitor traits
- template <typename Value, typename Parameters, typename Box, typename Allocators, bool IsVisitableConst>
- struct visitor<Value, Parameters, Box, Allocators, node_weak_dynamic_tag, IsVisitableConst>
- {
- typedef weak_visitor<Value, Parameters, Box, Allocators, node_weak_dynamic_tag, IsVisitableConst> type;
- };
- // allocators
- template <typename Allocator, typename Value, typename Parameters, typename Box, typename Tag>
- struct internal_node_alloc
- {
- typedef typename internal_node
- <
- Value, Parameters, Box,
- allocators<Allocator, Value, Parameters, Box, Tag>,
- Tag
- >::type node_type;
- typedef typename boost::container::allocator_traits
- <
- Allocator
- >::template rebind_alloc<node_type> type;
- };
- template <typename Allocator, typename Value, typename Parameters, typename Box, typename Tag>
- struct leaf_alloc
- {
- typedef typename leaf
- <
- Value, Parameters, Box,
- allocators<Allocator, Value, Parameters, Box, Tag>,
- Tag
- >::type node_type;
- typedef typename ::boost::container::allocator_traits
- <
- Allocator
- >::template rebind_alloc<node_type> type;
- };
- template <typename Allocator, typename Value, typename Parameters, typename Box, typename Tag>
- struct node_alloc
- {
- typedef typename weak_node
- <
- Value, Parameters, Box,
- allocators<Allocator, Value, Parameters, Box, Tag>,
- Tag
- >::type node_type;
- typedef typename ::boost::container::allocator_traits
- <
- Allocator
- >::template rebind_alloc<node_type> type;
- };
- template <typename Allocator, typename Value, typename Parameters, typename Box>
- class allocators<Allocator, Value, Parameters, Box, node_weak_dynamic_tag>
- : public internal_node_alloc<Allocator, Value, Parameters, Box, node_weak_dynamic_tag>::type
- , public leaf_alloc<Allocator, Value, Parameters, Box, node_weak_dynamic_tag>::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> value_allocator_type;
- typedef boost::container::allocator_traits<value_allocator_type> 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 <typename Alloc>
- 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 <typename Alloc>
- 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 <typename BaseNodePtr, typename Node>
- struct create_weak_node
- {
- template <typename AllocNode>
- static inline BaseNodePtr apply(AllocNode & alloc_node)
- {
- typedef boost::container::allocator_traits<AllocNode> 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<AllocNode> deallocator(p, alloc_node);
- Al::construct(alloc_node, boost::to_address(p), alloc_node);
- deallocator.release();
- return p;
- }
- };
- // destroy_node_impl
- template <typename Node>
- struct destroy_weak_node
- {
- template <typename AllocNode, typename BaseNodePtr>
- static inline void apply(AllocNode & alloc_node, BaseNodePtr n)
- {
- typedef boost::container::allocator_traits<AllocNode> Al;
- typedef typename Al::pointer P;
- P p(&static_cast<Node&>(rtree::get<Node>(*n)));
- Al::destroy(alloc_node, boost::addressof(*p));
- Al::deallocate(alloc_node, p, 1);
- }
- };
- // create_node
- template <typename Allocators, typename Value, typename Parameters, typename Box, typename Tag>
- struct create_node<
- Allocators,
- weak_internal_node<Value, Parameters, Box, Allocators, Tag>
- >
- {
- static inline typename Allocators::node_pointer
- apply(Allocators & allocators)
- {
- return create_weak_node<
- typename Allocators::node_pointer,
- weak_internal_node<Value, Parameters, Box, Allocators, Tag>
- >::apply(allocators.internal_node_allocator());
- }
- };
- template <typename Allocators, typename Value, typename Parameters, typename Box, typename Tag>
- struct create_node<
- Allocators,
- weak_leaf<Value, Parameters, Box, Allocators, Tag>
- >
- {
- static inline typename Allocators::node_pointer
- apply(Allocators & allocators)
- {
- return create_weak_node<
- typename Allocators::node_pointer,
- weak_leaf<Value, Parameters, Box, Allocators, Tag>
- >::apply(allocators.leaf_allocator());
- }
- };
- // destroy_node
- template <typename Allocators, typename Value, typename Parameters, typename Box, typename Tag>
- struct destroy_node<
- Allocators,
- weak_internal_node<Value, Parameters, Box, Allocators, Tag>
- >
- {
- static inline void apply(Allocators & allocators, typename Allocators::node_pointer n)
- {
- destroy_weak_node<
- weak_internal_node<Value, Parameters, Box, Allocators, Tag>
- >::apply(allocators.internal_node_allocator(), n);
- }
- };
- template <typename Allocators, typename Value, typename Parameters, typename Box, typename Tag>
- struct destroy_node<
- Allocators,
- weak_leaf<Value, Parameters, Box, Allocators, Tag>
- >
- {
- static inline void apply(Allocators & allocators, typename Allocators::node_pointer n)
- {
- destroy_weak_node<
- weak_leaf<Value, Parameters, Box, Allocators, Tag>
- >::apply(allocators.leaf_allocator(), n);
- }
- };
- }} // namespace detail::rtree
- }}} // namespace boost::geometry::index
- #endif // BOOST_GEOMETRY_INDEX_DETAIL_RTREE_NODE_WEAK_DYNAMIC_HPP
|