596 lines
		
	
	
		
			25 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			596 lines
		
	
	
		
			25 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| #ifndef STATIC_RTREE_HPP
 | |
| #define STATIC_RTREE_HPP
 | |
| 
 | |
| #include "storage/io.hpp"
 | |
| #include "util/bearing.hpp"
 | |
| #include "util/coordinate_calculation.hpp"
 | |
| #include "util/deallocating_vector.hpp"
 | |
| #include "util/exception.hpp"
 | |
| #include "util/hilbert_value.hpp"
 | |
| #include "util/integer_range.hpp"
 | |
| #include "util/rectangle.hpp"
 | |
| #include "util/shared_memory_vector_wrapper.hpp"
 | |
| #include "util/typedefs.hpp"
 | |
| #include "util/web_mercator.hpp"
 | |
| 
 | |
| #include "osrm/coordinate.hpp"
 | |
| 
 | |
| #include "storage/shared_memory.hpp"
 | |
| 
 | |
| #include <boost/assert.hpp>
 | |
| #include <boost/filesystem.hpp>
 | |
| #include <boost/filesystem/fstream.hpp>
 | |
| #include <boost/format.hpp>
 | |
| #include <boost/iostreams/device/mapped_file.hpp>
 | |
| 
 | |
| #include <tbb/parallel_for.h>
 | |
| #include <tbb/parallel_sort.h>
 | |
| 
 | |
| #include <algorithm>
 | |
| #include <array>
 | |
| #include <limits>
 | |
| #include <memory>
 | |
| #include <queue>
 | |
| #include <string>
 | |
| #include <vector>
 | |
| 
 | |
| // An extended alignment is implementation-defined, so use compiler attributes
 | |
| // until alignas(LEAF_PAGE_SIZE) is compiler-independent.
 | |
| #if defined(_MSC_VER)
 | |
| #define ALIGNED(x) __declspec(align(x))
 | |
| #elif defined(__GNUC__)
 | |
| #define ALIGNED(x) __attribute__((aligned(x)))
 | |
| #else
 | |
| #define ALIGNED(x)
 | |
| #endif
 | |
| 
 | |
| namespace osrm
 | |
