diff --git a/routing/edge_estimator.cpp b/routing/edge_estimator.cpp index cdca30ae3e..ce20bdd4a6 100644 --- a/routing/edge_estimator.cpp +++ b/routing/edge_estimator.cpp @@ -4,10 +4,21 @@ #include "std/algorithm.hpp" -namespace routing -{ using namespace traffic; +namespace +{ +double CalcTrafficFactor(SpeedGroup speedGroup) +{ + double const percentage = + 0.01 * static_cast(kSpeedGroupThresholdPercentage[static_cast(speedGroup)]); + CHECK_GREATER(percentage, 0.0, ("speedGroup:", speedGroup)); + return 1.0 / percentage; +} +} // namespace + +namespace routing +{ double constexpr kKMPH2MPS = 1000.0 / (60 * 60); inline double TimeBetweenSec(m2::PointD const & from, m2::PointD const & to, double speedMPS) @@ -25,10 +36,10 @@ public: // EdgeEstimator overrides: void Start(MwmSet::MwmId const & mwmId) override; + void Finish() override; double CalcEdgesWeight(uint32_t featureId, RoadGeometry const & road, uint32_t pointFrom, uint32_t pointTo) const override; double CalcHeuristic(m2::PointD const & from, m2::PointD const & to) const override; - void Finish() override; private: TrafficInfoGetter const & m_trafficGetter; @@ -47,6 +58,11 @@ void CarEdgeEstimator::Start(MwmSet::MwmId const & mwmId) m_trafficInfo = m_trafficGetter.GetTrafficInfo(mwmId); } +void CarEdgeEstimator::Finish() +{ + m_trafficInfo.reset(); +} + double CarEdgeEstimator::CalcEdgesWeight(uint32_t featureId, RoadGeometry const & road, uint32_t pointFrom, uint32_t pointTo) const { @@ -66,9 +82,7 @@ double CarEdgeEstimator::CalcEdgesWeight(uint32_t featureId, RoadGeometry const SpeedGroup const speedGroup = m_trafficInfo->GetSpeedGroup(TrafficInfo::RoadSegmentId(featureId, i, dir)); CHECK_LESS(speedGroup, SpeedGroup::Count, ()); - double const percentage = - 0.01 * static_cast(kSpeedGroupThresholdPercentage[static_cast(speedGroup)]); - factor = 1.0 / percentage; + factor = CalcTrafficFactor(speedGroup); } result += factor * TimeBetweenSec(road.GetPoint(i), road.GetPoint(i + 1), speedMPS); } @@ -80,11 +94,6 @@ double CarEdgeEstimator::CalcHeuristic(m2::PointD const & from, m2::PointD const { return TimeBetweenSec(from, to, m_maxSpeedMPS); } - -void CarEdgeEstimator::Finish() -{ - m_trafficInfo.reset(); -} } // namespace namespace routing diff --git a/routing/edge_estimator.hpp b/routing/edge_estimator.hpp index f1887502ef..74315ed622 100644 --- a/routing/edge_estimator.hpp +++ b/routing/edge_estimator.hpp @@ -20,17 +20,27 @@ public: virtual ~EdgeEstimator() = default; virtual void Start(MwmSet::MwmId const & mwmId) = 0; - + virtual void Finish() = 0; virtual double CalcEdgesWeight(uint32_t featureId, RoadGeometry const & road, uint32_t pointFrom, uint32_t pointTo) const = 0; virtual double CalcHeuristic(m2::PointD const & from, m2::PointD const & to) const = 0; - virtual void Finish() = 0; static shared_ptr CreateForCar(IVehicleModel const & vehicleModel, traffic::TrafficInfoGetter const & getter); +}; -protected: - shared_ptr m_trafficInfo; +class EstimatorGuard final +{ +public: + EstimatorGuard(MwmSet::MwmId const & mwmId, EdgeEstimator & estimator) : m_estimator(estimator) + { + m_estimator.Start(mwmId); + } + + ~EstimatorGuard() { m_estimator.Finish(); } + +private: + EdgeEstimator & m_estimator; }; } // namespace routing diff --git a/routing/routing_session.cpp b/routing/routing_session.cpp index c1afbf63ef..ea6b6baaf3 100644 --- a/routing/routing_session.cpp +++ b/routing/routing_session.cpp @@ -49,6 +49,25 @@ uint32_t constexpr kMinimumETASec = 60; namespace routing { +void TrafficCache::Set(TrafficInfo && info) +{ + MwmSet::MwmId const mwmId = info.GetMwmId(); + m_trafficInfo[mwmId] = make_shared(move(info)); +} + +void TrafficCache::Remove(MwmSet::MwmId const & mwmId) { m_trafficInfo.erase(mwmId); } + +shared_ptr TrafficCache::Get(MwmSet::MwmId const & mwmId) const +{ + auto it = m_trafficInfo.find(mwmId); + + if (it == m_trafficInfo.cend()) + return shared_ptr(); + return it->second; +} + +void TrafficCache::Clear() { m_trafficInfo.clear(); } + RoutingSession::RoutingSession() : m_router(nullptr) , m_route(make_shared(string())) @@ -585,32 +604,28 @@ void RoutingSession::OnTrafficEnabled(bool enable) threads::MutexGuard guard(m_routingSessionMutex); UNUSED_VALUE(guard); if (!enable) - m_trafficInfo.clear(); + m_trafficCache.Clear(); } void RoutingSession::OnTrafficInfoAdded(TrafficInfo && info) { threads::MutexGuard guard(m_routingSessionMutex); UNUSED_VALUE(guard); - m_trafficInfo.insert(make_pair(info.GetMwmId(), make_shared(move(info)))); + m_trafficCache.Set(move(info)); } void RoutingSession::OnTrafficInfoRemoved(MwmSet::MwmId const & mwmId) { threads::MutexGuard guard(m_routingSessionMutex); UNUSED_VALUE(guard); - m_trafficInfo.erase(mwmId); + m_trafficCache.Remove(mwmId); } shared_ptr RoutingSession::GetTrafficInfo(MwmSet::MwmId const & mwmId) const { threads::MutexGuard guard(m_routingSessionMutex); UNUSED_VALUE(guard); - auto it = m_trafficInfo.find(mwmId); - - if (it == m_trafficInfo.cend()) - return shared_ptr(); - return it->second; + return m_trafficCache.Get(mwmId); } string DebugPrint(RoutingSession::State state) diff --git a/routing/routing_session.hpp b/routing/routing_session.hpp index 6f581a2538..c45572b28c 100644 --- a/routing/routing_session.hpp +++ b/routing/routing_session.hpp @@ -38,6 +38,18 @@ struct SpeedCameraRestriction SpeedCameraRestriction() : m_index(0), m_maxSpeedKmH(numeric_limits::max()) {} }; +class TrafficCache +{ +public: + void Set(traffic::TrafficInfo && info); + void Remove(MwmSet::MwmId const & mwmId); + shared_ptr Get(MwmSet::MwmId const & mwmId) const; + void Clear(); + +private: + map> m_trafficInfo; +}; + class RoutingSession : public traffic::TrafficObserver, public traffic::TrafficInfoGetter { friend void UnitTest_TestFollowRoutePercentTest(); @@ -191,7 +203,7 @@ private: double GetCompletionPercent() const; private: - map> m_trafficInfo; + TrafficCache m_trafficCache; unique_ptr m_router; shared_ptr m_route; atomic m_state; diff --git a/routing/routing_tests/applying_traffic_test.cpp b/routing/routing_tests/applying_traffic_test.cpp index 53cea7cf4a..f381994ea3 100644 --- a/routing/routing_tests/applying_traffic_test.cpp +++ b/routing/routing_tests/applying_traffic_test.cpp @@ -4,6 +4,7 @@ #include "routing/geometry.hpp" #include "routing/index_graph.hpp" #include "routing/index_graph_starter.hpp" +#include "routing/routing_session.hpp" #include "routing/routing_tests/index_graph_tools.hpp" @@ -85,22 +86,24 @@ unique_ptr BuildXXGraph(shared_ptr estimator) return graph; } -class TrafficInfoGetterTest : public TrafficInfoGetter +class TrafficInfoGetterForTesting : public TrafficInfoGetter { public: - TrafficInfoGetterTest(shared_ptr trafficInfo = shared_ptr()) - : m_trafficInfo(trafficInfo) {} - - // TrafficInfoGetter overrides: - shared_ptr GetTrafficInfo(MwmSet::MwmId const &) const override + TrafficInfoGetterForTesting(TrafficInfo && trafficInfo) { - return m_trafficInfo; + m_trafficCache.Set(move(trafficInfo)); } - void UpdateTrafficInfo(shared_ptr trafficInfo) { m_trafficInfo = trafficInfo; } + // TrafficInfoGetter overrides: + shared_ptr GetTrafficInfo(MwmSet::MwmId const & mwmId) const override + { + return m_trafficCache.Get(mwmId); + } + + void UpdateTrafficInfo(TrafficInfo && trafficInfo) { m_trafficCache.Set(move(trafficInfo)); } private: - shared_ptr m_trafficInfo; + TrafficCache m_trafficCache; }; class ApplyingTrafficTest @@ -110,7 +113,7 @@ public: void SetEstimator(TrafficInfo::Coloring && coloring) { - m_trafficGetter = make_unique(make_shared(move(coloring))); + m_trafficGetter = make_unique(move(coloring)); m_estimator = EdgeEstimator::CreateForCar(*make_shared()->GetVehicleModel(), *m_trafficGetter); } @@ -119,11 +122,11 @@ public: void UpdateTrafficInfo(TrafficInfo::Coloring && coloring) { - m_trafficGetter->UpdateTrafficInfo(make_shared(move(coloring))); + m_trafficGetter->UpdateTrafficInfo(TrafficInfo(move(coloring))); } private: - unique_ptr m_trafficGetter; + unique_ptr m_trafficGetter; shared_ptr m_estimator; }; @@ -194,7 +197,6 @@ UNIT_CLASS_TEST(ApplyingTrafficTest, XXGraph_ChangingTraffic) { // No trafic at all. SetEstimator({}); - unique_ptr graph = BuildXXGraph(GetEstimator()); IndexGraphStarter starter(*graph, RoadPoint(1, 0) /* start */, RoadPoint(6, 1) /* finish */); vector const noTrafficGeom = {{2 /* x */, 0 /* y */}, {1, 1}, {2, 2}, {3, 3}}; @@ -211,5 +213,22 @@ UNIT_CLASS_TEST(ApplyingTrafficTest, XXGraph_ChangingTraffic) vector const heavyF3Geom = {{2 /* x */, 0 /* y */}, {3, 0}, {3, 1}, {2, 2}, {3, 3}}; TestRouteGeometry(starter, AStarAlgorithm::Result::OK, heavyF3Geom); GetEstimator()->Finish(); + + // Overloading traffic jam on F3. Middle traffic (SpeedGroup::G3) on F1, F3, F4, F7 and F8. + TrafficInfo::Coloring coloringMiddleF1F3F4F7F8 = { + {{1 /* feature id */, 0 /* segment id */, TrafficInfo::RoadSegmentId::kForwardDirection}, + SpeedGroup::G3}, + {{3 /* feature id */, 0 /* segment id */, TrafficInfo::RoadSegmentId::kForwardDirection}, + SpeedGroup::G3}, + {{4 /* feature id */, 0 /* segment id */, TrafficInfo::RoadSegmentId::kForwardDirection}, + SpeedGroup::G3}, + {{7 /* feature id */, 0 /* segment id */, TrafficInfo::RoadSegmentId::kForwardDirection}, + SpeedGroup::G3}, + {{8 /* feature id */, 0 /* segment id */, TrafficInfo::RoadSegmentId::kForwardDirection}, + SpeedGroup::G3}}; + UpdateTrafficInfo(move(coloringMiddleF1F3F4F7F8)); + GetEstimator()->Start(MwmSet::MwmId()); + TestRouteGeometry(starter, AStarAlgorithm::Result::OK, noTrafficGeom); + GetEstimator()->Finish(); } } // namespace diff --git a/routing/single_mwm_router.cpp b/routing/single_mwm_router.cpp index c43c94c5a9..bb8fa932b7 100644 --- a/routing/single_mwm_router.cpp +++ b/routing/single_mwm_router.cpp @@ -44,20 +44,6 @@ vector ConvertToJunctions(IndexGraphStarter & starter, vector