diff --git a/geometry/polyline2d.hpp b/geometry/polyline2d.hpp index 1c67c258a7..9f34a1ca86 100644 --- a/geometry/polyline2d.hpp +++ b/geometry/polyline2d.hpp @@ -129,5 +129,5 @@ public: } }; -typedef Polyline PolylineD; +using PolylineD = Polyline; } // namespace m2 diff --git a/map/framework.cpp b/map/framework.cpp index fc6d551a0d..71f9ea7a84 100644 --- a/map/framework.cpp +++ b/map/framework.cpp @@ -2398,7 +2398,7 @@ void Framework::SetRouterImpl(RouterType type) return m_model.GetIndex().GetMwmIdByCountryFile(CountryFile(countryFile)).IsAlive(); }; - router.reset(new CarRouter(&m_model.GetIndex(), countryFileGetter, + router.reset(new CarRouter(m_model.GetIndex(), countryFileGetter, CreateCarAStarBidirectionalRouter(m_model.GetIndex(), countryFileGetter))); fetcher.reset(new OnlineAbsentCountriesFetcher(countryFileGetter, localFileChecker)); m_routingSession.SetRoutingSettings(routing::GetCarRoutingSettings()); @@ -2552,7 +2552,7 @@ RouterType Framework::GetBestRouter(m2::PointD const & startPoint, m2::PointD co return m_infoGetter->GetRegionCountryId(pt); }; if (!CarRouter::CheckRoutingAbility(startPoint, finalPoint, countryFileGetter, - &m_model.GetIndex())) + m_model.GetIndex())) { return RouterType::Pedestrian; } diff --git a/routing/car_router.cpp b/routing/car_router.cpp index 54632a71de..8c5ef8b5e2 100644 --- a/routing/car_router.cpp +++ b/routing/car_router.cpp @@ -1,4 +1,5 @@ #include "routing/car_router.hpp" + #include "routing/cross_mwm_router.hpp" #include "routing/loaded_path_segment.hpp" #include "routing/online_cross_fetcher.hpp" @@ -197,54 +198,56 @@ private: RoutingMapping & m_routingMapping; }; -bool FindSingleOsrmRoute(FeatureGraphNode const & source, FeatureGraphNode const & target, - RouterDelegate const & delegate, TRoutingMappingPtr & mapping, Index const & index, - Route & mwmRoute) +IRouter::ResultCode FindSingleOsrmRoute(FeatureGraphNode const & source, FeatureGraphNode const & target, + RouterDelegate const & delegate, Index const & index, TRoutingMappingPtr & mapping, + Route & route) { - vector mwmRouteGeometry; - Route::TTurns mwmTurns; - Route::TTimes mwmTimes; - Route::TStreets mwmStreets; + vector geometry; + Route::TTurns turns; + Route::TTimes times; + Route::TStreets streets; LOG(LINFO, ("OSRM route from", MercatorBounds::ToLatLon(source.segmentPoint), "to", MercatorBounds::ToLatLon(target.segmentPoint))); RawRoutingResult routingResult; if (!FindSingleRoute(source, target, mapping->m_dataFacade, routingResult)) - return false; + return routing::IRouter::RouteNotFound; OSRMRoutingResult resultGraph(index, *mapping, routingResult); - if (MakeTurnAnnotation(resultGraph, delegate, mwmRouteGeometry, mwmTurns, mwmTimes, mwmStreets) - != routing::IRouter::NoError) + routing::IRouter::ResultCode const result = + MakeTurnAnnotation(resultGraph, delegate, geometry, turns, times, streets); + if (result != routing::IRouter::NoError) { - LOG(LWARNING, ("Can't load road path data from disk for", mapping->GetCountryName())); - return false; + LOG(LWARNING, ("Can't load road path data from disk for", mapping->GetCountryName(), + ". result code =", result)); + return result; } - mwmRoute.SetTurnInstructions(move(mwmTurns)); - mwmRoute.SetSectionTimes(move(mwmTimes)); - mwmRoute.SetStreetNames(move(mwmStreets)); + route.SetTurnInstructions(move(turns)); + route.SetSectionTimes(move(times)); + route.SetStreetNames(move(streets)); vector mwmPoints; - JunctionsToPoints(mwmRouteGeometry, mwmPoints); - mwmRoute.SetGeometry(mwmPoints.cbegin(), mwmPoints.cend()); + JunctionsToPoints(geometry, mwmPoints); + route.SetGeometry(mwmPoints.cbegin(), mwmPoints.cend()); - return true; + return routing::IRouter::NoError; } } // namespace // static bool CarRouter::CheckRoutingAbility(m2::PointD const & startPoint, m2::PointD const & finalPoint, - TCountryFileFn const & countryFileFn, Index * index) + TCountryFileFn const & countryFileFn, Index & index) { - RoutingIndexManager manager(countryFileFn, *index); + RoutingIndexManager manager(countryFileFn, index); return manager.GetMappingByPoint(startPoint)->IsValid() && manager.GetMappingByPoint(finalPoint)->IsValid(); } -CarRouter::CarRouter(Index * index, TCountryFileFn const & countryFileFn, - unique_ptr roadGraphRouter) - : m_pIndex(index), m_indexManager(countryFileFn, *index), m_aStarRouter(move(roadGraphRouter)) +CarRouter::CarRouter(Index & index, TCountryFileFn const & countryFileFn, + unique_ptr router) + : m_index(index), m_indexManager(countryFileFn, index), m_router(move(router)) { } @@ -258,22 +261,22 @@ void CarRouter::ClearState() m_cachedTargets.clear(); m_cachedTargetPoint = m2::PointD::Zero(); m_indexManager.Clear(); - m_aStarRouter->ClearState(); + m_router->ClearState(); } -bool CarRouter::FindRouteFromCases(TFeatureGraphNodeVec const & source, TFeatureGraphNodeVec const & target, - RouterDelegate const & delegate, TRoutingMappingPtr & mapping, Route & route) +bool CarRouter::FindRouteMsmt(TFeatureGraphNodeVec const & sources, TFeatureGraphNodeVec const & targets, + RouterDelegate const & delegate, TRoutingMappingPtr & mapping, Route & route) { ASSERT(mapping, ()); Route emptyRoute(GetName()); route.Swap(emptyRoute); /// @todo (ldargunov) make more complex nearest edge turnaround - for (auto const & targetEdge : target) + for (auto const & targetEdge : targets) { - for (auto const & sourceEdge : source) + for (auto const & sourceEdge : sources) { - if (FindSingleRouteDispatcher(sourceEdge, targetEdge, delegate, mapping, route)) + if (FindSingleRouteDispatcher(sourceEdge, targetEdge, delegate, mapping, route) == NoError) return true; } } @@ -357,10 +360,12 @@ CarRouter::ResultCode CarRouter::MakeRouteFromCrossesPath(TCheckedPath const & p ASSERT(mwmMapping->IsValid(), ()); MappingGuard mwmMappingGuard(mwmMapping); UNUSED_VALUE(mwmMappingGuard); - CalculatePhantomNodeForCross(mwmMapping, cross.startNode, m_pIndex, true /* forward */); - CalculatePhantomNodeForCross(mwmMapping, cross.finalNode, m_pIndex, false /* forward */); - if (!FindSingleRouteDispatcher(cross.startNode, cross.finalNode, delegate, mwmMapping, route)) - return RouteNotFound; + CalculatePhantomNodeForCross(mwmMapping, cross.startNode, &m_index, true /* forward */); + CalculatePhantomNodeForCross(mwmMapping, cross.finalNode, &m_index, false /* forward */); + IRouter::ResultCode const result = + FindSingleRouteDispatcher(cross.startNode, cross.finalNode, delegate, mwmMapping, route); + if (result != NoError) + return result; } return NoError; @@ -439,7 +444,7 @@ CarRouter::ResultCode CarRouter::CalculateRoute(m2::PointD const & startPoint, delegate.OnProgress(kPointsFoundProgress); // 4. Find route. - double crossCostM = 0; + double crossDistanceM = 0; TCheckedPath finalPath; // Manually load facade to avoid unmaping files we routing on. @@ -453,10 +458,10 @@ CarRouter::ResultCode CarRouter::CalculateRoute(m2::PointD const & startPoint, { indexPair.second->FreeCrossContext(); }); - ResultCode crossCode = CalculateCrossMwmPath(startTask, m_cachedTargets, m_indexManager, crossCostM, + ResultCode crossCode = CalculateCrossMwmPath(startTask, m_cachedTargets, m_indexManager, crossDistanceM, delegate, finalPath); LOG(LINFO, ("Found cross path in", timer.ElapsedNano(), "ns.")); - if (!FindRouteFromCases(startTask, m_cachedTargets, delegate, startMapping, route)) + if (!FindRouteMsmt(startTask, m_cachedTargets, delegate, startMapping, route)) { if (crossCode == NoError) { @@ -469,9 +474,9 @@ CarRouter::ResultCode CarRouter::CalculateRoute(m2::PointD const & startPoint, } INTERRUPT_WHEN_CANCELLED(delegate); - if (crossCode == NoError && crossCostM < route.GetTotalDistanceMeters()) + if (crossCode == NoError && crossDistanceM < route.GetTotalDistanceMeters()) { - LOG(LINFO, ("Cross mwm path shorter. Cross cost:", crossCostM, "single cost:", route.GetTotalDistanceMeters())); + LOG(LINFO, ("Cross mwm path shorter. Cross cost:", crossDistanceM, "single cost:", route.GetTotalDistanceMeters())); auto code = MakeRouteFromCrossesPath(finalPath, delegate, route); LOG(LINFO, ("Made final route in", timer.ElapsedNano(), "ns.")); timer.Reset(); @@ -486,7 +491,7 @@ CarRouter::ResultCode CarRouter::CalculateRoute(m2::PointD const & startPoint, else //4.2 Multiple mwm case { LOG(LINFO, ("Multiple mwm routing case")); - ResultCode code = CalculateCrossMwmPath(startTask, m_cachedTargets, m_indexManager, crossCostM, + ResultCode code = CalculateCrossMwmPath(startTask, m_cachedTargets, m_indexManager, crossDistanceM, delegate, finalPath); timer.Reset(); INTERRUPT_WHEN_CANCELLED(delegate); @@ -515,12 +520,11 @@ IRouter::ResultCode CarRouter::FindPhantomNodes(m2::PointD const & point, TRoutingMappingPtr const & mapping) { ASSERT(mapping, ()); - helpers::Point2PhantomNode getter(*mapping, *m_pIndex, direction); + helpers::Point2PhantomNode getter(*mapping, m_index, direction); getter.SetPoint(point); - m_pIndex->ForEachInRectForMWM(getter, MercatorBounds::RectByCenterXYAndSizeInMeters( - point, kFeatureFindingRectSideRadiusMeters), - scales::GetUpperScale(), mapping->GetMwmId()); + m_index.ForEachInRectForMWM(getter, MercatorBounds::RectByCenterXYAndSizeInMeters( + point, kFeatureFindingRectSideRadiusMeters), scales::GetUpperScale(), mapping->GetMwmId()); if (!getter.HasCandidates()) return RouteNotFound; @@ -529,9 +533,9 @@ IRouter::ResultCode CarRouter::FindPhantomNodes(m2::PointD const & point, return NoError; } -bool CarRouter::IsEdgeIndexExisting(Index::MwmId const & mwmId) +bool CarRouter::DoesEdgeIndexExist(Index::MwmId const & mwmId) { - MwmSet::MwmHandle const handle = m_pIndex->GetMwmHandleById(mwmId); + MwmSet::MwmHandle const handle = m_index.GetMwmHandleById(mwmId); if (!handle.IsAlive()) { ASSERT(false, ("Mwm handle is not alive.")); @@ -539,49 +543,47 @@ bool CarRouter::IsEdgeIndexExisting(Index::MwmId const & mwmId) } MwmValue const * value = handle.GetValue(); - ASSERT(value, ()); + CHECK(value, ()); if (value->GetHeader().GetFormat() < version::Format::v8) return false; - if (value->m_cont.IsExist(EDGE_INDEX_FILE_TAG)) - return true; - return false; + if (!value->m_cont.IsExist(EDGE_INDEX_FILE_TAG)) + return false; + + return true; } -bool CarRouter::FindSingleRouteDispatcher(FeatureGraphNode const & source, FeatureGraphNode const & target, - RouterDelegate const & delegate, TRoutingMappingPtr & mapping, - Route & route) +IRouter::ResultCode CarRouter::FindSingleRouteDispatcher(FeatureGraphNode const & source, FeatureGraphNode const & target, + RouterDelegate const & delegate, TRoutingMappingPtr & mapping, + Route & route) { ASSERT_EQUAL(source.mwmId, target.mwmId, ()); - ASSERT(m_pIndex, ()); - ASSERT(m_aStarRouter, ()); + ASSERT(m_router, ()); + IRouter::ResultCode result = IRouter::NoError; Route mwmRoute(GetName()); // @TODO It's not the best place for checking availability of edge index section in mwm. // Probably it's better to keep if mwm has an edge index section in mwmId. - if (IsEdgeIndexExisting(source.mwmId) && m_aStarRouter) + if (DoesEdgeIndexExist(source.mwmId) && m_router) { // A* routing LOG(LINFO, ("A* route from", MercatorBounds::ToLatLon(source.segmentPoint), "to", MercatorBounds::ToLatLon(target.segmentPoint))); - - if (m_aStarRouter->CalculateRoute(source.segmentPoint, m2::PointD(0, 0), target.segmentPoint, - delegate, mwmRoute) != IRouter::NoError) - { - return false; - } + result = m_router->CalculateRoute(source.segmentPoint, m2::PointD(0, 0) /* direction */, + target.segmentPoint, delegate, mwmRoute); } else { // OSRM Routing // @TODO This branch is implemented to support old maps with osrm section. When osrm // section is not supported this branch should be removed. - if (!FindSingleOsrmRoute(source, target, delegate, mapping, *m_pIndex, mwmRoute)) - return false; + result = FindSingleOsrmRoute(source, target, delegate, m_index, mapping, mwmRoute); } - route.AppendRoute(mwmRoute); - return true; + if (result == IRouter::NoError) + route.AppendRoute(mwmRoute); + + return result; } } // namespace routing diff --git a/routing/car_router.hpp b/routing/car_router.hpp index 7064688f42..1e64f35927 100644 --- a/routing/car_router.hpp +++ b/routing/car_router.hpp @@ -30,7 +30,7 @@ class CarRouter : public IRouter public: typedef vector GeomTurnCandidateT; - CarRouter(Index * index, TCountryFileFn const & countryFileFn, + CarRouter(Index & index, TCountryFileFn const & countryFileFn, unique_ptr roadGraphRouter); virtual string GetName() const override; @@ -49,7 +49,7 @@ public: * @returns true if we can start routing process with a given data. */ static bool CheckRoutingAbility(m2::PointD const & startPoint, m2::PointD const & finalPoint, - TCountryFileFn const & countryFileFn, Index * index); + TCountryFileFn const & countryFileFn, Index & index); protected: /*! @@ -80,35 +80,35 @@ private: // @TODO(bykoianko) When edgeidx section implementation is merged to master // this method should be moved to edge index loader. - bool IsEdgeIndexExisting(Index::MwmId const & mwmId); + bool DoesEdgeIndexExist(Index::MwmId const & mwmId); /*! * \brief Builds a route within one mwm using A* if edge index section is available and osrm otherwise. * Then reconstructs the route and restores all route attributes. * \param route The found route is added to the |route| if the method returns true. - * \return true if route is built and false otherwise. */ - bool FindSingleRouteDispatcher(FeatureGraphNode const & source, FeatureGraphNode const & target, - RouterDelegate const & delegate, TRoutingMappingPtr & mapping, - Route & route); + IRouter::ResultCode FindSingleRouteDispatcher(FeatureGraphNode const & source, FeatureGraphNode const & target, + RouterDelegate const & delegate, TRoutingMappingPtr & mapping, + Route & route); - /*! Finds single shortest path in a single MWM between 2 sets of edges + /*! Finds single shortest path in a single MWM between 2 sets of edges. + * It's a route from multiple sources to multiple targets. * \param source: vector of source edges to make path * \param target: vector of target edges to make path * \param facade: OSRM routing data facade to recover graph information * \param rawRoutingResult: routing result store * \return true when path exists, false otherwise. */ - bool FindRouteFromCases(TFeatureGraphNodeVec const & source, TFeatureGraphNodeVec const & target, - RouterDelegate const & delegate, TRoutingMappingPtr & mapping, Route & route); + bool FindRouteMsmt(TFeatureGraphNodeVec const & source, TFeatureGraphNodeVec const & target, + RouterDelegate const & delegate, TRoutingMappingPtr & mapping, Route & route); - Index const * m_pIndex; + Index & m_index; TFeatureGraphNodeVec m_cachedTargets; m2::PointD m_cachedTargetPoint; RoutingIndexManager m_indexManager; - unique_ptr m_aStarRouter; + unique_ptr m_router; }; } // namespace routing diff --git a/routing/features_road_graph.cpp b/routing/features_road_graph.cpp index 40bf8ba932..25e1cbc0b5 100644 --- a/routing/features_road_graph.cpp +++ b/routing/features_road_graph.cpp @@ -46,7 +46,7 @@ FeaturesRoadGraph::Value::Value(MwmSet::MwmHandle handle) : m_mwmHandle(move(han m_altitudeLoader = make_unique(*m_mwmHandle.GetValue()); } -FeaturesRoadGraph::CrossCountryVehicleModel::CrossCountryVehicleModel(unique_ptr && vehicleModelFactory) +FeaturesRoadGraph::CrossCountryVehicleModel::CrossCountryVehicleModel(unique_ptr vehicleModelFactory) : m_vehicleModelFactory(move(vehicleModelFactory)) , m_maxSpeedKMPH(m_vehicleModelFactory->GetVehicleModel()->GetMaxSpeed()) { @@ -108,7 +108,7 @@ void FeaturesRoadGraph::RoadInfoCache::Clear() } FeaturesRoadGraph::FeaturesRoadGraph(Index const & index, IRoadGraph::Mode mode, - unique_ptr && vehicleModelFactory) + unique_ptr vehicleModelFactory) : m_index(index) , m_mode(mode) , m_vehicleModel(move(vehicleModelFactory)) diff --git a/routing/features_road_graph.hpp b/routing/features_road_graph.hpp index ecde6fb869..9e21db0d87 100644 --- a/routing/features_road_graph.hpp +++ b/routing/features_road_graph.hpp @@ -26,7 +26,7 @@ private: class CrossCountryVehicleModel : public IVehicleModel { public: - CrossCountryVehicleModel(unique_ptr && vehicleModelFactory); + CrossCountryVehicleModel(unique_ptr vehicleModelFactory); // IVehicleModel overrides: double GetSpeed(FeatureType const & f) const override; @@ -59,7 +59,7 @@ private: public: FeaturesRoadGraph(Index const & index, IRoadGraph::Mode mode, - unique_ptr && vehicleModelFactory); + unique_ptr vehicleModelFactory); static uint32_t GetStreetReadScale(); diff --git a/routing/road_graph_router.cpp b/routing/road_graph_router.cpp index 71c2cecd39..c81713a6d6 100644 --- a/routing/road_graph_router.cpp +++ b/routing/road_graph_router.cpp @@ -312,15 +312,15 @@ unique_ptr CreateBicycleAStarBidirectionalRouter(Index & index, TCountr return router; } -unique_ptr CreateCarAStarBidirectionalRouter(Index & index, TCountryFileFn const & countryFileFn) +unique_ptr CreateCarAStarBidirectionalRouter(Index & index, TCountryFileFn const & countryFileFn) { unique_ptr vehicleModelFactory = make_unique(); unique_ptr algorithm = make_unique(); // @TODO Bicycle turn generation engine is used now. It's ok for the time being. // But later a special car turn generation engine should be implemented. unique_ptr directionsEngine = make_unique(index); - unique_ptr router = make_unique( - "astar-bidirectional-bicycle", index, countryFileFn, IRoadGraph::Mode::ObeyOnewayTag, + unique_ptr router = make_unique( + "astar-bidirectional-car", index, countryFileFn, IRoadGraph::Mode::ObeyOnewayTag, move(vehicleModelFactory), move(algorithm), move(directionsEngine)); return router; } diff --git a/routing/road_graph_router.hpp b/routing/road_graph_router.hpp index 88f756fc2b..6d0a2fc67b 100644 --- a/routing/road_graph_router.hpp +++ b/routing/road_graph_router.hpp @@ -56,5 +56,5 @@ private: unique_ptr CreatePedestrianAStarRouter(Index & index, TCountryFileFn const & countryFileFn); unique_ptr CreatePedestrianAStarBidirectionalRouter(Index & index, TCountryFileFn const & countryFileFn); unique_ptr CreateBicycleAStarBidirectionalRouter(Index & index, TCountryFileFn const & countryFileFn); -unique_ptr CreateCarAStarBidirectionalRouter(Index & index, TCountryFileFn const & countryFileFn); +unique_ptr CreateCarAStarBidirectionalRouter(Index & index, TCountryFileFn const & countryFileFn); } // namespace routing diff --git a/routing/route.cpp b/routing/route.cpp index dfc3ee91e1..a60391cda2 100644 --- a/routing/route.cpp +++ b/routing/route.cpp @@ -365,39 +365,43 @@ void Route::AppendRoute(Route const & route) ASSERT(!m_times.empty(), ()); // Remove road end point and turn instruction. + ASSERT_LESS(MercatorBounds::DistanceOnEarth(m_poly.End().m_pt, route.m_poly.Begin().m_pt), 2 /* meters */, ()); m_poly.PopBack(); + CHECK(!m_turns.empty(), ()); + ASSERT_EQUAL(m_turns.back().m_turn, turns::TurnDirection::ReachedYourDestination, ()); m_turns.pop_back(); + CHECK(!m_times.empty(), ()); m_times.pop_back(); // Streets might not point to the last point of the path. } - size_t const polySize = m_poly.GetPolyline().GetSize(); + size_t const indexOffset = m_poly.GetPolyline().GetSize(); // Appending turns. for (auto t : route.m_turns) { if (t.m_index == 0) continue; - t.m_index += polySize; + t.m_index += indexOffset; m_turns.push_back(move(t)); } // Appending street names. - for (TStreetItem s : route.m_streets) + for (auto s : route.m_streets) { if (s.first == 0) continue; - s.first += polySize; + s.first += indexOffset; m_streets.push_back(move(s)); } // Appending times. double const estimationTime = m_times.empty() ? 0.0 : m_times.back().second; - for (TTimeItem t : route.m_times) + for (auto t : route.m_times) { if (t.first == 0) continue; - t.first += polySize; + t.first += indexOffset; t.second += estimationTime; m_times.push_back(move(t)); } diff --git a/routing/route.hpp b/routing/route.hpp index b3fe071212..3e464eb10e 100644 --- a/routing/route.hpp +++ b/routing/route.hpp @@ -63,7 +63,7 @@ public: /// \brief Glues all |route| attributes to |this| except for |m_altitudes|. // @TODO In the future this method should append |m_altitudes| as well. // It's not implemented now because it's not easy to do it and it'll not be used in - // the short future. The problem is routes genetated by osrm have empty |m_altitudes|. + // the short future. The problem is routes generated by osrm have empty |m_altitudes|. // So |m_altitudes| should be filled (with correnct or default values) on osrm route // reconstruction stage to be added with this method. On the other // hand it seems this method'll not be not used for bicycle and pedestrian routing diff --git a/routing/routing_integration_tests/routing_test_tools.cpp b/routing/routing_integration_tests/routing_test_tools.cpp index aae10bd28b..80ad52ad1f 100644 --- a/routing/routing_integration_tests/routing_test_tools.cpp +++ b/routing/routing_integration_tests/routing_test_tools.cpp @@ -81,8 +81,9 @@ namespace integration { return infoGetter.GetRegionCountryId(pt); }; - unique_ptr carRouter(new CarRouter(&index, countryFileGetter, - CreateCarAStarBidirectionalRouter(index, countryFileGetter))); + + auto carRouter = make_unique(index, countryFileGetter, + CreateCarAStarBidirectionalRouter(index, countryFileGetter)); return carRouter; }