| {
 | |
| namespace util
 | |
| {
 | |
| 
 | |
| // Static RTree for serving nearest neighbour queries
 | |
| // All coordinates are pojected first to Web Mercator before the bounding boxes
 | |
| // are computed, this means the internal distance metric doesn not represent meters!
 | |
| template <class EdgeDataT,
 | |
|           class CoordinateListT = std::vector<Coordinate>,
 | |
|           osrm::storage::Ownership Ownership = osrm::storage::Ownership::Container,
 | |
|           std::uint32_t BRANCHING_FACTOR = 128,
 | |
|           std::uint32_t LEAF_PAGE_SIZE = 4096>
 | |
| class StaticRTree
 | |
| {
 | |
|   public:
 | |
|     using Rectangle = RectangleInt2D;
 | |
|     using EdgeData = EdgeDataT;
 | |
|     using CoordinateList = CoordinateListT;
 | |
| 
 | |
|     static_assert(LEAF_PAGE_SIZE >= sizeof(uint32_t) + sizeof(EdgeDataT), "page size is too small");
 | |
|     static_assert(((LEAF_PAGE_SIZE - 1) & LEAF_PAGE_SIZE) == 0, "page size is not a power of 2");
 | |
|     static constexpr std::uint32_t LEAF_NODE_SIZE =
 | |
|         (LEAF_PAGE_SIZE - sizeof(uint32_t) - sizeof(Rectangle)) / sizeof(EdgeDataT);
 | |
| 
 | |
|     struct CandidateSegment
 | |
|     {
 | |
|         Coordinate fixed_projected_coordinate;
 | |
|         EdgeDataT data;
 | |
|     };
 | |
| 
 | |
|     struct TreeIndex
 | |
|     {
 | |
|         TreeIndex() : index(0), is_leaf(false) {}
 | |
|         TreeIndex(std::size_t index, bool is_leaf) : index(index), is_leaf(is_leaf) {}
 | |
|         std::uint32_t index : 31;
 | |
|         std::uint32_t is_leaf : 1;
 | |
|     };
 | |
| 
 | |
|     struct TreeNode
 | |
|     {
 | |
|         TreeNode() : child_count(0) {}
 | |
|         std::uint32_t child_count;
 | |
|         Rectangle minimum_bounding_rectangle;
 | |
|         TreeIndex children[BRANCHING_FACTOR];
 | |
|     };
 | |
| 
 | |
|     struct ALIGNED(LEAF_PAGE_SIZE) LeafNode
 | |
|     {
 | |
|         LeafNode() : object_count(0), objects() {}
 | |
|         std::uint32_t object_count;
 | |
|         Rectangle minimum_bounding_rectangle;
 | |
|         std::array<EdgeDataT, LEAF_NODE_SIZE> objects;
 | |
|     };
 | |
|     static_assert(sizeof(LeafNode) == LEAF_PAGE_SIZE, "LeafNode size does not fit the page size");
 | |
| 
 | |
|   private:
 | |
|     struct WrappedInputElement
 | |
|     {
 | |
|         explicit WrappedInputElement(const uint64_t _hilbert_value,
 | |
|                                      const std::uint32_t _array_index)
 | |
|             : m_hilbert_value(_hilbert_value), m_array_index(_array_index)
 | |
|         {
 | |
|         }
 | |
| 
 | |
|         WrappedInputElement() : m_hilbert_value(0), m_array_index(UINT_MAX) {}
 | |
| 
 | |
|         uint64_t m_hilbert_value;
 | |
|         std::uint32_t m_array_index;
 | |
| 
 | |
|         inline bool operator<(const WrappedInputElement &other) const
 | |
|         {
 | |
|             return m_hilbert_value < other.m_hilbert_value;
 | |
|         }
 | |
|     };
 | |
| 
 | |
|     struct QueryCandidate
 | |
|     {
 | |
|         QueryCandidate(std::uint64_t squared_min_dist, TreeIndex tree_index)
 | |
|             : squared_min_dist(squared_min_dist), tree_index(tree_index),
 | |
|               segment_index(std::numeric_limits<std::uint32_t>::max())
 | |
|         {
 | |
|         }
 | |
| 
 | |
|         QueryCandidate(std::uint64_t squared_min_dist,
 | |
|                        TreeIndex tree_index,
 | |
|                        std::uint32_t segment_index,
 | |
|                        const Coordinate &coordinate)
 | |
|             : squared_min_dist(squared_min_dist), tree_index(tree_index),
 | |
|               segment_index(segment_index), fixed_projected_coordinate(coordinate)
 | |
|         {
 | |
|         }
 | |
| 
 | |
|         inline bool is_segment() const
 | |
|         {
 | |
|             return segment_index != std::numeric_limits<std::uint32_t>::max();
 | |
|         }
 | |
| 
 | |
|         inline bool operator<(const QueryCandidate &other) const
 | |
|         {
 | |
|             // Attn: this is reversed order. std::pq is a max pq!
 | |
|             return other.squared_min_dist < squared_min_dist;
 | |
|         }
 | |
| 
 | |
|         std::uint64_t squared_min_dist;
 | |
|         TreeIndex tree_index;
 | |
|         std::uint32_t segment_index;
 | |
|         Coordinate fixed_projected_coordinate;
 | |
|     };
 | |
| 
 | |
|     typename ShM<TreeNode, Ownership>::vector m_search_tree;
 | |
|     const CoordinateListT &m_coordinate_list;
 | |
| 
 | |
|     boost::iostreams::mapped_file_source m_leaves_region;
 | |
|     // read-only view of leaves
 | |
|     typename ShM<const LeafNode, osrm::storage::Ownership::View>::vector m_leaves;
 | |
| 
 | |
|   public:
 | |
|     StaticRTree(const StaticRTree &) = delete;
 | |
|     StaticRTree &operator=(const StaticRTree &) = delete;
 | |
| 
 | |
|     template <typename CoordinateT>
 | |
|     // Construct a packed Hilbert-R-Tree with Kamel-Faloutsos algorithm [1]
 | |
|     explicit StaticRTree(const std::vector<EdgeDataT> &input_data_vector,
 | |
|                          const std::string &tree_node_filename,
 | |
|                          const std::string &leaf_node_filename,
 | |
|                          const std::vector<CoordinateT> &coordinate_list)
 | |
|         : m_coordinate_list(coordinate_list)
 | |
|     {
 | |
|         const uint64_t element_count = input_data_vector.size();
 | |
|         std::vector<WrappedInputElement> input_wrapper_vector(element_count);
 | |
| 
 | |
|         // generate auxiliary vector of hilbert-values
 | |
|         tbb::parallel_for(
 | |
|             tbb::blocked_range<uint64_t>(0, element_count),
 | |
|             [&input_data_vector, &input_wrapper_vector, this](
 | |
|                 const tbb::blocked_range<uint64_t> &range) {
 | |
|                 for (uint64_t element_counter = range.begin(), end = range.end();
 | |
|                      element_counter != end;
 | |
|                      ++element_counter)
 | |
|                 {
 | |
|                     WrappedInputElement ¤t_wrapper = input_wrapper_vector[element_counter];
 | |
|                     current_wrapper.m_array_index = element_counter;
 | |
| 
 | |
|                     EdgeDataT const ¤t_element = input_data_vector[element_counter];
 | |
| 
 | |
|                     // Get Hilbert-Value for centroid in mercartor projection
 | |
|                     BOOST_ASSERT(current_element.u < m_coordinate_list.size());
 | |
|                     BOOST_ASSERT(current_element.v < m_coordinate_list.size());
 | |
| 
 | |
|                     Coordinate current_centroid = coordinate_calculation::centroid(
 | |
|                         m_coordinate_list[current_element.u], m_coordinate_list[current_element.v]);
 | |
|                     current_centroid.lat = FixedLatitude{static_cast<std::int32_t>(
 | |
|                         COORDINATE_PRECISION *
 | |
|                         web_mercator::latToY(toFloating(current_centroid.lat)))};
 | |
| 
 | |
|                     current_wrapper.m_hilbert_value = GetHilbertCode(current_centroid);
 | |
|                 }
 | |
|             });
 | |
| 
 | |
|         // open leaf file
 | |
|         boost::filesystem::ofstream leaf_node_file(leaf_node_filename, std::ios::binary);
 | |
| 
 | |
|         // sort the hilbert-value representatives
 | |
|         tbb::parallel_sort(input_wrapper_vector.begin(), input_wrapper_vector.end());
 | |
|         std::vector<TreeNode> tree_nodes_in_level;
 | |
| 
 | |
|         // pack M elements into leaf node, write to leaf file and add child index to the parent node
 | |
|         uint64_t wrapped_element_index = 0;
 | |
|         for (std::uint32_t node_index = 0; wrapped_element_index < element_count; ++node_index)
 | |
|         {
 | |
|             TreeNode current_node;
 | |
|             for (std::uint32_t leaf_index = 0;
 | |
|                  leaf_index < BRANCHING_FACTOR && wrapped_element_index < element_count;
 | |
|                  ++leaf_index)
 | |
|             {
 | |
|                 LeafNode current_leaf;
 | |
|                 Rectangle &rectangle = current_leaf.minimum_bounding_rectangle;
 | |
|                 for (std::uint32_t object_index = 0;
 | |
|                      object_index < LEAF_NODE_SIZE && wrapped_element_index < element_count;
 | |
|                      ++object_index, ++wrapped_element_index)
 | |
|                 {
 | |
|                     const std::uint32_t input_object_index =
 | |
|                         input_wrapper_vector[wrapped_element_index].m_array_index;
 | |
|                     const EdgeDataT &object = input_data_vector[input_object_index];
 | |
| 
 | |
|                     current_leaf.object_count += 1;
 | |
|                     current_leaf.objects[object_index] = object;
 | |
| 
 | |
|                     Coordinate projected_u{
 | |
|                         web_mercator::fromWGS84(Coordinate{m_coordinate_list[object.u]})};
 | |
|                     Coordinate projected_v{
 | |
|                         web_mercator::fromWGS84(Coordinate{m_coordinate_list[object.v]})};
 | |
| 
 | |
|                     BOOST_ASSERT(std::abs(toFloating(projected_u.lon).operator double()) <= 180.);
 | |
|                     BOOST_ASSERT(std::abs(toFloating(projected_u.lat).operator double()) <= 180.);
 | |
|                     BOOST_ASSERT(std::abs(toFloating(projected_v.lon).operator double()) <= 180.);
 | |
|                     BOOST_ASSERT(std::abs(toFloating(projected_v.lat).operator double()) <= 180.);
 | |
| 
 | |
|                     rectangle.min_lon =
 | |
|                         std::min(rectangle.min_lon, std::min(projected_u.lon, projected_v.lon));
 | |
|                     rectangle.max_lon =
 | |
|                         std::max(rectangle.max_lon, std::max(projected_u.lon, projected_v.lon));
 | |
| 
 | |
|                     rectangle.min_lat =
 | |
|                         std::min(rectangle.min_lat, std::min(projected_u.lat, projected_v.lat));
 | |
|                     rectangle.max_lat =
 | |
|                         std::max(rectangle.max_lat, std::max(projected_u.lat, projected_v.lat));
 | |
| 
 | |
|                     BOOST_ASSERT(rectangle.IsValid());
 | |
|                 }
 | |
| 
 | |
|                 // append the leaf node to the current tree node
 | |
|                 current_node.child_count += 1;
 | |
|                 current_node.children[leaf_index] =
 | |
|                     TreeIndex{node_index * BRANCHING_FACTOR + leaf_index, true};
 | |
|                 current_node.minimum_bounding_rectangle.MergeBoundingBoxes(
 | |
|                     current_leaf.minimum_bounding_rectangle);
 | |
| 
 | |
|                 // write leaf_node to leaf node file
 | |
|                 leaf_node_file.write((char *)¤t_leaf, sizeof(current_leaf));
 | |
|             }
 | |
| 
 | |
|             tree_nodes_in_level.emplace_back(current_node);
 | |
|         }
 | |
|         leaf_node_file.flush();
 | |
|         leaf_node_file.close();
 | |
| 
 | |
|         std::uint32_t processing_level = 0;
 | |
|         while (1 < tree_nodes_in_level.size())
 | |
|         {
 | |
|             std::vector<TreeNode> tree_nodes_in_next_level;
 | |
|             std::uint32_t processed_tree_nodes_in_level = 0;
 | |
|             while (processed_tree_nodes_in_level < tree_nodes_in_level.size())
 | |
|             {
 | |
|                 TreeNode parent_node;
 | |
|                 // pack BRANCHING_FACTOR elements into tree_nodes each
 | |
|                 for (std::uint32_t current_child_node_index = 0;
 | |
|                      current_child_node_index < BRANCHING_FACTOR;
 | |
|                      ++current_child_node_index)
 | |
|                 {
 | |
|                     if (processed_tree_nodes_in_level < tree_nodes_in_level.size())
 | |
|                     {
 | |
|                         TreeNode ¤t_child_node =
 | |
|                             tree_nodes_in_level[processed_tree_nodes_in_level];
 | |
|                         // add tree node to parent entry
 | |
|                         parent_node.children[current_child_node_index] =
 | |
|                             TreeIndex{m_search_tree.size(), false};
 | |
|                         m_search_tree.emplace_back(current_child_node);
 | |
|                         // merge MBRs
 | |
|                         parent_node.minimum_bounding_rectangle.MergeBoundingBoxes(
 | |
|                             current_child_node.minimum_bounding_rectangle);
 | |
|                         // increase counters
 | |
|                         ++parent_node.child_count;
 | |
|                         ++processed_tree_nodes_in_level;
 | |
|                     }
 | |
|                 }
 | |
|                 tree_nodes_in_next_level.emplace_back(parent_node);
 | |
|             }
 | |
|             tree_nodes_in_level.swap(tree_nodes_in_next_level);
 | |
|             ++processing_level;
 | |
|         }
 | |
|         BOOST_ASSERT_MSG(tree_nodes_in_level.size() == 1, "tree broken, more than one root node");
 | |
|         // last remaining entry is the root node, store it
 | |
|         m_search_tree.emplace_back(tree_nodes_in_level[0]);
 | |
| 
 | |
|         // reverse and renumber tree to have root at index 0
 | |
|         std::reverse(m_search_tree.begin(), m_search_tree.end());
 | |
| 
 | |
|         std::uint32_t search_tree_size = m_search_tree.size();
 | |
|         tbb::parallel_for(
 | |
|             tbb::blocked_range<std::uint32_t>(0, search_tree_size),
 | |
|             [this, &search_tree_size](const tbb::blocked_range<std::uint32_t> &range) {
 | |
|                 for (std::uint32_t i = range.begin(), end = range.end(); i != end; ++i)
 | |
|                 {
 | |
|                     TreeNode ¤t_tree_node = this->m_search_tree[i];
 | |
|                     for (std::uint32_t j = 0; j < current_tree_node.child_count; ++j)
 | |
|                     {
 | |
|                         if (!current_tree_node.children[j].is_leaf)
 | |
|                         {
 | |
|                             const std::uint32_t old_id = current_tree_node.children[j].index;
 | |
|                             const std::uint32_t new_id = search_tree_size - old_id - 1;
 | |
|                             current_tree_node.children[j].index = new_id;
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
|             });
 | |
| 
 | |
|         // open tree file
 | |
|         storage::io::FileWriter tree_node_file(tree_node_filename,
 | |
|                                                storage::io::FileWriter::HasNoFingerprint);
 | |
| 
 | |
|         std::uint64_t size_of_tree = m_search_tree.size();
 | |
|         BOOST_ASSERT_MSG(0 < size_of_tree, "tree empty");
 | |
| 
 | |
|         tree_node_file.WriteOne(size_of_tree);
 | |
|         tree_node_file.WriteFrom(&m_search_tree[0], size_of_tree);
 | |
| 
 | |
|         MapLeafNodesFile(leaf_node_filename);
 | |
|     }
 | |
| 
 | |
|     explicit StaticRTree(const boost::filesystem::path &node_file,
 | |
|                          const boost::filesystem::path &leaf_file,
 | |
|                          const CoordinateListT &coordinate_list)
 | |
|         : m_coordinate_list(coordinate_list)
 | |
|     {
 | |
|         storage::io::FileReader tree_node_file(node_file,
 | |
|                                                storage::io::FileReader::HasNoFingerprint);
 | |
| 
 | |
|         const auto tree_size = tree_node_file.ReadElementCount64();
 | |
| 
 | |
|         m_search_tree.resize(tree_size);
 | |
|         tree_node_file.ReadInto(&m_search_tree[0], tree_size);
 | |
| 
 | |
|         MapLeafNodesFile(leaf_file);
 | |
|     }
 | |
| 
 | |
|     explicit StaticRTree(TreeNode *tree_node_ptr,
 | |
|                          const uint64_t number_of_nodes,
 | |
|                          const boost::filesystem::path &leaf_file,
 | |
|                          const CoordinateListT &coordinate_list)
 | |
|         : m_search_tree(tree_node_ptr, number_of_nodes), m_coordinate_list(coordinate_list)
 | |
|     {
 | |
|         MapLeafNodesFile(leaf_file);
 | |
|     }
 | |
| 
 | |
|     void MapLeafNodesFile(const boost::filesystem::path &leaf_file)
 | |
|     {
 | |
|         // open leaf node file and return a pointer to the mapped leaves data
 | |
|         try
 | |
|         {
 | |
|             m_leaves_region.open(leaf_file);
 | |
|             std::size_t num_leaves = m_leaves_region.size() / sizeof(LeafNode);
 | |
|             auto data_ptr = m_leaves_region.data();
 | |
|             BOOST_ASSERT(reinterpret_cast<uintptr_t>(data_ptr) % alignof(LeafNode) == 0);
 | |
|             m_leaves.reset(reinterpret_cast<const LeafNode *>(data_ptr), num_leaves);
 | |
|         }
 | |
|         catch (const std::exception &exc)
 | |
|         {
 | |
|             throw exception(boost::str(boost::format("Leaf file %1% mapping failed: %2%") %
 | |
|                                        leaf_file % exc.what()) +
 | |
|                             SOURCE_REF);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     /* Returns all features inside the bounding box.
 | |
|        Rectangle needs to be projected!*/
 | |
|     std::vector<EdgeDataT> SearchInBox(const Rectangle &search_rectangle) const
 | |
|     {
 | |
|         const Rectangle projected_rectangle{
 | |
|             search_rectangle.min_lon,
 | |
|             search_rectangle.max_lon,
 | |
|             toFixed(FloatLatitude{
 | |
|                 web_mercator::latToY(toFloating(FixedLatitude(search_rectangle.min_lat)))}),
 | |
|             toFixed(FloatLatitude{
 | |
|                 web_mercator::latToY(toFloating(FixedLatitude(search_rectangle.max_lat)))})};
 | |
|         std::vector<EdgeDataT> results;
 | |
| 
 | |
|         std::queue<TreeIndex> traversal_queue;
 | |
|         traversal_queue.push(TreeIndex{});
 | |
| 
 | |
|         while (!traversal_queue.empty())
 | |
|         {
 | |
|             auto const current_tree_index = traversal_queue.front();
 | |
|             traversal_queue.pop();
 | |
| 
 | |
|             if (current_tree_index.is_leaf)
 | |
|             {
 | |
|                 const LeafNode ¤t_leaf_node = m_leaves[current_tree_index.index];
 | |
| 
 | |
|                 for (const auto i : irange(0u, current_leaf_node.object_count))
 | |
|                 {
 | |
|                     const auto ¤t_edge = current_leaf_node.objects[i];
 | |
| 
 | |
|                     // we don't need to project the coordinates here,
 | |
|                     // because we use the unprojected rectangle to test against
 | |
|                     const Rectangle bbox{std::min(m_coordinate_list[current_edge.u].lon,
 | |
|                                                   m_coordinate_list[current_edge.v].lon),
 | |
|                                          std::max(m_coordinate_list[current_edge.u].lon,
 | |
|                                                   m_coordinate_list[current_edge.v].lon),
 | |
|                                          std::min(m_coordinate_list[current_edge.u].lat,
 | |
|                                                   m_coordinate_list[current_edge.v].lat),
 | |
|                                          std::max(m_coordinate_list[current_edge.u].lat,
 | |
|                                                   m_coordinate_list[current_edge.v].lat)};
 | |
| 
 | |
|                     // use the _unprojected_ input rectangle here
 | |
|                     if (bbox.Intersects(search_rectangle))
 | |
|                     {
 | |
|                         results.push_back(current_edge);
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 const TreeNode ¤t_tree_node = m_search_tree[current_tree_index.index];
 | |
| 
 | |
|                 // If it's a tree node, look at all children and add them
 | |
|                 // to the search queue if their bounding boxes intersect
 | |
|                 for (std::uint32_t i = 0; i < current_tree_node.child_count; ++i)
 | |
|                 {
 | |
|                     const TreeIndex child_id = current_tree_node.children[i];
 | |
|                     const auto &child_rectangle =
 | |
|                         child_id.is_leaf ? m_leaves[child_id.index].minimum_bounding_rectangle
 | |
|                                          : m_search_tree[child_id.index].minimum_bounding_rectangle;
 | |
| 
 | |
|                     if (child_rectangle.Intersects(projected_rectangle))
 | |
|                     {
 | |
|                         traversal_queue.push(child_id);
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|         return results;
 | |
|     }
 | |
| 
 | |
|     // Override filter and terminator for the desired behaviour.
 | |
|     std::vector<EdgeDataT> Nearest(const Coordinate input_coordinate,
 | |
|                                    const std::size_t max_results) const
 | |
|     {
 | |
|         return Nearest(input_coordinate,
 | |
|                        [](const CandidateSegment &) { return std::make_pair(true, true); },
 | |
|                        [max_results](const std::size_t num_results, const CandidateSegment &) {
 | |
|                            return num_results >= max_results;
 | |
|                        });
 | |
|     }
 | |
| 
 | |
|     // Override filter and terminator for the desired behaviour.
 | |
|     template <typename FilterT, typename TerminationT>
 | |
|     std::vector<EdgeDataT> Nearest(const Coordinate input_coordinate,
 | |
|                                    const FilterT filter,
 | |
|                                    const TerminationT terminate) const
 | |
|     {
 | |
|         std::vector<EdgeDataT> results;
 | |
|         auto projected_coordinate = web_mercator::fromWGS84(input_coordinate);
 | |
|         Coordinate fixed_projected_coordinate{projected_coordinate};
 | |
| 
 | |
|         // initialize queue with root element
 | |
|         std::priority_queue<QueryCandidate> traversal_queue;
 | |
|         traversal_queue.push(QueryCandidate{0, TreeIndex{}});
 | |
| 
 | |
|         while (!traversal_queue.empty())
 | |
|         {
 | |
|             QueryCandidate current_query_node = traversal_queue.top();
 | |
|             traversal_queue.pop();
 | |
| 
 | |
|             const TreeIndex ¤t_tree_index = current_query_node.tree_index;
 | |
|             if (!current_query_node.is_segment())
 | |
|             { // current object is a tree node
 | |
|                 if (current_tree_index.is_leaf)
 | |
|                 {
 | |
|                     ExploreLeafNode(current_tree_index,
 | |
|                                     fixed_projected_coordinate,
 | |
|                                     projected_coordinate,
 | |
|                                     traversal_queue);
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     ExploreTreeNode(
 | |
|                         current_tree_index, fixed_projected_coordinate, traversal_queue);
 | |
|                 }
 | |
|             }
 | |
|             else
 | |
|             { // current candidate is an actual road segment
 | |
|                 auto edge_data =
 | |
|                     m_leaves[current_tree_index.index].objects[current_query_node.segment_index];
 | |
|                 const auto ¤t_candidate =
 | |
|                     CandidateSegment{current_query_node.fixed_projected_coordinate, edge_data};
 | |
| 
 | |
|                 // to allow returns of no-results if too restrictive filtering, this needs to be
 | |
|                 // done here even though performance would indicate that we want to stop after
 | |
|                 // adding the first candidate
 | |
|                 if (terminate(results.size(), current_candidate))
 | |
|                 {
 | |
|                     break;
 | |
|                 }
 | |
| 
 | |
|                 auto use_segment = filter(current_candidate);
 | |
|                 if (!use_segment.first && !use_segment.second)
 | |
|                 {
 | |
|                     continue;
 | |
|                 }
 | |
|                 edge_data.forward_segment_id.enabled &= use_segment.first;
 | |
|                 edge_data.reverse_segment_id.enabled &= use_segment.second;
 | |
| 
 | |
|                 // store phantom node in result vector
 | |
|                 results.push_back(std::move(edge_data));
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         return results;
 | |
|     }
 | |
| 
 | |
|   private:
 | |
|     template <typename QueueT>
 | |
|     void ExploreLeafNode(const TreeIndex &leaf_id,
 | |
|                          const Coordinate &projected_input_coordinate_fixed,
 | |
|                          const FloatCoordinate &projected_input_coordinate,
 | |
|                          QueueT &traversal_queue) const
 | |
|     {
 | |
|         const LeafNode ¤t_leaf_node = m_leaves[leaf_id.index];
 | |
| 
 | |
|         // current object represents a block on disk
 | |
|         for (const auto i : irange(0u, current_leaf_node.object_count))
 | |
|         {
 | |
|             const auto ¤t_edge = current_leaf_node.objects[i];
 | |
|             const auto projected_u = web_mercator::fromWGS84(m_coordinate_list[current_edge.u]);
 | |
|             const auto projected_v = web_mercator::fromWGS84(m_coordinate_list[current_edge.v]);
 | |
| 
 | |
|             FloatCoordinate projected_nearest;
 | |
|             std::tie(std::ignore, projected_nearest) =
 | |
|                 coordinate_calculation::projectPointOnSegment(
 | |
|                     projected_u, projected_v, projected_input_coordinate);
 | |
| 
 | |
|             const auto squared_distance = coordinate_calculation::squaredEuclideanDistance(
 | |
|                 projected_input_coordinate_fixed, projected_nearest);
 | |
|             // distance must be non-negative
 | |
|             BOOST_ASSERT(0. <= squared_distance);
 | |
|             traversal_queue.push(
 | |
|                 QueryCandidate{squared_distance, leaf_id, i, Coordinate{projected_nearest}});
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     template <class QueueT>
 | |
|     void ExploreTreeNode(const TreeIndex &parent_id,
 | |
|                          const Coordinate &fixed_projected_input_coordinate,
 | |
|                          QueueT &traversal_queue) const
 | |
|     {
 | |
|         const TreeNode &parent = m_search_tree[parent_id.index];
 | |
|         for (std::uint32_t i = 0; i < parent.child_count; ++i)
 | |
|         {
 | |
|             const TreeIndex child_id = parent.children[i];
 | |
|             const auto &child_rectangle =
 | |
|                 child_id.is_leaf ? m_leaves[child_id.index].minimum_bounding_rectangle
 | |
|                                  : m_search_tree[child_id.index].minimum_bounding_rectangle;
 | |
|             const auto squared_lower_bound_to_element =
 | |
|                 child_rectangle.GetMinSquaredDist(fixed_projected_input_coordinate);
 | |
|             traversal_queue.push(QueryCandidate{squared_lower_bound_to_element, child_id});
 | |
|         }
 | |
|     }
 | |
| };
 | |
| 
 | |
| //[1] "On Packing R-Trees"; I. Kamel, C. Faloutsos; 1993; DOI: 10.1145/170088.170403
 | |
| //[2] "Nearest Neighbor Queries", N. Roussopulos et al; 1995; DOI: 10.1145/223784.223794
 | |
| //[3] "Distance Browsing in Spatial Databases"; G. Hjaltason, H. Samet; 1999; ACM Trans. DB Sys
 | |
| // Vol.24 No.2, pp.265-318
 | |
| }
 | |
| }
 | |
| 
 | |
| #endif // STATIC_RTREE_HPP
 |