From 9d5562b78ed170dbc0ac52ad09ddcab7f13d9ceb Mon Sep 17 00:00:00 2001 From: Vladimir Byko-Ianko Date: Fri, 1 Jun 2018 17:26:04 +0300 Subject: [PATCH] Review fixes. --- map/extrapolation/extrapolator.cpp | 43 +++++++-------- map/extrapolation/extrapolator.hpp | 7 +++ .../extrapolation_benchmark.cpp | 55 +++++++++---------- platform/location.hpp | 2 +- routing/base/followed_polyline.hpp | 2 +- track_analyzing/track_analyzer/cmd_table.cpp | 2 - .../track_analyzer/track_analyzer.cpp | 2 +- 7 files changed, 56 insertions(+), 57 deletions(-) diff --git a/map/extrapolation/extrapolator.cpp b/map/extrapolation/extrapolator.cpp index 00be0f8e51..cd4e7178f2 100644 --- a/map/extrapolation/extrapolator.cpp +++ b/map/extrapolation/extrapolator.cpp @@ -11,10 +11,10 @@ namespace { -// If the difference of an value between two instances of GpsInfo is greater +// If the difference of values between two instances of GpsInfo is greater // than the appropriate constant below the extrapolator will be switched off for // these two instances of GpsInfo. -double constexpr kMaxExtrapolationSpeedMPS = 120.0; +double constexpr kMaxExtrapolationSpeedMPS = 85.0; double constexpr kMaxExtrapolationDistMeters = 120.0; double constexpr kMaxExtrapolationTimeSeconds = 2.1; @@ -53,7 +53,7 @@ location::GpsInfo LinearExtrapolation(location::GpsInfo const & gpsInfo1, location::GpsInfo result = gpsInfo2; LinearExtrapolator e(timeBetweenPointsMs, timeAfterPoint2Ms); - result.m_timestamp += timeAfterPoint2Ms; + result.m_timestamp += static_cast(timeAfterPoint2Ms) / 1000.0; result.m_longitude = e.Extrapolate(gpsInfo1.m_longitude, gpsInfo2.m_longitude); result.m_latitude = e.Extrapolate(gpsInfo1.m_latitude, gpsInfo2.m_latitude); result.m_horizontalAccuracy = e.Extrapolate(gpsInfo1.m_horizontalAccuracy, gpsInfo2.m_horizontalAccuracy); @@ -78,19 +78,22 @@ location::GpsInfo LinearExtrapolation(location::GpsInfo const & gpsInfo1, bool AreCoordsGoodForExtrapolation(location::GpsInfo const & beforeLastGpsInfo, location::GpsInfo const & lastGpsInfo) { + if (!beforeLastGpsInfo.IsValid() || !lastGpsInfo.IsValid()) + return false; + double const distM = ms::DistanceOnEarth(beforeLastGpsInfo.m_latitude, beforeLastGpsInfo.m_longitude, lastGpsInfo.m_latitude, lastGpsInfo.m_longitude); double const timeS = lastGpsInfo.m_timestamp - beforeLastGpsInfo.m_timestamp; - // Note. |timeS| may be less than zero. (beforeLastGpsInfo.m_timestamp >= lastGpsInfo.m_timestamp) - // It may happen in rare cases because GpsInfo::m_timestamp is not monotonic generally. + // Note. |timeS| may be less than zero. (beforeLastGpsInfo.m_timestampS >= lastGpsInfo.m_timestampS) + // It may happen in rare cases because GpsInfo::m_timestampS is not monotonic generally. // Please see comment in declaration of class GpsInfo for details. // @TODO(bykoianko) Switching off extrapolation based on acceleration should be implemented. // Switching off extrapolation based on speed, distance and time. - return distM / timeS <= kMaxExtrapolationSpeedMPS && distM <= kMaxExtrapolationDistMeters && - timeS <= kMaxExtrapolationTimeSeconds && timeS > 0; + return timeS > 0 && distM / timeS <= kMaxExtrapolationSpeedMPS && + distM <= kMaxExtrapolationDistMeters && timeS <= kMaxExtrapolationTimeSeconds; } // Extrapolator ------------------------------------------------------------------------------------ @@ -122,22 +125,21 @@ void Extrapolator::Enable(bool enabled) 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()) + if (extrapolationTimeMs < kMaxExtrapolationTimeMs && m_lastGpsInfo.IsValid()) { - gpsInfo = LinearExtrapolation(m_beforeLastGpsInfo, m_lastGpsInfo, extrapolationTimeMs); - break; + if (DoesExtrapolationWork()) + { + gpsInfo = LinearExtrapolation(m_beforeLastGpsInfo, m_lastGpsInfo, extrapolationTimeMs); + } + else if (m_lastGpsInfo.IsValid()) + { + gpsInfo = m_lastGpsInfo; + } } - - if (m_lastGpsInfo.IsValid()) - gpsInfo = m_lastGpsInfo; - } while (false); + } if (gpsInfo.IsValid()) { @@ -162,11 +164,8 @@ void Extrapolator::ExtrapolatedLocationUpdate() bool Extrapolator::DoesExtrapolationWork() const { - if (!m_isEnabled || m_extrapolationCounter == kExtrapolationCounterUndefined || - !m_lastGpsInfo.IsValid() || !m_beforeLastGpsInfo.IsValid()) - { + if (!m_isEnabled || m_extrapolationCounter == kExtrapolationCounterUndefined) return false; - } return AreCoordsGoodForExtrapolation(m_beforeLastGpsInfo, m_lastGpsInfo); } diff --git a/map/extrapolation/extrapolator.hpp b/map/extrapolation/extrapolator.hpp index b665b91440..d0a71d6ff0 100644 --- a/map/extrapolation/extrapolator.hpp +++ b/map/extrapolation/extrapolator.hpp @@ -25,7 +25,14 @@ class Extrapolator public: using ExtrapolatedLocationUpdateFn = std::function; + // |kMaxExtrapolationTimeMs| is time in milliseconds showing how long location will be + // extrapolated after last location gotten from GPS. static uint64_t constexpr kMaxExtrapolationTimeMs = 1000; + // |kExtrapolationPeriodMs| is time in milliseconds showing how often location will be + // extrapolated. So if the last location was gotten from GPS at time X the next location + // will be emulated by Extrapolator at X + kExtrapolationPeriodMs. + // Then X + 2 * kExtrapolationPeriodMs and so on till + // X + n * kExtrapolationPeriodMs <= kMaxExtrapolationTimeMs. static uint64_t constexpr kExtrapolationPeriodMs = 200; /// \param update is a function which is called with params according to extrapolated position. diff --git a/map/extrapolation_benchmark/extrapolation_benchmark.cpp b/map/extrapolation_benchmark/extrapolation_benchmark.cpp index 0db0829ffb..d9fd187e56 100644 --- a/map/extrapolation_benchmark/extrapolation_benchmark.cpp +++ b/map/extrapolation_benchmark/extrapolation_benchmark.cpp @@ -27,7 +27,7 @@ // This tool is written to estimate quality of location extrapolation. To launch the benchmark // you need tracks in csv file with the format described below. To generate the csv file // you need to go through following steps: -// * take logs form Trafim project production in gz files +// * take logs from "trafin" project production in gz files // * extract them (gzip -d) // * run track_analyzer tool with unmatched_tracks command to generate csv // (track_analyzer -cmd unmatched_tracks -in ./trafin_log.20180517-0000) @@ -48,31 +48,24 @@ namespace { struct GpsPoint { - GpsPoint(double timestamp, double lat, double lon) - : m_timestamp(timestamp) + GpsPoint(double timestampS, double lat, double lon) + : m_timestampS(timestampS) , m_lat(lat) , m_lon(lon) { } - double m_timestamp; + double m_timestampS; // @TODO(bykoianko) Using LatLog type instead of two double should be considered. double m_lat; double m_lon; }; -class MathematicalExpectation +class Expectation { public: void Add(double value) { - if (m_counter == 0) - { - m_averageValue = value; - ++m_counter; - return; - } - m_averageValue = (m_averageValue * m_counter + value) / (m_counter + 1); ++m_counter; } @@ -85,10 +78,10 @@ private: size_t m_counter = 0; }; -class MathematicalExpectationVec +class ExpectationVec { public: - explicit MathematicalExpectationVec(size_t size) { m_mes.resize(size); } + explicit ExpectationVec(size_t size) { m_mes.resize(size); } void Add(vector const & values) { @@ -97,10 +90,10 @@ public: m_mes[i].Add(values[i]); } - vector const & Get() const { return m_mes; } + vector const & Get() const { return m_mes; } private: - vector m_mes; + vector m_mes; }; using Track = vector; @@ -130,9 +123,9 @@ bool GetUint64(stringstream & lineStream, uint64_t & result) return strings::to_uint64(strResult, result); } -bool GetGpsPoint(stringstream & lineStream, uint64_t & timestamp, double & lat, double & lon) +bool GetGpsPoint(stringstream & lineStream, uint64_t & timestampS, double & lat, double & lon) { - if (!GetUint64(lineStream, timestamp)) + if (!GetUint64(lineStream, timestampS)) return false; if (!GetDouble(lineStream, lat)) return false; @@ -180,10 +173,10 @@ bool Parse(string const & pathToCsv, Tracks & tracks) void GpsPointToGpsInfo(GpsPoint const gpsPoint, GpsInfo & gpsInfo) { - gpsInfo.m_timestamp = gpsPoint.m_timestamp; + gpsInfo.m_source = TLocationSource::EAppleNative; + gpsInfo.m_timestamp = gpsPoint.m_timestampS; gpsInfo.m_latitude = gpsPoint.m_lat; gpsInfo.m_longitude = gpsPoint.m_lon; - } } // namespace @@ -192,7 +185,7 @@ void GpsPointToGpsInfo(GpsPoint const gpsPoint, GpsInfo & gpsInfo) int main(int argc, char * argv[]) { google::SetUsageMessage( - "Location extrapolation benchmark. Calculates mathematical expectation and dispersion for " + "Location extrapolation benchmark. Calculates expected value and variance for " "all extrapolation deviations from tracks passed in csv with with csv_path."); google::ParseCommandLineFlags(&argc, &argv, true); @@ -229,17 +222,17 @@ int main(int argc, char * argv[]) LOG(LINFO, ("General tracks statistics." "\n Number of tracks:", tracks.size(), "\n Number of track points:", trackPointNum, - "\n Points per track in average:", trackPointNum / tracks.size(), + "\n Average points per track:", trackPointNum / tracks.size(), "\n Average track length:", trackLengths / tracks.size(), "meters")); // For all points of each track in |tracks| some extrapolations will be calculated. // The number of extrapolations depends on |Extrapolator::kExtrapolationPeriodMs| // and |Extrapolator::kMaxExtrapolationTimeMs| and equal for all points. - // Then mathematical expectation and dispersion each extrapolation will be printed. + // Then expected value and variance each extrapolation will be printed. auto const extrapolationNumber = static_cast(Extrapolator::kMaxExtrapolationTimeMs / Extrapolator::kExtrapolationPeriodMs); - MathematicalExpectationVec mes(extrapolationNumber); - MathematicalExpectationVec squareMes(extrapolationNumber); + ExpectationVec mes(extrapolationNumber); + ExpectationVec squareMes(extrapolationNumber); // Number of extrapolations which projections are calculated successfully for. size_t projectionCounter = 0; for (auto const & t : tracks) @@ -277,8 +270,10 @@ int main(int argc, char * argv[]) m2::RectD const posRect = MercatorBounds::MetresToXY( extrapolated.m_longitude, extrapolated.m_latitude, 100.0 /* half square in meters */); - // Note. One is deducted from polyline size because in GetClosestProjectionInInterval() + // Note 1. One is deducted from polyline size because in GetClosestProjectionInInterval() // is used segment indices but not point indices. + // Note 2. Calculating projection of the center of |posRect| to polyline segments which + // are inside of |posRect|. auto const & iter = followedPoly.GetClosestProjectionInInterval( posRect, [&extrapolatedMerc](FollowedPolyline::Iter const & it) { @@ -318,13 +313,13 @@ int main(int argc, char * argv[]) " ", mes.Get()[0].GetCounter() * extrapolationNumber, "extrapolations is calculated.\n", " Projection is calculated for", projectionCounter, "extrapolations.")); - LOG(LINFO, ("Mathematical expectation for each extrapolation:")); + LOG(LINFO, ("Expected value for each extrapolation:")); for (size_t i = 0; i < extrapolationNumber; ++i) { - double const dispersion = squareMes.Get()[i].Get() - mes.Get()[i].Get() * mes.Get()[i].Get(); + double const variance = squareMes.Get()[i].Get() - mes.Get()[i].Get() * mes.Get()[i].Get(); LOG(LINFO, ("Extrapolation", i + 1, ",", Extrapolator::kExtrapolationPeriodMs * (i + 1), - "seconds after point two. Mathematical expectation =", mes.Get()[i].Get(), - "meters.", "Dispersion =", dispersion, ". Standard deviation =", sqrt(dispersion))); + "seconds after point two. Expected value =", mes.Get()[i].Get(), + "meters.", "Variance =", variance, ". Standard deviation =", sqrt(variance))); } return 0; diff --git a/platform/location.hpp b/platform/location.hpp index e108f51ccb..7e161bf673 100644 --- a/platform/location.hpp +++ b/platform/location.hpp @@ -43,7 +43,7 @@ namespace location { public: TLocationSource m_source = EUndefined; - /// \note |m_timestamp| is calculated based on platform methods which don't + /// @TODO(bykoianko) |m_timestamp| is calculated based on platform methods which don't /// guarantee that |m_timestamp| is monotonic. |m_monotonicTimeMs| should be added to /// class |GpsInfo|. This time should be calculated based on Location::getElapsedRealtimeNanos() /// method in case of Android. How to calculate such time in case of iOS should be diff --git a/routing/base/followed_polyline.hpp b/routing/base/followed_polyline.hpp index 77adfe10b0..ae4c0c87c3 100644 --- a/routing/base/followed_polyline.hpp +++ b/routing/base/followed_polyline.hpp @@ -78,7 +78,7 @@ public: Iter GetIterToIndex(size_t index) const; /// \brief Calculates projection of center of |posRect| to the polyline. - /// \param posRect Only projection inside the rect are considered. + /// \param posRect Only projection inside the rect is considered. /// \param distFn A method which is used to calculate the destination between points. /// \param startIdx Start segment index in |m_segProj|. /// \param endIdx The index after the last one in |m_segProj|. diff --git a/track_analyzing/track_analyzer/cmd_table.cpp b/track_analyzing/track_analyzer/cmd_table.cpp index 07a8a70bf5..4fa130b228 100644 --- a/track_analyzing/track_analyzer/cmd_table.cpp +++ b/track_analyzing/track_analyzer/cmd_table.cpp @@ -228,8 +228,6 @@ void CmdTagsTable(string const & filepath, string const & trackExtension, String for (size_t trackIdx = 0; trackIdx < kv.second.size(); ++trackIdx) { MatchedTrack const & track = kv.second[trackIdx]; - // Note. There's no need in tracks with length one point. CalcSpeedKMpH() used below - // requires |timeElapsed| is greater then zero. It's impossible if track length is one. if (track.size() <= 1) continue; diff --git a/track_analyzing/track_analyzer/track_analyzer.cpp b/track_analyzing/track_analyzer/track_analyzer.cpp index 92f70c3075..10033a177e 100644 --- a/track_analyzing/track_analyzer/track_analyzer.cpp +++ b/track_analyzing/track_analyzer/track_analyzer.cpp @@ -25,7 +25,7 @@ namespace DEFINE_string_ext(cmd, "", "command:\n" - "match - based on raw logs gathers points to tracks and matchs them to features\n" + "match - based on raw logs gathers points to tracks and matches them to features\n" "unmatched_tracks - based on raw logs gathers points to tracks " "and save tracks to csv. Track points save as lat, log, timestamp in seconds\n" "tracks - prints track statistics\n"