diff --git a/openlr/CMakeLists.txt b/openlr/CMakeLists.txt index 025e80ed57..d9961881cf 100644 --- a/openlr/CMakeLists.txt +++ b/openlr/CMakeLists.txt @@ -18,6 +18,8 @@ set( openlr_model.hpp openlr_model_xml.cpp openlr_model_xml.hpp + paths_connector.cpp + paths_connector.hpp road_info_getter.cpp road_info_getter.hpp road_type_checkers.cpp diff --git a/openlr/candidate_paths_getter.cpp b/openlr/candidate_paths_getter.cpp index 21e9bb8f4e..be7a952049 100644 --- a/openlr/candidate_paths_getter.cpp +++ b/openlr/candidate_paths_getter.cpp @@ -169,8 +169,6 @@ bool CandidatePathsGetter::GetLineCandidatesForPoints( void CandidatePathsGetter::GetStartLines(vector const & points, bool const isLastPoint, Graph::EdgeVector & edges) { - ScopedTimer t(m_stats.m_startLinesTime); - for (auto const & pc : points) { if (!isLastPoint) @@ -188,8 +186,6 @@ void CandidatePathsGetter::GetAllSuitablePaths(Graph::EdgeVector const & startLi FunctionalRoadClass const frc, vector & allPaths) { - ScopedTimer t(m_stats.m_allStatPathsTime); - queue q; for (auto const & e : startLines) @@ -246,8 +242,6 @@ void CandidatePathsGetter::GetBestCandidatePaths( vector const & allPaths, bool const isLastPoint, uint32_t const requiredBearing, uint32_t const bearDistM, m2::PointD const & startPoint, vector & candidates) { - ScopedTimer t(m_stats.m_bestCandidatePathsTime); - set candidatePaths; set fakeEndingsCandidatePaths; @@ -330,8 +324,6 @@ void CandidatePathsGetter::GetLineCandidates(openlr::LocationReferencePoint cons vector const & pointCandidates, vector & candidates) { - ScopedTimer t(m_stats.m_lineCandidatesTime); - uint32_t const kDefaultBearDistM = 25; uint32_t const bearDistM = min(kDefaultBearDistM, distanceToNextPoint); diff --git a/openlr/candidate_points_getter.cpp b/openlr/candidate_points_getter.cpp index 1a8168d09d..275f7c0d84 100644 --- a/openlr/candidate_points_getter.cpp +++ b/openlr/candidate_points_getter.cpp @@ -13,8 +13,6 @@ 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; diff --git a/openlr/helpers.cpp b/openlr/helpers.cpp index 9343472e8b..7f6403819f 100644 --- a/openlr/helpers.cpp +++ b/openlr/helpers.cpp @@ -6,8 +6,6 @@ #include "geometry/mercator.hpp" -#include "base/timer.hpp" - #include #include #include diff --git a/openlr/helpers.hpp b/openlr/helpers.hpp index b784a3eab9..475415f5c7 100644 --- a/openlr/helpers.hpp +++ b/openlr/helpers.hpp @@ -5,8 +5,6 @@ #include "geometry/point2d.hpp" -#include "base/timer.hpp" - #include namespace openlr @@ -32,18 +30,6 @@ 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; -}; - bool PassesRestriction(Graph::Edge const & e, FunctionalRoadClass const restriction, int const frcThreshold, RoadInfoGetter & infoGetter); - } // namespace openlr diff --git a/openlr/paths_connector.cpp b/openlr/paths_connector.cpp new file mode 100644 index 0000000000..ddc1494581 --- /dev/null +++ b/openlr/paths_connector.cpp @@ -0,0 +1,247 @@ +#include "openlr/paths_connector.hpp" +#include "openlr/helpers.hpp" + +#include "base/checked_cast.hpp" +#include "base/stl_iterator.hpp" + +#include +#include +#include +#include +#include + +using namespace std; + +namespace openlr +{ +namespace +{ +size_t IntersectionLen(Graph::EdgeVector a, Graph::EdgeVector b) +{ + sort(begin(a), end(a)); + sort(begin(b), end(b)); + return set_intersection(begin(a), end(a), begin(b), end(b), CounterIterator()).GetCount(); +} + +bool PrefEqualsSuff(Graph::EdgeVector const & a, Graph::EdgeVector const & b, size_t const len) +{ + ASSERT_LESS_OR_EQUAL(len, a.size(), ()); + ASSERT_LESS_OR_EQUAL(len, b.size(), ()); + return equal(end(a) - len, end(a), begin(b)); +} + +// Returns a length of the longest suffix of |a| that matches any prefix of |b|. +// Neither |a| nor |b| can contain several repetitions of any edge. +// Returns -1 if |a| intersection |b| is not equal to some suffix of |a| and some prefix of |b|. +int32_t PathOverlappingLen(Graph::EdgeVector const & a, Graph::EdgeVector const & b) +{ + auto const len = IntersectionLen(a, b); + if (PrefEqualsSuff(a, b, len)) + return base::checked_cast(len); + return -1; +} + +bool ValidatePath(Graph::EdgeVector const & path, + uint32_t const distanceToNextPoint, + double const pathLengthTolerance) +{ + + double pathLen = 0.0; + for (auto const & e : path) + pathLen += EdgeLength(e); + + double pathDiffPercent = AbsDifference(static_cast(distanceToNextPoint), pathLen) / + static_cast(distanceToNextPoint); + + LOG(LDEBUG, ("Validating path:", LogAs2GisPath(path))); + + if (pathDiffPercent > pathLengthTolerance) + { + LOG(LDEBUG, + ("Shortest path doest not meet required length constraints, error:", pathDiffPercent)); + return false; + } + + return true; +} +} // namespace + +PathsConnector::PathsConnector(double const pathLengthTolerance, Graph const & graph, + v2::Stats & stat) + : m_pathLengthTolerance(pathLengthTolerance), m_graph(graph), m_stat(stat) +{ +} + +bool PathsConnector::ConnectCandidates(vector const & points, + vector> const & lineCandidates, + vector & resultPath) +{ + ASSERT(!points.empty(), ()); + + resultPath.resize(points.size() - 1); + + // TODO(mgsergio): Discard last point on failure. + // TODO(mgsergio): Do not iterate more than kMaxRetries times. + // TODO(mgserigio): Make kMaxRetries depend on points number in the segment. + for (size_t i = 1; i < points.size(); ++i) + { + bool found = false; + + auto const & point = points[i - 1]; + auto const distanceToNextPoint = point.m_distanceToNextPoint; + auto const & fromCandidates = lineCandidates[i - 1]; + auto const & toCandidates = lineCandidates[i]; + auto & resultPathPart = resultPath[i - 1]; + + Graph::EdgeVector fakePath; + + for (size_t fromInd = 0; fromInd < fromCandidates.size() && !found; ++fromInd) + { + for (size_t toInd = 0; toInd < toCandidates.size() && !found; ++toInd) + { + resultPathPart.clear(); + + found = ConnectAdjacentCandidateLines(fromCandidates[fromInd], toCandidates[toInd], + point.m_functionalRoadClass, distanceToNextPoint, + resultPathPart); + + if (!found) + continue; + + found = ValidatePath(resultPathPart, distanceToNextPoint, m_pathLengthTolerance); + if (fakePath.empty() && found && + (resultPathPart.front().IsFake() || resultPathPart.back().IsFake())) + { + fakePath = resultPathPart; + found = false; + } + } + } + + if (!fakePath.empty() && !found) + { + found = true; + resultPathPart = fakePath; + } + + if (!found) + { + LOG(LDEBUG, ("No shortest path found")); + ++m_stat.m_noShortestPathFound; + resultPathPart.clear(); + return false; + } + } + + ASSERT_EQUAL(resultPath.size(), points.size() - 1, ()); + + return true; +} + +bool PathsConnector::FindShortestPath(Graph::Edge const & from, Graph::Edge const & to, + FunctionalRoadClass const frc, uint32_t const maxPathLength, + Graph::EdgeVector & path) +{ + // TODO(mgsergio): Turn Dijkstra to A*. + + uint32_t const kLengthToleranceM = 10; + + struct State + { + State(Graph::Edge const & e, uint32_t const s) : m_edge(e), m_score(s) {} + + bool operator>(State const & o) const + { + return make_tuple(m_score, m_edge) > make_tuple(o.m_score, o.m_edge); + } + + Graph::Edge m_edge; + uint32_t m_score; + }; + + priority_queue, greater> q; + map scores; + map links; + + q.emplace(from, 0); + scores[from] = 0; + + while (!q.empty()) + { + auto const state = q.top(); + q.pop(); + + auto const & u = state.m_edge; + // TODO(mgsergio): Unify names: use either score or distance. + auto const us = state.m_score; + + if (us > maxPathLength + kLengthToleranceM) + continue; + + if (us > scores[u]) + continue; + + if (u == to) + { + for (auto e = u; e != from; e = links[e]) + path.push_back(e); + path.push_back(from); + reverse(begin(path), end(path)); + return true; + } + + Graph::EdgeVector edges; + m_graph.GetOutgoingEdges(u.GetEndJunction(), edges); + for (auto const & e : edges) + { + // TODO(mgsergio): Use frc to filter edges. + + auto const it = scores.find(e); + auto const eScore = us + EdgeLength(e); + if (it == end(scores) || it->second > eScore) + { + scores[e] = eScore; + links[e] = u; + q.emplace(e, eScore); + } + } + } + + return false; +} + +bool PathsConnector::ConnectAdjacentCandidateLines(Graph::EdgeVector const & from, + Graph::EdgeVector const & to, + FunctionalRoadClass const frc, + uint32_t const distanceToNextPoint, + Graph::EdgeVector & resultPath) + +{ + ASSERT(!to.empty(), ()); + + if (auto const skip = PathOverlappingLen(from, to)) + { + if (skip == -1) + return false; + copy(begin(from), end(from), back_inserter(resultPath)); + copy(begin(to) + skip, end(to), back_inserter(resultPath)); + return true; + } + + ASSERT(from.back() != to.front(), ()); + + Graph::EdgeVector shortestPath; + auto const found = + FindShortestPath(from.back(), to.front(), frc, distanceToNextPoint, shortestPath); + if (!found) + return false; + + // Skip the last edge from |from| because it already took its place at begin(shortestPath). + copy(begin(from), prev(end(from)), back_inserter(resultPath)); + copy(begin(shortestPath), end(shortestPath), back_inserter(resultPath)); + // Skip the first edge from |to| because it already took its place at prev(end(shortestPath)). + copy(next(begin(to)), end(to), back_inserter(resultPath)); + + return found; +} +} // namespace openlr diff --git a/openlr/paths_connector.hpp b/openlr/paths_connector.hpp new file mode 100644 index 0000000000..ff48942123 --- /dev/null +++ b/openlr/paths_connector.hpp @@ -0,0 +1,36 @@ +#pragma once + +#include "openlr/graph.hpp" +#include "openlr/openlr_model.hpp" +#include "openlr/stats.hpp" + +#include +#include +#include + +namespace openlr +{ +class PathsConnector +{ +public: + PathsConnector(double const pathLengthTolerance, Graph const & graph, v2::Stats & stat); + + bool ConnectCandidates(std::vector const & points, + std::vector> const & lineCandidates, + std::vector & resultPath); + +private: + bool FindShortestPath(Graph::Edge const & from, Graph::Edge const & to, + FunctionalRoadClass const frc, uint32_t const maxPathLength, + Graph::EdgeVector & path); + + bool ConnectAdjacentCandidateLines(Graph::EdgeVector const & from, Graph::EdgeVector const & to, + FunctionalRoadClass const frc, + uint32_t const distanceToNextPoint, + Graph::EdgeVector & resultPath); + + double const m_pathLengthTolerance; + Graph const & m_graph; + v2::Stats & m_stat; +}; +} // namespace openlr diff --git a/openlr/stats.hpp b/openlr/stats.hpp index 5c6301759e..90f4d8ac2c 100644 --- a/openlr/stats.hpp +++ b/openlr/stats.hpp @@ -16,13 +16,6 @@ struct Stats 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