diff --git a/map/extrapolation/extrapolator.cpp b/map/extrapolation/extrapolator.cpp index 828325ddbb..ac66fc424e 100644 --- a/map/extrapolation/extrapolator.cpp +++ b/map/extrapolation/extrapolator.cpp @@ -70,52 +70,17 @@ location::GpsInfo LinearExtrapolation(location::GpsInfo const & gpsInfo1, return result; } -// Extrapolator::Routine --------------------------------------------------------------------------- -Extrapolator::Routine::Routine(ExtrapolatedLocationUpdate const & update) +// Extrapolator ------------------------------------------------------------------------------------ +Extrapolator::Extrapolator(ExtrapolatedLocationUpdateFn const & update) : m_extrapolatedLocationUpdate(update) { -} - -void Extrapolator::Routine::Do() -{ - while (!IsCancelled()) + GetPlatform().RunTask(Platform::Thread::Background, [this] { - { - GetPlatform().RunTask(Platform::Thread::Gui, [this]() { - lock_guard guard(m_mutex); - uint64_t const extrapolationTimeMs = kExtrapolationPeriodMs * m_extrapolationCounter; - if (extrapolationTimeMs >= kMaxExtrapolationTimeMs) - return; - - if (DoesExtrapolationWork(extrapolationTimeMs)) - { - location::GpsInfo gpsInfo = - LinearExtrapolation(m_beforeLastGpsInfo, m_lastGpsInfo, extrapolationTimeMs); - m_extrapolatedLocationUpdate(gpsInfo); - } - else - { - if (m_lastGpsInfo.m_source != location::EUndefined) - { - location::GpsInfo gpsInfo = m_lastGpsInfo; - m_extrapolatedLocationUpdate(gpsInfo); - } - } - }); - - lock_guard guard(m_mutex); - if (m_extrapolationCounter != m_extrapolationCounterUndefined) - ++m_extrapolationCounter; - } - // @TODO(bykoinako) Method m_extrapolatedLocationUpdate() is run on gui thread every - // |kExtrapolationPeriodMs| milliseconds. But after changing GPS position - // (that means after a call of method Routine::SetGpsInfo()) - // m_extrapolatedLocationUpdate() should be run immediately on gui thread. - this_thread::sleep_for(std::chrono::milliseconds(kExtrapolationPeriodMs)); - } + ExtrapolatedLocationUpdate(); + }); } -void Extrapolator::Routine::SetGpsInfo(location::GpsInfo const & gpsInfo) +void Extrapolator::OnLocationUpdate(location::GpsInfo const & gpsInfo) { lock_guard guard(m_mutex); m_beforeLastGpsInfo = m_lastGpsInfo; @@ -123,15 +88,55 @@ void Extrapolator::Routine::SetGpsInfo(location::GpsInfo const & gpsInfo) m_extrapolationCounter = 0; } -bool Extrapolator::Routine::DoesExtrapolationWork(uint64_t extrapolationTimeMs) const +void Extrapolator::ExtrapolatedLocationUpdate() +{ + location::GpsInfo gpsInfo; + do + { + lock_guard guard(m_mutex); + uint64_t const extrapolationTimeMs = kExtrapolationPeriodMs * m_extrapolationCounter; + if (extrapolationTimeMs >= kMaxExtrapolationTimeMs || !m_lastGpsInfo.IsValid()) + break; + + if (DoesExtrapolationWork(extrapolationTimeMs)) + { + gpsInfo = LinearExtrapolation(m_beforeLastGpsInfo, m_lastGpsInfo, extrapolationTimeMs); + break; + } + + if (m_lastGpsInfo.IsValid()) + gpsInfo = m_lastGpsInfo; + } while (false); + + if (gpsInfo.IsValid()) + { + GetPlatform().RunTask(Platform::Thread::Gui, [this, gpsInfo]() { + m_extrapolatedLocationUpdate(gpsInfo); + }); + } + + { + lock_guard guard(m_mutex); + if (m_extrapolationCounter != m_extrapolationCounterUndefined) + ++m_extrapolationCounter; + } + + // Call ExtrapolatedLocationUpdate() every |kExtrapolationPeriodMs| milliseconds. + auto constexpr kSExtrapolationPeriod = std::chrono::milliseconds(kExtrapolationPeriodMs); + GetPlatform().RunDelayedTask(Platform::Thread::Background, kSExtrapolationPeriod, [this] + { + ExtrapolatedLocationUpdate(); + }); +} + +bool Extrapolator::DoesExtrapolationWork(uint64_t extrapolationTimeMs) const { // Note. It's possible that m_beforeLastGpsInfo.m_timestamp >= m_lastGpsInfo.m_timestamp. // It may happen in rare cases because GpsInfo::m_timestamp is not monotonic generally. // Please see comment in declaration of class GpsInfo for details. if (m_extrapolationCounter == m_extrapolationCounterUndefined || - m_lastGpsInfo.m_source == location::EUndefined || - m_beforeLastGpsInfo.m_source == location::EUndefined || + !m_lastGpsInfo.IsValid() || !m_beforeLastGpsInfo.IsValid() || m_beforeLastGpsInfo.m_timestamp >= m_lastGpsInfo.m_timestamp) { return false; @@ -146,18 +151,4 @@ bool Extrapolator::Routine::DoesExtrapolationWork(uint64_t extrapolationTimeMs) return distM / timeS < kMaxExtrapolationSpeedMPS; // @TODO(bykoianko) Switching off extrapolation based on acceleration should be implemented. } - -// Extrapolator ------------------------------------------------------------------------------------ -Extrapolator::Extrapolator(ExtrapolatedLocationUpdate const & update) - : m_extrapolatedLocationThread() -{ - m_extrapolatedLocationThread.Create(make_unique(update)); -} - -void Extrapolator::OnLocationUpdate(location::GpsInfo const & info) -{ - auto * routine = m_extrapolatedLocationThread.GetRoutineAs(); - CHECK(routine, ()); - routine->SetGpsInfo(info); -} } // namespace extrapolation diff --git a/map/extrapolation/extrapolator.hpp b/map/extrapolation/extrapolator.hpp index 0ed1501e84..e8c3b7fd30 100644 --- a/map/extrapolation/extrapolator.hpp +++ b/map/extrapolation/extrapolator.hpp @@ -2,13 +2,10 @@ #include "platform/location.hpp" -#include "base/thread.hpp" - #include #include #include #include -#include namespace extrapolation { @@ -21,41 +18,26 @@ location::GpsInfo LinearExtrapolation(location::GpsInfo const & gpsInfo1, class Extrapolator { public: - using ExtrapolatedLocationUpdate = std::function; + using ExtrapolatedLocationUpdateFn = std::function; static uint64_t constexpr m_extrapolationCounterUndefined = std::numeric_limits::max(); /// \param update is a function which is called with params according to extrapolated position. /// |update| will be called on gui thread. - explicit Extrapolator(ExtrapolatedLocationUpdate const & update); - void OnLocationUpdate(location::GpsInfo const & info); + explicit Extrapolator(ExtrapolatedLocationUpdateFn const & update); + void OnLocationUpdate(location::GpsInfo const & gpsInfo); // @TODO(bykoianko) Gyroscope information should be taken into account as well for calculation // extrapolated position. - void Cancel() { m_extrapolatedLocationThread.Cancel(); } + void ExtrapolatedLocationUpdate(); private: - class Routine : public threads::IRoutine - { - public: - explicit Routine(ExtrapolatedLocationUpdate const & update); + bool DoesExtrapolationWork(uint64_t extrapolationTimeMs) const; - // threads::IRoutine overrides: - void Do() override; - - void SetGpsInfo(location::GpsInfo const & gpsInfo); - - private: - bool DoesExtrapolationWork(uint64_t extrapolationTimeMs) const; - - ExtrapolatedLocationUpdate m_extrapolatedLocationUpdate; - - std::mutex m_mutex; - location::GpsInfo m_lastGpsInfo; - location::GpsInfo m_beforeLastGpsInfo; - uint64_t m_extrapolationCounter = m_extrapolationCounterUndefined; - }; - - threads::Thread m_extrapolatedLocationThread; + std::mutex m_mutex; + ExtrapolatedLocationUpdateFn m_extrapolatedLocationUpdate; + location::GpsInfo m_lastGpsInfo; + location::GpsInfo m_beforeLastGpsInfo; + uint64_t m_extrapolationCounter = m_extrapolationCounterUndefined; }; } // namespace extrapolation diff --git a/map/framework.cpp b/map/framework.cpp index 7c4309469f..4e8886b505 100644 --- a/map/framework.cpp +++ b/map/framework.cpp @@ -367,7 +367,7 @@ Framework::Framework(FrameworkParams const & params) }, [this]() -> StringsBundle const & { return m_stringsBundle; }), static_cast(*this)) - , m_extrapolator([this](location::GpsInfo & gpsInfo) { + , m_extrapolator([this](location::GpsInfo const & gpsInfo) { this->GetRoutingManager().OnLocationUpdate(gpsInfo); }) , m_trafficManager(bind(&Framework::GetMwmsByRect, this, _1, false /* rough */), @@ -497,7 +497,6 @@ Framework::~Framework() GetBookmarkManager().Teardown(); m_trafficManager.Teardown(); - m_extrapolator.Cancel(); DestroyDrapeEngine(); m_model.SetOnMapDeregisteredCallback(nullptr); diff --git a/map/routing_manager.cpp b/map/routing_manager.cpp index 2dcbd2a985..1a57ca8932 100644 --- a/map/routing_manager.cpp +++ b/map/routing_manager.cpp @@ -918,17 +918,18 @@ void RoutingManager::MatchLocationToRoute(location::GpsInfo & location, m_routingSession.MatchLocationToRoute(location, routeMatchingInfo); } -void RoutingManager::OnLocationUpdate(location::GpsInfo & info) +void RoutingManager::OnLocationUpdate(location::GpsInfo const & info) { + location::GpsInfo gpsInfo(info); if (!m_drapeEngine) - m_gpsInfoCache = my::make_unique(info); + m_gpsInfoCache = my::make_unique(gpsInfo); - auto routeMatchingInfo = GetRouteMatchingInfo(info); - m_drapeEngine.SafeCall(&df::DrapeEngine::SetGpsInfo, info, + auto routeMatchingInfo = GetRouteMatchingInfo(gpsInfo); + m_drapeEngine.SafeCall(&df::DrapeEngine::SetGpsInfo, gpsInfo, m_routingSession.IsNavigable(), routeMatchingInfo); if (IsTrackingReporterEnabled()) - m_trackingReporter.AddLocation(info, m_routingSession.MatchTraffic(routeMatchingInfo)); + m_trackingReporter.AddLocation(gpsInfo, m_routingSession.MatchTraffic(routeMatchingInfo)); } location::RouteMatchingInfo RoutingManager::GetRouteMatchingInfo(location::GpsInfo & info) diff --git a/map/routing_manager.hpp b/map/routing_manager.hpp index 73cdf78996..4da46cc9e6 100644 --- a/map/routing_manager.hpp +++ b/map/routing_manager.hpp @@ -209,7 +209,7 @@ public: void OnBuildRouteReady(routing::Route const & route, routing::RouterResultCode code); void OnRebuildRouteReady(routing::Route const & route, routing::RouterResultCode code); void OnRoutePointPassed(RouteMarkType type, size_t intermediateIndex); - void OnLocationUpdate(location::GpsInfo & info); + void OnLocationUpdate(location::GpsInfo const & info); void SetAllowSendingPoints(bool isAllowed) { m_trackingReporter.SetAllowSendingPoints(isAllowed);