From 0055557332fcc4bcfd82763909931e11c8ea8f36 Mon Sep 17 00:00:00 2001 From: Sergey Magidovich Date: Mon, 18 Dec 2017 13:12:07 +0300 Subject: [PATCH] [OpenLR] Add CandidatePointGetter and staff. [Squash me]. --- base/math.hpp | 4 +- openlr/CMakeLists.txt | 6 +++ openlr/candidate_points_getter.cpp | 86 ++++++++++++++++++++++++++++++ openlr/candidate_points_getter.hpp | 51 ++++++++++++++++++ openlr/graph.cpp | 54 +++++++++++++++++++ openlr/graph.hpp | 48 +++++++++++++++++ openlr/helpers.hpp | 83 ++++++++++++++++++++++++++++ openlr/openlr_decoder.cpp | 9 ++-- openlr/stats.hpp | 28 ++++++++++ routing/features_road_graph.cpp | 2 + routing/features_road_graph.hpp | 5 ++ routing/road_graph.cpp | 16 ++++-- routing/road_graph.hpp | 2 + 13 files changed, 384 insertions(+), 10 deletions(-) create mode 100644 openlr/candidate_points_getter.cpp create mode 100644 openlr/candidate_points_getter.hpp create mode 100644 openlr/graph.cpp create mode 100644 openlr/graph.hpp create mode 100644 openlr/helpers.hpp create mode 100644 openlr/stats.hpp diff --git a/base/math.hpp b/base/math.hpp index 15f7359ad0..9dd4ac5e66 100644 --- a/base/math.hpp +++ b/base/math.hpp @@ -18,7 +18,7 @@ namespace math double constexpr twicePi = 2. * pi; template T sqr(T t) { return (t*t); } -} +} // namespace math namespace my { @@ -215,4 +215,4 @@ int constexpr Sign(Number const number) noexcept { return number == 0 ? 0 : number > 0 ? 1 : -1; } -} +} // namespace my diff --git a/openlr/CMakeLists.txt b/openlr/CMakeLists.txt index a7ab919166..c9cb897ad7 100644 --- a/openlr/CMakeLists.txt +++ b/openlr/CMakeLists.txt @@ -2,8 +2,13 @@ project(openlr) set( SRC + candidate_points_getter.cpp + candidate_points_getter.hpp decoded_path.cpp decoded_path.hpp + graph.cpp + graph.hpp + helpers.hpp openlr_decoder.cpp openlr_decoder.hpp openlr_model.cpp @@ -16,6 +21,7 @@ set( road_type_checkers.hpp router.cpp router.hpp + stats.hpp way_point.hpp ) diff --git a/openlr/candidate_points_getter.cpp b/openlr/candidate_points_getter.cpp new file mode 100644 index 0000000000..1a8168d09d --- /dev/null +++ b/openlr/candidate_points_getter.cpp @@ -0,0 +1,86 @@ +#include "openlr/candidate_points_getter.hpp" + +#include "openlr/helpers.hpp" + +#include "routing/road_graph.hpp" + +#include "storage/country_info_getter.hpp" + +using namespace routing; + +namespace openlr +{ +void CandidatePointsGetter::GetJunctionPointCandidates(m2::PointD const & p, + vector & candidates) +{ + ScopedTimer t(m_stat.m_pointCadtidatesTime); + + // TODO(mgsergio): Get optimal value using experiments on a sample. + // Or start with small radius and scale it up when there are too few points. + size_t const kRectSideMeters = 110; + + auto const mwmId = m_mwmIdByPointFn(p); + + auto const rect = MercatorBounds::RectByCenterXYAndSizeInMeters(p, kRectSideMeters); + auto const selectCandidates = [&rect, &candidates](FeatureType & ft) { + ft.ParseGeometry(FeatureType::BEST_GEOMETRY); + ft.ForEachPoint( + [&rect, &candidates](m2::PointD const & candidate) { + if (rect.IsPointInside(candidate)) + candidates.emplace_back(candidate); + }, + scales::GetUpperScale()); + }; + + m_index.ForEachInRect(selectCandidates, rect, scales::GetUpperScale()); + + // TODO: Move this to a separate stage. + // 1030292476 Does not match. Some problem occur with points. + // Either points duplicatate or something alike. Check this + // later. The idea to fix this was to move SortUnique to the stage + // after enriching with projections. + + my::SortUnique(candidates, + [&p](m2::PointD const & a, m2::PointD const & b) { + return MercatorBounds::DistanceOnEarth(a, p) < + MercatorBounds::DistanceOnEarth(b, p); + }, + [](m2::PointD const & a, m2::PointD const & b) { return a == b; }); + + LOG(LDEBUG, + (candidates.size(), "candidate points are found for point", MercatorBounds::ToLatLon(p))); + candidates.resize(min(m_maxJunctionCandidates, candidates.size())); + LOG(LDEBUG, + (candidates.size(), "candidates points are remained for point", MercatorBounds::ToLatLon(p))); +} + +void CandidatePointsGetter::EnrichWithProjectionPoints(m2::PointD const & p, + vector & candidates) +{ + m_graph.ResetFakes(); + + vector> vicinities; + m_graph.FindClosestEdges(p, m_maxProjectionCandidates, vicinities); + for (auto const & v : vicinities) + { + auto const & edge = v.first; + auto const & junction = v.second; + + ASSERT(edge.HasRealPart() && !edge.IsFake(), ()); + + if (PointsAreClose(edge.GetStartPoint(), junction.GetPoint()) || + PointsAreClose(edge.GetEndPoint(), junction.GetPoint())) + { + continue; + } + + auto const firstHalf = Edge::MakeFake(edge.GetStartJunction(), junction, edge); + auto const secondHalf = Edge::MakeFake(junction, edge.GetEndJunction(), edge); + + m_graph.AddIngoingFakeEdge(firstHalf); + m_graph.AddOutgoingFakeEdge(secondHalf); + candidates.push_back(junction.GetPoint()); + } + LOG(LDEBUG, (vicinities.size(), "projections candidates were added")); +} +} // namespace openlr diff --git a/openlr/candidate_points_getter.hpp b/openlr/candidate_points_getter.hpp new file mode 100644 index 0000000000..213d214ef3 --- /dev/null +++ b/openlr/candidate_points_getter.hpp @@ -0,0 +1,51 @@ +#pragma once + +#include "openlr/graph.hpp" +#include "openlr/stats.hpp" + +#include "indexer/index.hpp" + +#include "geometry/point2d.hpp" + +#include +#include +#include + +namespace openlr +{ +class CandidatePointsGetter +{ + using MwmIdByPointFn = std::function; + +public: + CandidatePointsGetter(size_t const maxJunctionCandidates, size_t const maxProjectionCandidates, + Index const & index, MwmIdByPointFn const & mwmIdByPointFn, Graph & graph, + v2::Stats & stat) + : m_maxJunctionCandidates(maxJunctionCandidates) + , m_maxProjectionCandidates(maxProjectionCandidates) + , m_index(index) + , m_mwmIdByPointFn(mwmIdByPointFn) + , m_graph(graph) + , m_stat(stat) + { + } + + void GetCandidatePoints(m2::PointD const & p, std::vector & candidates) + { + GetJunctionPointCandidates(p, candidates); + EnrichWithProjectionPoints(p, candidates); + } + +private: + void GetJunctionPointCandidates(m2::PointD const & p, std::vector & candidates); + void EnrichWithProjectionPoints(m2::PointD const & p, std::vector & candidates); + + size_t const m_maxJunctionCandidates; + size_t const m_maxProjectionCandidates; + + Index const & m_index; + Graph & m_graph; + v2::Stats & m_stat; + MwmIdByPointFn m_mwmIdByPointFn; +}; +} // namespace openlr diff --git a/openlr/graph.cpp b/openlr/graph.cpp new file mode 100644 index 0000000000..2114d3de65 --- /dev/null +++ b/openlr/graph.cpp @@ -0,0 +1,54 @@ +#include "openlr/graph.hpp" + +#include "indexer/index.hpp" + +using namespace routing; + +namespace openlr +{ +Graph::Graph(Index const & index, shared_ptr carModelFactory) + : m_graph(index, IRoadGraph::Mode::ObeyOnewayTag, carModelFactory) +{ +} + +void Graph::GetOutgoingEdges(Junction const & junction, EdgeVector & edges) const +{ + m_graph.GetOutgoingEdges(junction, edges); +} + +void Graph::GetIngoingEdges(Junction const & junction, EdgeVector & edges) const +{ + m_graph.GetIngoingEdges(junction, edges); +} + +void Graph::GetRegularOutgoingEdges(Junction const & junction, EdgeVector & edges) const +{ + m_graph.GetRegularOutgoingEdges(junction, edges); +} + +void Graph::GetRegularIngoingEdges(Junction const & junction, EdgeVector & edges) const +{ + m_graph.GetRegularIngoingEdges(junction, edges); +} + +void Graph::FindClosestEdges(m2::PointD const & point, uint32_t count, + vector> & vicinities) const +{ + m_graph.FindClosestEdges(point, count, vicinities); +} + +void Graph::AddFakeEdges(Junction const & junction, vector> const & vicinities) +{ + m_graph.AddFakeEdges(junction, vicinities); +} + +void Graph::AddIngoingFakeEdge(Edge const & e) +{ + m_graph.AddIngoingFakeEdge(e); +} + +void Graph::AddOutgoingFakeEdge(Edge const & e) +{ + m_graph.AddOutgoingFakeEdge(e); +} +} // namespace openlr diff --git a/openlr/graph.hpp b/openlr/graph.hpp new file mode 100644 index 0000000000..6908ff993b --- /dev/null +++ b/openlr/graph.hpp @@ -0,0 +1,48 @@ +#pragma once + +#include "routing/features_road_graph.hpp" +#include "routing/road_graph.hpp" + +#include "routing_common/car_model.hpp" + +#include "geometry/point2d.hpp" + +#include +#include +#include + +class Index; + +namespace openlr +{ +// TODO(mgsergio): Inherit from FeaturesRoadGraph. +class Graph +{ +public: + using Edge = routing::Edge; + using EdgeVector = routing::FeaturesRoadGraph::TEdgeVector; + using Junction = routing::Junction; + + Graph(Index const & index, std::shared_ptr carModelFactory); + + void GetOutgoingEdges(routing::Junction const & junction, EdgeVector & edges) const; + void GetIngoingEdges(routing::Junction const & junction, EdgeVector & edges) const; + + void GetRegularIngoingEdges(Junction const & junction, EdgeVector & edges) const; + void GetRegularOutgoingEdges(Junction const & junction, EdgeVector & edges) const; + + void FindClosestEdges(m2::PointD const & point, uint32_t count, + std::vector> & vicinities) const; + + void AddFakeEdges(Junction const & junction, + std::vector> const & vicinities); + + void AddIngoingFakeEdge(Edge const & e); + void AddOutgoingFakeEdge(Edge const & e); + + void ResetFakes() { m_graph.ResetFakes(); } + +private: + routing::FeaturesRoadGraph m_graph; +}; +} // namespace openlr diff --git a/openlr/helpers.hpp b/openlr/helpers.hpp new file mode 100644 index 0000000000..9071b1d092 --- /dev/null +++ b/openlr/helpers.hpp @@ -0,0 +1,83 @@ +#pragma once + +#include "openlr/graph.hpp" + +#include "routing/features_road_graph.hpp" + +#include "geometry/mercator.hpp" +#include "geometry/point2d.hpp" + +#include "base/timer.hpp" + +#include +#include +#include + +namespace openlr +{ +inline bool PointsAreClose(m2::PointD const & p1, m2::PointD const & p2) +{ + double const kMwmRoadCrossingRadiusMeters = routing::GetRoadCrossingRadiusMeters(); + return MercatorBounds::DistanceOnEarth(p1, p2) < kMwmRoadCrossingRadiusMeters; +} + +// TODO(mgsergio): Try to use double instead of uint32_t and leave whait is better. +inline uint32_t EdgeLength(Graph::Edge const & e) +{ + return static_cast(MercatorBounds::DistanceOnEarth(e.GetStartPoint(), e.GetEndPoint())); +} + +inline bool EdgesAraAlmostEqual(Graph::Edge const & e1, Graph::Edge const & e2) +{ + // TODO(mgsergio): Do I need to check fields other than points? + return PointsAreClose(e1.GetStartPoint(), e2.GetStartPoint()) && + PointsAreClose(e1.GetEndPoint(), e2.GetEndPoint()); +} + +// TODO(mgsergio): Remove when unused. +inline std::string LogAs2GisPath(Graph::EdgeVector const & path) +{ + CHECK(!path.empty(), ("Paths should not be empty")); + + std::ostringstream ost; + ost << "https://2gis.ru/moscow?queryState="; + + auto ll = MercatorBounds::ToLatLon(path.front().GetStartPoint()); + ost << "center%2F" << ll.lon << "%2C" << ll.lat << "%2F"; + ost << "zoom%2F" << 17 <<"%2F"; + ost << "ruler%2Fpoints%2F"; + for (auto const & e : path) + { + ll = MercatorBounds::ToLatLon(e.GetStartPoint()); + ost << ll.lon << "%20" << ll.lat << "%2C"; + } + ll = MercatorBounds::ToLatLon(path.back().GetEndPoint()); + ost << ll.lon << "%20" << ll.lat; + + return ost.str(); +} + +inline std::string LogAs2GisPath(Graph::Edge const & e) +{ + return LogAs2GisPath(Graph::EdgeVector({e})); +} + +template < + typename T, typename U, + typename std::enable_if::value ^ std::is_signed::value), int>::type = 0> +typename std::common_type::type AbsDifference(T const a, U const b) +{ + return a >= b ? a - b : b - a; +} + +// TODO(mgsergio): Remove when/if unused. +class ScopedTimer : private my::Timer +{ +public: + ScopedTimer(std::chrono::milliseconds & ms) : m_ms(ms) {} + ~ScopedTimer() { m_ms += TimeElapsedAs(); } + +private: + std::chrono::milliseconds & m_ms; +}; +} // namespace openlr diff --git a/openlr/openlr_decoder.cpp b/openlr/openlr_decoder.cpp index a318a3d179..6c4ab9932a 100644 --- a/openlr/openlr_decoder.cpp +++ b/openlr/openlr_decoder.cpp @@ -19,10 +19,11 @@ #include "base/logging.hpp" #include "base/math.hpp" -#include "std/fstream.hpp" -#include "std/thread.hpp" +#include +#include using namespace routing; +using namespace std; namespace openlr { @@ -83,8 +84,8 @@ OpenLRDecoder::OpenLRDecoder( { } -void OpenLRDecoder::Decode(std::vector const & segments, - uint32_t const numThreads, std::vector & paths) +void OpenLRDecoder::Decode(vector const & segments, uint32_t const numThreads, + vector & paths) { double const kOffsetToleranceM = 10; diff --git a/openlr/stats.hpp b/openlr/stats.hpp new file mode 100644 index 0000000000..5c6301759e --- /dev/null +++ b/openlr/stats.hpp @@ -0,0 +1,28 @@ +#pragma once + +#include +#include + +namespace openlr +{ +namespace v2 +{ +struct Stats +{ + uint32_t m_routesHandled = 0; + uint32_t m_routesFailed = 0; + uint32_t m_noCandidateFound = 0; + uint32_t m_noShortestPathFound = 0; + uint32_t m_wrongOffsets = 0; + // Number of zeroed distance-to-next point values in the input. + uint32_t m_dnpIsZero = 0; + + std::chrono::milliseconds m_pointCadtidatesTime{0}; + std::chrono::milliseconds m_startLinesTime{0}; + std::chrono::milliseconds m_lineCandidatesTime{0}; + std::chrono::milliseconds m_allStatPathsTime{0}; + std::chrono::milliseconds m_bestCandidatePathsTime{0}; + std::chrono::milliseconds m_shortestPathsTime{0}; +}; +} // namespace V2 +} // namespace openlr diff --git a/routing/features_road_graph.cpp b/routing/features_road_graph.cpp index ea5cdd7b09..da1e6c9ef9 100644 --- a/routing/features_road_graph.cpp +++ b/routing/features_road_graph.cpp @@ -27,6 +27,8 @@ double constexpr kMwmCrossingNodeEqualityRadiusMeters = 100.0; } // namespace +double GetRoadCrossingRadiusMeters() { return kMwmRoadCrossingRadiusMeters; } + FeaturesRoadGraph::Value::Value(Index const & index, MwmSet::MwmHandle handle) : m_mwmHandle(move(handle)) { diff --git a/routing/features_road_graph.hpp b/routing/features_road_graph.hpp index 39148d116a..6d06208faf 100644 --- a/routing/features_road_graph.hpp +++ b/routing/features_road_graph.hpp @@ -120,4 +120,9 @@ private: mutable map m_mwmLocks; }; +// @returns a distance d such as that for a given point p any edge +// with start point s such as that |s - p| < d, and edge is considered outgouing from p. +// Symmetrically for ingoing edges. +double GetRoadCrossingRadiusMeters(); + } // namespace routing diff --git a/routing/road_graph.cpp b/routing/road_graph.cpp index 3fa096a51e..3512759679 100644 --- a/routing/road_graph.cpp +++ b/routing/road_graph.cpp @@ -266,14 +266,22 @@ void IRoadGraph::AddFakeEdges(Junction const & junction, for (auto const & uv : edges) { - auto const & u = uv.GetStartJunction(); - auto const & v = uv.GetEndJunction(); - AddEdge(u, uv, m_fakeOutgoingEdges); - AddEdge(v, uv, m_fakeIngoingEdges); + AddOutgoingFakeEdge(uv); + AddIngoingFakeEdge(uv); } } } +void IRoadGraph::AddOutgoingFakeEdge(Edge const & e) +{ + AddEdge(e.GetStartJunction(), e, m_fakeOutgoingEdges); +} + +void IRoadGraph::AddIngoingFakeEdge(Edge const & e) +{ + AddEdge(e.GetEndJunction(), e, m_fakeIngoingEdges); +} + double IRoadGraph::GetSpeedKMPH(Edge const & edge) const { double const speedKMPH = (edge.IsFake() ? GetMaxSpeedKMPH() : GetSpeedKMPH(edge.GetFeatureId())); diff --git a/routing/road_graph.hpp b/routing/road_graph.hpp index 6cc61943ea..4c414a2875 100644 --- a/routing/road_graph.hpp +++ b/routing/road_graph.hpp @@ -272,6 +272,8 @@ public: /// Adds fake edges from fake position rp to real vicinity /// positions. void AddFakeEdges(Junction const & junction, vector> const & vicinities); + void AddOutgoingFakeEdge(Edge const & e); + void AddIngoingFakeEdge(Edge const & e); /// Returns RoadInfo for a road corresponding to featureId. virtual RoadInfo GetRoadInfo(FeatureID const & featureId) const = 0;