From 8cb081727dda229dc25742cb914c40bca68a1651 Mon Sep 17 00:00:00 2001 From: Vladimir Byko-Ianko Date: Fri, 14 Oct 2016 10:48:00 +0300 Subject: [PATCH] Creating mixed router instead of osrm router. --- geometry/polyline2d.hpp | 12 + map/framework.cpp | 3 +- routing/base/followed_polyline.hpp | 12 + routing/osrm_router.cpp | 207 +++++++++--------- routing/osrm_router.hpp | 46 ++-- routing/road_graph_router.cpp | 12 + routing/road_graph_router.hpp | 2 + routing/route.cpp | 54 +++++ routing/route.hpp | 4 + .../routing_test_tools.cpp | 6 +- 10 files changed, 238 insertions(+), 120 deletions(-) diff --git a/geometry/polyline2d.hpp b/geometry/polyline2d.hpp index 681a4e811b..701f888192 100644 --- a/geometry/polyline2d.hpp +++ b/geometry/polyline2d.hpp @@ -67,6 +67,18 @@ public: void Clear() { m_points.clear(); } void Add(Point const & pt) { m_points.push_back(pt); } + + void Append(PolylineT const & poly) + { + m_points.insert(m_points.end(), poly.m_points.cbegin(), poly.m_points.cend()); + } + + void PopBack() + { + if (!m_points.empty()) + m_points.pop_back(); + } + void Swap(PolylineT & rhs) { m_points.swap(rhs.m_points); diff --git a/map/framework.cpp b/map/framework.cpp index 3276a0fa46..2ecddda15f 100644 --- a/map/framework.cpp +++ b/map/framework.cpp @@ -2398,7 +2398,8 @@ void Framework::SetRouterImpl(RouterType type) return m_model.GetIndex().GetMwmIdByCountryFile(CountryFile(countryFile)).IsAlive(); }; - router.reset(new OsrmRouter(&m_model.GetIndex(), countryFileGetter)); + router.reset(new OsrmRouter(&m_model.GetIndex(), countryFileGetter, + CreateCarAStarBidirectionalRouter(m_model.GetIndex(), countryFileGetter))); fetcher.reset(new OnlineAbsentCountriesFetcher(countryFileGetter, localFileChecker)); m_routingSession.SetRoutingSettings(routing::GetCarRoutingSettings()); } diff --git a/routing/base/followed_polyline.hpp b/routing/base/followed_polyline.hpp index ebf2a87d78..4ceaa5a4b2 100644 --- a/routing/base/followed_polyline.hpp +++ b/routing/base/followed_polyline.hpp @@ -20,6 +20,18 @@ public: void Swap(FollowedPolyline & rhs); + void Append(FollowedPolyline const & poly) + { + m_poly.Append(poly.m_poly); + Update(); + } + + void PopBack() + { + m_poly.PopBack(); + Update(); + } + bool IsValid() const { return (m_current.IsValid() && m_poly.GetSize() > 1); } m2::PolylineD const & GetPolyline() const { return m_poly; } diff --git a/routing/osrm_router.cpp b/routing/osrm_router.cpp index 3ce50d4db3..31c8b87eaf 100644 --- a/routing/osrm_router.cpp +++ b/routing/osrm_router.cpp @@ -5,6 +5,7 @@ #include "routing/osrm_helpers.hpp" #include "routing/osrm_path_segment_factory.hpp" #include "routing/osrm_router.hpp" +#include "routing/road_graph_router.hpp" #include "routing/turns_generator.hpp" #include "platform/country_file.hpp" @@ -15,8 +16,10 @@ #include "geometry/distance_on_sphere.hpp" #include "geometry/mercator.hpp" +#include "indexer/feature_altitude.hpp" #include "indexer/ftypes_matcher.hpp" #include "indexer/index.hpp" +#include "indexer/mwm_set.hpp" #include "indexer/scales.hpp" #include "coding/reader_wrapper.hpp" @@ -205,8 +208,9 @@ bool OsrmRouter::CheckRoutingAbility(m2::PointD const & startPoint, m2::PointD c manager.GetMappingByPoint(finalPoint)->IsValid(); } -OsrmRouter::OsrmRouter(Index * index, TCountryFileFn const & countryFileFn) - : m_pIndex(index), m_indexManager(countryFileFn, *index) +OsrmRouter::OsrmRouter(Index * index, TCountryFileFn const & countryFileFn, + unique_ptr roadGraphRouter) + : m_pIndex(index), m_indexManager(countryFileFn, *index), m_roadGraphRouter(move(roadGraphRouter)) { } @@ -222,14 +226,17 @@ void OsrmRouter::ClearState() m_indexManager.Clear(); } -bool OsrmRouter::FindRouteFromCases(TFeatureGraphNodeVec const & source, - TFeatureGraphNodeVec const & target, TDataFacade & facade, - RawRoutingResult & rawRoutingResult) +bool OsrmRouter::FindRouteFromCases(TFeatureGraphNodeVec const & source, TFeatureGraphNodeVec const & target, + 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 & sourceEdge : source) - if (FindSingleRoute(sourceEdge, targetEdge, facade, rawRoutingResult)) + if (FindSingleRouteDispatcher(sourceEdge, targetEdge, delegate, mapping, route)) return true; return false; } @@ -279,7 +286,7 @@ void FindGraphNodeOffsets(uint32_t const nodeId, m2::PointD const & point, } void CalculatePhantomNodeForCross(TRoutingMappingPtr & mapping, FeatureGraphNode & graphNode, - Index const * pIndex, bool forward) + Index const * pIndex, bool forward) { if (graphNode.segment.IsValid()) return; @@ -301,85 +308,22 @@ OsrmRouter::ResultCode OsrmRouter::MakeRouteFromCrossesPath(TCheckedPath const & RouterDelegate const & delegate, Route & route) { - Route::TTurns turnsDir; - Route::TTimes times; - Route::TStreets streets; - vector points; + Route emptyRoute(GetName()); + route.Swap(emptyRoute); + for (RoutePathCross cross : path) { ASSERT_EQUAL(cross.startNode.mwmId, cross.finalNode.mwmId, ()); - RawRoutingResult routingResult; TRoutingMappingPtr mwmMapping = m_indexManager.GetMappingById(cross.startNode.mwmId); 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 (!FindSingleRoute(cross.startNode, cross.finalNode, mwmMapping->m_dataFacade, routingResult)) + if (!FindSingleRouteDispatcher(cross.startNode, cross.finalNode, delegate, mwmMapping, route)) return RouteNotFound; - - if (!points.empty()) - { - // Remove road end point and turn instruction. - points.pop_back(); - turnsDir.pop_back(); - times.pop_back(); - // Streets might not point to the last point of the path. - } - - // Get annotated route. - Route::TTurns mwmTurnsDir; - Route::TTimes mwmTimes; - Route::TStreets mwmStreets; - vector mwmJunctions; - - OSRMRoutingResult resultGraph(*m_pIndex, *mwmMapping, routingResult); - if (MakeTurnAnnotation(resultGraph, delegate, mwmJunctions, mwmTurnsDir, mwmTimes, mwmStreets) != NoError) - { - LOG(LWARNING, ("Can't load road path data from disk for", mwmMapping->GetCountryName())); - return RouteNotFound; - } - - vector mwmPoints; - JunctionsToPoints(mwmJunctions, mwmPoints); - - // Connect annotated route. - auto const pSize = static_cast(points.size()); - for (auto turn : mwmTurnsDir) - { - if (turn.m_index == 0) - continue; - turn.m_index += pSize; - turnsDir.push_back(turn); - } - - if (!mwmStreets.empty() && !streets.empty() && mwmStreets.front().second == streets.back().second) - mwmStreets.erase(mwmStreets.begin()); - for (auto street : mwmStreets) - { - if (street.first == 0) - continue; - street.first += pSize; - streets.push_back(street); - } - - double const estimationTime = times.size() ? times.back().second : 0.0; - for (auto time : mwmTimes) - { - if (time.first == 0) - continue; - time.first += pSize; - time.second += estimationTime; - times.push_back(time); - } - - points.insert(points.end(), mwmPoints.begin(), mwmPoints.end()); } - route.SetGeometry(points.begin(), points.end()); - route.SetTurnInstructions(move(turnsDir)); - route.SetSectionTimes(move(times)); - route.SetStreetNames(move(streets)); return NoError; } @@ -456,8 +400,7 @@ OsrmRouter::ResultCode OsrmRouter::CalculateRoute(m2::PointD const & startPoint, delegate.OnProgress(kPointsFoundProgress); // 4. Find route. - RawRoutingResult routingResult; - double crossCost = 0; + double crossCostM = 0; TCheckedPath finalPath; // Manually load facade to avoid unmaping files we routing on. @@ -471,11 +414,10 @@ OsrmRouter::ResultCode OsrmRouter::CalculateRoute(m2::PointD const & startPoint, { indexPair.second->FreeCrossContext(); }); - ResultCode crossCode = CalculateCrossMwmPath(startTask, m_cachedTargets, m_indexManager, crossCost, + ResultCode crossCode = CalculateCrossMwmPath(startTask, m_cachedTargets, m_indexManager, crossCostM, delegate, finalPath); LOG(LINFO, ("Found cross path in", timer.ElapsedNano(), "ns.")); - if (!FindRouteFromCases(startTask, m_cachedTargets, startMapping->m_dataFacade, - routingResult)) + if (!FindRouteFromCases(startTask, m_cachedTargets, delegate, startMapping, route)) { if (crossCode == NoError) { @@ -488,9 +430,9 @@ OsrmRouter::ResultCode OsrmRouter::CalculateRoute(m2::PointD const & startPoint, } INTERRUPT_WHEN_CANCELLED(delegate); - if (crossCode == NoError && crossCost < routingResult.shortestPathLength) + if (crossCode == NoError && crossCostM < route.GetTotalDistanceMeters()) { - LOG(LINFO, ("Cross mwm path shorter. Cross cost:", crossCost, "single cost:", routingResult.shortestPathLength)); + LOG(LINFO, ("Cross mwm path shorter. Cross cost:", crossCostM, "single cost:", route.GetTotalDistanceMeters())); auto code = MakeRouteFromCrossesPath(finalPath, delegate, route); LOG(LINFO, ("Made final route in", timer.ElapsedNano(), "ns.")); timer.Reset(); @@ -500,34 +442,12 @@ OsrmRouter::ResultCode OsrmRouter::CalculateRoute(m2::PointD const & startPoint, INTERRUPT_WHEN_CANCELLED(delegate); delegate.OnProgress(kPathFoundProgress); - // 5. Restore route. - - Route::TTurns turnsDir; - Route::TTimes times; - Route::TStreets streets; - vector junctions; - - OSRMRoutingResult resultGraph(*m_pIndex, *startMapping, routingResult); - if (MakeTurnAnnotation(resultGraph, delegate, junctions, turnsDir, times, streets) != NoError) - { - LOG(LWARNING, ("Can't load road path data from disk!")); - return RouteNotFound; - } - - vector points; - JunctionsToPoints(junctions, points); - - route.SetGeometry(points.begin(), points.end()); - route.SetTurnInstructions(move(turnsDir)); - route.SetSectionTimes(move(times)); - route.SetStreetNames(move(streets)); - return NoError; } else //4.2 Multiple mwm case { LOG(LINFO, ("Multiple mwm routing case")); - ResultCode code = CalculateCrossMwmPath(startTask, m_cachedTargets, m_indexManager, crossCost, + ResultCode code = CalculateCrossMwmPath(startTask, m_cachedTargets, m_indexManager, crossCostM, delegate, finalPath); timer.Reset(); INTERRUPT_WHEN_CANCELLED(delegate); @@ -569,4 +489,83 @@ IRouter::ResultCode OsrmRouter::FindPhantomNodes(m2::PointD const & point, getter.MakeResult(res, maxCount); return NoError; } + +bool OsrmRouter::IsEdgeIndexExisting(Index::MwmId const & mwmId) +{ + MwmSet::MwmHandle const handle = m_pIndex->GetMwmHandleById(mwmId); + if (!handle.IsAlive()) + { + ASSERT(false, ("m_mwmHandle is not alive.")); + return false; + } + + MwmValue const * value = handle.GetValue(); + ASSERT(value, ()); + if (value->GetHeader().GetFormat() < version::Format::v8) + return false; + + // @TODO Remove this string before merge. + string const EDGE_INDEX_FILE_TAG = "edgeidx"; + if (value->m_cont.IsExist(EDGE_INDEX_FILE_TAG.c_str())) + return true; + return false; +} + +bool OsrmRouter::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_roadGraphRouter, ()); + + 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 had edge index section in mwmId. + if (IsEdgeIndexExisting(source.mwmId) && m_roadGraphRouter) + { + LOG(LINFO, ("A* route form", MercatorBounds::ToLatLon(source.segmentPoint), + "to", MercatorBounds::ToLatLon(target.segmentPoint))); + + if (m_roadGraphRouter->CalculateRoute(source.segmentPoint, m2::PointD(0, 0), target.segmentPoint, + delegate, mwmRoute) != IRouter::NoError) + { + return false; + } + + } + else + { + vector mwmRouteGeometry; + Route::TTurns mwmTurns; + Route::TTimes mwmTimes; + Route::TStreets mwmStreets; + + LOG(LINFO, ("OSRM route form", MercatorBounds::ToLatLon(source.segmentPoint), + "to", MercatorBounds::ToLatLon(target.segmentPoint))); + + RawRoutingResult routingResult; + if (!FindSingleRoute(source, target, mapping->m_dataFacade, routingResult)) + return false; + + OSRMRoutingResult resultGraph(*m_pIndex, *mapping, routingResult); + if (MakeTurnAnnotation(resultGraph, delegate, mwmRouteGeometry, mwmTurns, mwmTimes, mwmStreets) != NoError) + { + LOG(LWARNING, ("Can't load road path data from disk for", mapping->GetCountryName())); + return false; + } + + mwmRoute.SetTurnInstructions(move(mwmTurns)); + mwmRoute.SetSectionTimes(move(mwmTimes)); + mwmRoute.SetStreetNames(move(mwmStreets)); + + vector mwmPoints; + JunctionsToPoints(mwmRouteGeometry, mwmPoints); + mwmRoute.SetGeometry(mwmPoints.cbegin(), mwmPoints.cend()); + } + + route.AppendRoute(mwmRoute); + return true; +} } // namespace routing diff --git a/routing/osrm_router.hpp b/routing/osrm_router.hpp index 6c28e3b361..a4a96c559c 100644 --- a/routing/osrm_router.hpp +++ b/routing/osrm_router.hpp @@ -6,6 +6,9 @@ #include "routing/router.hpp" #include "routing/routing_mapping.hpp" +#include "std/unique_ptr.hpp" +#include "std/vector.hpp" + namespace feature { class TypesHolder; } class Index; @@ -16,6 +19,7 @@ class FeatureType; namespace routing { +class RoadGraphRouter; struct RoutePathCross; using TCheckedPath = vector; @@ -26,7 +30,8 @@ class OsrmRouter : public IRouter public: typedef vector GeomTurnCandidateT; - OsrmRouter(Index * index, TCountryFileFn const & countryFileFn); + OsrmRouter(Index * index, TCountryFileFn const & countryFileFn, + unique_ptr roadGraphRouter); virtual string GetName() const override; @@ -36,22 +41,11 @@ public: virtual void ClearState() override; - /*! Find single shortest path in a single MWM between 2 sets of edges - * \param source: vector of source edges to make path - * \param taget: 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. - */ - static bool FindRouteFromCases(TFeatureGraphNodeVec const & source, - TFeatureGraphNodeVec const & target, TDataFacade & facade, - RawRoutingResult & rawRoutingResult); - /*! Fast checking ability of route construction * @param startPoint starting road point * @param finalPoint final road point * @param countryFileFn function for getting filename from point - * @oaram index mwmSet index + * @param index mwmSet index * @returns true if we can start routing process with a given data. */ static bool CheckRoutingAbility(m2::PointD const & startPoint, m2::PointD const & finalPoint, @@ -84,11 +78,37 @@ private: ResultCode MakeRouteFromCrossesPath(TCheckedPath const & path, RouterDelegate const & delegate, Route & route); + bool IsEdgeIndexExisting(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 restore all route attributes. + * \param route The found route is added the the |route| if the method returns true. + * \return true if route is build and false otherwise. + */ + // @TODO. The behavior of the method should be changed. This method should check if osrm section + // available and if so use them. If not, RoadGraphRouter and A* should be used. + bool FindSingleRouteDispatcher(FeatureGraphNode const & source, FeatureGraphNode const & target, + RouterDelegate const & delegate, TRoutingMappingPtr & mapping, + Route & route); + + /*! Find single shortest path in a single MWM between 2 sets of edges + * \param source: vector of source edges to make path + * \param taget: 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); + Index const * m_pIndex; TFeatureGraphNodeVec m_cachedTargets; m2::PointD m_cachedTargetPoint; RoutingIndexManager m_indexManager; + + unique_ptr m_roadGraphRouter; }; } // namespace routing diff --git a/routing/road_graph_router.cpp b/routing/road_graph_router.cpp index 312b466a3c..6204c141b2 100644 --- a/routing/road_graph_router.cpp +++ b/routing/road_graph_router.cpp @@ -310,4 +310,16 @@ unique_ptr CreateBicycleAStarBidirectionalRouter(Index & index, TCountr move(vehicleModelFactory), move(algorithm), move(directionsEngine))); return router; } + +unique_ptr CreateCarAStarBidirectionalRouter(Index & index, TCountryFileFn const & countryFileFn) +{ + // @TODO It's necessary to use car classes instead of bicycle ones. + unique_ptr vehicleModelFactory = make_unique(); + unique_ptr algorithm = make_unique(); + unique_ptr directionsEngine = make_unique(index); + unique_ptr router = make_unique( + "astar-bidirectional-bicycle", index, countryFileFn, IRoadGraph::Mode::ObeyOnewayTag, + move(vehicleModelFactory), move(algorithm), move(directionsEngine)); + return router; +} } // namespace routing diff --git a/routing/road_graph_router.hpp b/routing/road_graph_router.hpp index 17701bdc7a..019a30a584 100644 --- a/routing/road_graph_router.hpp +++ b/routing/road_graph_router.hpp @@ -56,4 +56,6 @@ 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); } // namespace routing diff --git a/routing/route.cpp b/routing/route.cpp index 27ed38b71c..c7f20933f3 100644 --- a/routing/route.cpp +++ b/routing/route.cpp @@ -354,6 +354,60 @@ void Route::Update() m_currentTime = 0.0; } +void Route::AppendRoute(Route const & route) +{ + if (!route.IsValid()) + return; + + if (m_poly.GetPolyline().GetSize() != 0) + { + ASSERT(!m_turns.empty(), ()); + ASSERT(!m_times.empty(), ()); + + // Remove road end point and turn instruction. + m_poly.PopBack(); + m_turns.pop_back(); + m_times.pop_back(); + // Streets might not point to the last point of the path. + } + + size_t const polySize = m_poly.GetPolyline().GetSize(); + + // Appending turns. + for (turns::TurnItem t : route.m_turns) + { + if (t.m_index == 0) + continue; + t.m_index += polySize; + m_turns.push_back(move(t)); + } + + // Appending street names. + for (TStreetItem s : route.m_streets) + { + if (s.first == 0) + continue; + s.first += polySize; + m_streets.push_back(move(s)); + } + + // @TODO Implement altitude appending. + + // Appending times. + double const estimationTime = m_times.empty() ? 0.0 : m_times.back().second; + for (TTimeItem t : route.m_times) + { + if (t.first == 0) + continue; + t.first += polySize; + t.second += estimationTime; + m_times.push_back(move(t)); + } + + m_poly.Append(route.m_poly); + Update(); +} + string DebugPrint(Route const & r) { return DebugPrint(r.m_poly.GetPolyline()); diff --git a/routing/route.hpp b/routing/route.hpp index 634ea55e7a..093306b800 100644 --- a/routing/route.hpp +++ b/routing/route.hpp @@ -59,6 +59,10 @@ public: inline void SetSectionTimes(TTimes && v) { m_times = move(v); } inline void SetStreetNames(TStreets && v) { m_streets = move(v); } inline void SetAltitudes(feature::TAltitudes && v) { m_altitudes = move(v); } + + /// \brief Appends all |route| attributes except for altitude. + void AppendRoute(Route const & route); + uint32_t GetTotalTimeSec() const; uint32_t GetCurrentTimeToEndSec() const; diff --git a/routing/routing_integration_tests/routing_test_tools.cpp b/routing/routing_integration_tests/routing_test_tools.cpp index 9447911d3e..c292ef0330 100644 --- a/routing/routing_integration_tests/routing_test_tools.cpp +++ b/routing/routing_integration_tests/routing_test_tools.cpp @@ -77,10 +77,12 @@ namespace integration unique_ptr CreateOsrmRouter(Index & index, storage::CountryInfoGetter const & infoGetter) { - unique_ptr osrmRouter(new OsrmRouter(&index, [&infoGetter](m2::PointD const & pt) + auto const countryFileGetter = [&infoGetter](m2::PointD const & pt) { return infoGetter.GetRegionCountryId(pt); - })); + }; + unique_ptr osrmRouter(new OsrmRouter(&index, countryFileGetter, + CreateCarAStarBidirectionalRouter(index, countryFileGetter))); return osrmRouter; }