From f3908f761bd6eb32095bae6d028d915ceb701c6f Mon Sep 17 00:00:00 2001 From: Vladimir Byko-Ianko Date: Fri, 14 Jun 2019 11:32:26 +0300 Subject: [PATCH] [routing] Fixing routing quality tests. --- generator/transit_generator.cpp | 3 +- routing/fake_ending.cpp | 6 +- routing/index_router.cpp | 85 +++++++++++-------- routing/index_router.hpp | 19 ++--- routing/nearest_edge_finder.cpp | 9 +- routing/nearest_edge_finder.hpp | 2 +- .../leaps_postprocessing_tests.cpp | 2 +- 7 files changed, 69 insertions(+), 57 deletions(-) diff --git a/generator/transit_generator.cpp b/generator/transit_generator.cpp index 4475dfb1a9..76cf499c3e 100644 --- a/generator/transit_generator.cpp +++ b/generator/transit_generator.cpp @@ -114,7 +114,8 @@ void CalculateBestPedestrianSegments(string const & mwmPath, CountryId const & c if (indexRouter.FindBestSegments(gate.GetPoint(), m2::PointD::Zero() /* direction */, true /* isOutgoing */, *worldGraph, bestSegments)) { - // IndexRouter::FindBestSegments() returns a few good segments but the best one is the first. + // IndexRouter::FindBestSegments() returns a few good segments but the best one is the + // first. // @TODO It should be considered to change the format of transit section to keep all // candidates for every gate. CHECK(!bestSegments.empty(), ()); diff --git a/routing/fake_ending.cpp b/routing/fake_ending.cpp index e096701bc2..6ec3d756b1 100644 --- a/routing/fake_ending.cpp +++ b/routing/fake_ending.cpp @@ -52,7 +52,8 @@ FakeEnding MakeFakeEnding(vector const & segments, m2::PointD const & p auto const & backJunction = graph.GetJunction(segment, false /* front */); auto const & projectedJunction = CalcProjectionToSegment(backJunction, frontJunction, point); - ending.m_projections.emplace_back(segment, oneWay, frontJunction, backJunction, projectedJunction); + ending.m_projections.emplace_back(segment, oneWay, frontJunction, backJunction, + projectedJunction); averageAltitude = (i * averageAltitude + projectedJunction.GetAltitude()) / (i + 1); } @@ -71,7 +72,8 @@ FakeEnding MakeFakeEnding(Segment const & segment, m2::PointD const & point, Ind FakeEnding ending; ending.m_originJunction = Junction(point, projectedJunction.GetAltitude()); - ending.m_projections.emplace_back(segment, oneWay, frontJunction, backJunction, projectedJunction); + ending.m_projections.emplace_back(segment, oneWay, frontJunction, backJunction, + projectedJunction); return ending; } } // namespace routing diff --git a/routing/index_router.cpp b/routing/index_router.cpp index ac4986a6cb..fd2c54c0fe 100644 --- a/routing/index_router.cpp +++ b/routing/index_router.cpp @@ -469,8 +469,9 @@ RouterResultCode IndexRouter::DoCalculateRoute(Checkpoints const & checkpoints, subroutes.emplace_back(subrouteStarter.GetStartJunction(), subrouteStarter.GetFinishJunction(), subrouteSegmentsBegin, subrouteSegmentsEnd); subrouteSegmentsBegin = subrouteSegmentsEnd; - // For every subroute except for the first one the last real segment is used as a start segment. - // It's implemented this way to prevent jumping from one road to another one using a via point. + // For every subroute except for the first one the last real segment is used as a start + // segment. It's implemented this way to prevent jumping from one road to another one using a + // via point. startSegments.resize(1); bool const hasRealOrPart = GetLastRealOrPart(subrouteStarter, subroute, startSegments[0]); CHECK(hasRealOrPart, ("No real or part of real segments in route.")); @@ -815,7 +816,7 @@ unique_ptr IndexRouter::MakeWorldGraph() } bool IndexRouter::IsFencedOff(m2::PointD const & point, pair const & edgeProjection, - vector const & fences) const + vector const & fences) const { auto const & edge = edgeProjection.first; auto const & projPoint = edgeProjection.second.GetPoint(); @@ -832,7 +833,8 @@ bool IndexRouter::IsFencedOff(m2::PointD const & point, pair con continue; } - if (m2::SegmentsIntersect(point, projPoint, fencePointFrom.GetPoint(), fencePointTo.GetPoint())) + if (m2::SegmentsIntersect(point, projPoint, fencePointFrom.GetPoint(), + fencePointTo.GetPoint())) return true; } } @@ -840,17 +842,15 @@ bool IndexRouter::IsFencedOff(m2::PointD const & point, pair con } void IndexRouter::FetchRoadGeom(m2::RectD const & rect, - vector & roadFeatureGeoms) const + vector & roadFeatureGeoms) const { - auto const roadFetcher = [this, &roadFeatureGeoms](FeatureType & ft) - { + auto const roadFetcher = [this, &roadFeatureGeoms](FeatureType & ft) { if (!m_roadGraph.IsRoad(ft)) return; - // Skipping all the features of zero or one point. auto roadGeom = m_roadGraph.GetRoadGeom(ft); - if (roadGeom.size() < 2) - return; + + CHECK_GREATER_OR_EQUAL(roadGeom.size(), 2, ()); roadFeatureGeoms.emplace_back(move(roadGeom)); }; @@ -858,30 +858,31 @@ void IndexRouter::FetchRoadGeom(m2::RectD const & rect, m_dataSource.ForEachInRect(roadFetcher, rect, scales::GetUpperScale()); } -bool IndexRouter::FindClosetCodirectianalEdge(BestEdgeComparator const & bestEdgeComparator, - vector> const & candidates, - Edge & closetCodirectianalEdge) const +bool IndexRouter::FindClosestCodirectionalEdge(BestEdgeComparator const & bestEdgeComparator, + vector> const & candidates, + Edge & closestCodirectionalEdge) const { double constexpr kInvalidDist = numeric_limits::max(); - double squareDistToClosetCodirectianalEdgeM = kInvalidDist; + double squareDistToClosestCodirectionalEdgeM = kInvalidDist; - if (bestEdgeComparator.IsDirectionValid()) + if (!bestEdgeComparator.IsDirectionValid()) + return false; + + for (auto const & c : candidates) { - for (auto const & c : candidates) - { - auto const & edge = c.first; - if (bestEdgeComparator.IsAlmostCodirectional(edge)) - { - double const distM = bestEdgeComparator.GetSquaredDist(edge); - if (distM < squareDistToClosetCodirectianalEdgeM) - { - closetCodirectianalEdge = edge; - squareDistToClosetCodirectianalEdgeM = distM; - } - } - } + auto const & edge = c.first; + if (!bestEdgeComparator.IsAlmostCodirectional(edge)) + continue; + + double const distM = bestEdgeComparator.GetSquaredDist(edge); + if (distM >= squareDistToClosestCodirectionalEdgeM) + continue; + + closestCodirectionalEdge = edge; + squareDistToClosestCodirectionalEdgeM = distM; } - return squareDistToClosetCodirectianalEdgeM != kInvalidDist; + + return squareDistToClosestCodirectionalEdgeM != kInvalidDist; } bool IndexRouter::FindBestSegments(m2::PointD const & point, m2::PointD const & direction, @@ -900,6 +901,15 @@ bool IndexRouter::FindBestSegments(m2::PointD const & point, m2::PointD const & vector> candidates; m_roadGraph.FindClosestEdges(point, kMaxRoadCandidates, candidates); + // Removing all candidates which are far from |point|. + base::EraseIf(candidates, [&point](pair const & p) { + auto const dist = MercatorBounds::DistanceOnEarth(point, p.second.GetPoint()); + return dist > FeaturesRoadGraph::kClosestEdgesRadiusM; + }); + + if (candidates.empty()) + return false; + auto const getSegmentByEdge = [&numMwmId](Edge const & edge) { return Segment(numMwmId, edge.GetFeatureId().m_index, edge.GetSegId(), edge.IsForward()); }; @@ -916,8 +926,9 @@ bool IndexRouter::FindBestSegments(m2::PointD const & point, m2::PointD const & // Removing all candidates which are fenced off by the road graph from |point|. vector roadFeatureGeoms; - FetchRoadGeom(MercatorBounds::RectByCenterXYAndSizeInMeters(point, - FeaturesRoadGraph::kClosestEdgesRadiusM), roadFeatureGeoms); + FetchRoadGeom( + MercatorBounds::RectByCenterXYAndSizeInMeters(point, FeaturesRoadGraph::kClosestEdgesRadiusM), + roadFeatureGeoms); base::EraseIf(candidates, [&](pair const & candidate) { return IsFencedOff(point, candidate, roadFeatureGeoms); @@ -927,23 +938,23 @@ bool IndexRouter::FindBestSegments(m2::PointD const & point, m2::PointD const & return false; // Looking for the closest codirectional edge. If it's not found add all good candidates. - Edge closetCodirectianalEdge; + Edge closestCodirectionalEdge; BestEdgeComparator bestEdgeComparator(point, direction); bestSegmentIsAlmostCodirectional = - FindClosetCodirectianalEdge(bestEdgeComparator, candidates, closetCodirectianalEdge); + FindClosestCodirectionalEdge(bestEdgeComparator, candidates, closestCodirectionalEdge); if (bestSegmentIsAlmostCodirectional) { - bestSegments = {getSegmentByEdge(closetCodirectianalEdge)}; + bestSegments = {getSegmentByEdge(closestCodirectionalEdge)}; } else { bestSegments.clear(); // Sorting from better candidates to worse ones. sort(candidates.begin(), candidates.end(), - [&bestEdgeComparator](pair const & lhs, pair const & rhs){ - return bestEdgeComparator.Compare(lhs.first, rhs.first) < 0; - }); + [&bestEdgeComparator](pair const & lhs, pair const & rhs) { + return bestEdgeComparator.Compare(lhs.first, rhs.first) < 0; + }); for (auto const & c : candidates) bestSegments.emplace_back(getSegmentByEdge(c.first)); } diff --git a/routing/index_router.hpp b/routing/index_router.hpp index eaf34d6e31..73a5c10708 100644 --- a/routing/index_router.hpp +++ b/routing/index_router.hpp @@ -32,8 +32,8 @@ #include #include #include -#include #include +#include class DataSource; @@ -76,9 +76,8 @@ public: traffic::TrafficCache const & trafficCache, DataSource & dataSource); std::unique_ptr MakeSingleMwmWorldGraph(); - bool FindBestSegments(m2::PointD const & point, m2::PointD const & direction, - bool isOutgoing, WorldGraph & worldGraph, - std::vector & bestSegments); + bool FindBestSegments(m2::PointD const & point, m2::PointD const & direction, bool isOutgoing, + WorldGraph & worldGraph, std::vector & bestSegments); // IRouter overrides: std::string GetName() const override { return m_name; } @@ -104,18 +103,18 @@ private: /// \returns true if a segment (|point|, |edgeProjection.second|) crosses one of segments /// in |fences| except for a one which has the same geometry with |edgeProjection.first|. bool IsFencedOff(m2::PointD const & point, std::pair const & edgeProjection, - std::vector const & fences) const; + std::vector const & fences) const; /// \brief Filling |roadFeatureGeoms| with geometry of road features which lies in |rect|. /// \note Some additional road features which lies near |rect| may be added to |roadFeatureGeoms|. void FetchRoadGeom(m2::RectD const & rect, - vector & roadFeatureGeoms) const; + vector & roadFeatureGeoms) const; - /// \brief Fills |closetCodirectianalEdge| with a codirectianal edge which is closet to + /// \brief Fills |closestCodirectionalEdge| with a codirectional edge which is closet to /// |BestEdgeComparator::point| and returns true if there's any. If not returns false. - bool FindClosetCodirectianalEdge(BestEdgeComparator const & bestEdgeComparator, - std::vector> const & candidates, - Edge & closetCodirectianalEdge) const; + bool FindClosestCodirectionalEdge(BestEdgeComparator const & bestEdgeComparator, + std::vector> const & candidates, + Edge & closestCodirectionalEdge) const; /// \brief Finds the best segments (edges) which may be considered as starts or finishes /// of the route. According to current implementation the closest to |point| segment which diff --git a/routing/nearest_edge_finder.cpp b/routing/nearest_edge_finder.cpp index 4c33ede89c..7127a2d323 100644 --- a/routing/nearest_edge_finder.cpp +++ b/routing/nearest_edge_finder.cpp @@ -25,15 +25,14 @@ void NearestEdgeFinder::AddInformationSource(FeatureID const & featureId, ASSERT_GREATER(count, 1, ()); for (size_t i = 1; i < count; ++i) { - // @todo Probably, we need to get exact projection distance in meters. // @TODO All the calculations below should be done after the closest point is found. // Otherway we do unnecessary calculation significant number of times. m2::ParametrizedSegment segment(junctions[i - 1].GetPoint(), junctions[i].GetPoint()); m2::PointD const pt = segment.ClosestPointTo(m_point); - double const d = m_point.SquaredLength(pt); - if (d < res.m_dist) + double const squaredDist = m_point.SquaredLength(pt); + if (squaredDist < res.m_squaredDist) { Junction const & segStart = junctions[i - 1]; Junction const & segEnd = junctions[i]; @@ -53,7 +52,7 @@ void NearestEdgeFinder::AddInformationSource(FeatureID const & featureId, projPointAlt = startAlt + static_cast((endAlt - startAlt) * distFromStartM / segLenM); } - res.m_dist = d; + res.m_squaredDist = squaredDist; res.m_fid = featureId; res.m_segId = static_cast(i - 1); res.m_segStart = segStart; @@ -75,7 +74,7 @@ void NearestEdgeFinder::MakeResult(vector> & res, size_t co { sort(m_candidates.begin(), m_candidates.end(), [](Candidate const & r1, Candidate const & r2) { - return (r1.m_dist < r2.m_dist); + return (r1.m_squaredDist < r2.m_squaredDist); }); res.clear(); diff --git a/routing/nearest_edge_finder.hpp b/routing/nearest_edge_finder.hpp index 1374ae2c81..aa6026ad86 100644 --- a/routing/nearest_edge_finder.hpp +++ b/routing/nearest_edge_finder.hpp @@ -25,7 +25,7 @@ class NearestEdgeFinder { struct Candidate { - double m_dist = std::numeric_limits::max(); + double m_squaredDist = std::numeric_limits::max(); uint32_t m_segId = 0; Junction m_segStart; Junction m_segEnd; diff --git a/routing/routing_quality/routing_quality_tests/leaps_postprocessing_tests.cpp b/routing/routing_quality/routing_quality_tests/leaps_postprocessing_tests.cpp index 2451defc42..3fbf87e97c 100644 --- a/routing/routing_quality/routing_quality_tests/leaps_postprocessing_tests.cpp +++ b/routing/routing_quality/routing_quality_tests/leaps_postprocessing_tests.cpp @@ -16,7 +16,7 @@ namespace { UNIT_TEST(RoutingQuality_NoLoop_MoscowToKazan) { - TEST(!CheckCarRoute({55.63113, 37.63054} /* start */, {55.68213, 52.37379} /* finish */, + TEST(!CheckCarRoute({55.63113, 37.63054} /* start */, {55.67914, 52.37389} /* finish */, {{{55.80643, 37.83981}}} /* reference point */), ()); }