diff --git a/routing/features_road_graph.cpp b/routing/features_road_graph.cpp index da5f9ecf10..8f5292682d 100644 --- a/routing/features_road_graph.cpp +++ b/routing/features_road_graph.cpp @@ -57,6 +57,11 @@ SpeedKMpH FeaturesRoadGraph::CrossCountryVehicleModel::GetSpeed( return GetVehicleModel(f.GetID())->GetSpeed(f, speedParams); } +HighwayType FeaturesRoadGraph::CrossCountryVehicleModel::GetHighwayType(FeatureType & f) const +{ + return GetVehicleModel(f.GetID())->GetHighwayType(f); +} + double FeaturesRoadGraph::CrossCountryVehicleModel::GetOffroadSpeed() const { return m_offroadSpeedKMpH; diff --git a/routing/features_road_graph.hpp b/routing/features_road_graph.hpp index 84297e0919..c5686552b3 100644 --- a/routing/features_road_graph.hpp +++ b/routing/features_road_graph.hpp @@ -34,6 +34,7 @@ private: // VehicleModelInterface overrides: SpeedKMpH GetSpeed(FeatureType & f, SpeedParams const & speedParams) const override; + HighwayType GetHighwayType(FeatureType & f) const override; double GetMaxWeightSpeed() const override { return m_maxSpeed; }; double GetOffroadSpeed() const override; bool IsOneWay(FeatureType & f) const override; diff --git a/routing/geometry.cpp b/routing/geometry.cpp index c8a13042f3..f7f10d693d 100644 --- a/routing/geometry.cpp +++ b/routing/geometry.cpp @@ -159,6 +159,7 @@ void RoadGeometry::Load(VehicleModelInterface const & vehicleModel, FeatureType m_isOneWay = vehicleModel.IsOneWay(feature); m_forwardSpeed = vehicleModel.GetSpeed(feature, {true /* forward */, inCity, maxspeed}); m_backwardSpeed = vehicleModel.GetSpeed(feature, {false /* forward */, inCity, maxspeed}); + m_highwayType = vehicleModel.GetHighwayType(feature); m_isPassThroughAllowed = vehicleModel.IsPassThroughAllowed(feature); feature::TypesHolder types(feature); diff --git a/routing/geometry.hpp b/routing/geometry.hpp index a7d88a0a13..d5e819303c 100644 --- a/routing/geometry.hpp +++ b/routing/geometry.hpp @@ -20,6 +20,8 @@ #include #include +#include + class DataSource; namespace routing @@ -37,6 +39,7 @@ public: bool IsOneWay() const { return m_isOneWay; } SpeedKMpH const & GetSpeed(bool forward) const; + HighwayType GetHighwayType() const { return *m_highwayType; } bool IsPassThroughAllowed() const { return m_isPassThroughAllowed; } Junction const & GetJunction(uint32_t junctionId) const @@ -77,6 +80,7 @@ private: buffer_vector m_junctions; SpeedKMpH m_forwardSpeed; SpeedKMpH m_backwardSpeed; + boost::optional m_highwayType; bool m_isOneWay = false; bool m_valid = false; bool m_isPassThroughAllowed = false; diff --git a/routing/road_point.hpp b/routing/road_point.hpp index dab000bec5..8c450eecb0 100644 --- a/routing/road_point.hpp +++ b/routing/road_point.hpp @@ -41,7 +41,7 @@ public: struct Hash { - uint64_t operator() (RoadPoint const & roadPoint) const + uint64_t operator()(RoadPoint const & roadPoint) const { uint32_t featureId = roadPoint.m_featureId; uint32_t pointId = roadPoint.m_pointId; diff --git a/routing/transit_world_graph.cpp b/routing/transit_world_graph.cpp index e45fea6ed5..1d1a8d34f7 100644 --- a/routing/transit_world_graph.cpp +++ b/routing/transit_world_graph.cpp @@ -156,7 +156,7 @@ RouteWeight TransitWorldGraph::CalcOffroadWeight(m2::PointD const & from, return RouteWeight(m_estimator->CalcOffroadWeight(from, to)); } -double TransitWorldGraph::CalcSegmentETA(routing::Segment const & segment) +double TransitWorldGraph::CalcSegmentETA(Segment const & segment) { // TODO: Separate weight and ETA of transit segments. if (TransitGraph::IsTransitSegment(segment)) diff --git a/routing_common/vehicle_model.cpp b/routing_common/vehicle_model.cpp index 462307d5ba..318455ea2f 100644 --- a/routing_common/vehicle_model.cpp +++ b/routing_common/vehicle_model.cpp @@ -114,20 +114,37 @@ SpeedKMpH VehicleModel::GetSpeed(FeatureType & f, SpeedParams const & speedParam return GetTypeSpeed(types, speedParams); } +HighwayType VehicleModel::GetHighwayType(FeatureType & f) const +{ + feature::TypesHolder const types(f); + boost::optional ret; + for (auto const t : types) + { + ret = GetHighwayType(t); + if (ret) + break; + + if (FindAdditionalRoadType(t) != m_addRoadTypes.end()) + return static_cast(classif().GetIndexForType(t)); + } + + return *ret; +} + double VehicleModel::GetMaxWeightSpeed() const { return max(m_maxModelSpeed.m_inCity.m_weight, m_maxModelSpeed.m_outCity.m_weight); } -void VehicleModel::GetHighwayType(uint32_t type, boost::optional & hwType) const +boost::optional VehicleModel::GetHighwayType(uint32_t type) const { - if (hwType) - return; - + boost::optional hwType; type = ftypes::BaseChecker::PrepareToMatch(type, 2); auto const it = m_roadTypes.find(type); if (it != m_roadTypes.cend()) hwType = it->second.GetHighwayType(); + + return hwType; } void VehicleModel::GetSurfaceFactor(uint32_t type, SpeedFactor & factor) const @@ -154,7 +171,7 @@ void VehicleModel::GetAdditionalRoadSpeed(uint32_t type, bool isCityRoad, boost: speed = speed ? Pick(*speed, res) : res; } -SpeedKMpH VehicleModel::GetSpeedOnFeatureWithMaxspeed(boost::optional const & type, +SpeedKMpH VehicleModel::GetSpeedOnFeatureWithMaxspeed(HighwayType const & type, SpeedParams const & speedParams) const { CHECK(speedParams.m_maxspeed.IsValid(), ()); @@ -162,17 +179,11 @@ SpeedKMpH VehicleModel::GetSpeedOnFeatureWithMaxspeed(boost::optional(speedParams.m_maxspeed.GetSpeedKmPH(speedParams.m_forward)); SpeedKMpH speed = Pick(SpeedKMpH(maxspeedKmPH, maxspeedKmPH), maxModelSpeed); - - // Note. If a highway type is not found and |maxspeed| is set, |maxspeed| in kms per hour - // should be returned. That means |maxspeedFactor| should be 1.0. - if (!type) - return speed; - // We assume that all link roads are equal to its parents and drop "_link" suffix // while searching for the particular factor. auto const maxspeedKey = GetMaxspeedKey(maxspeedKmPH); SpeedFactor maxspeedFactor(kInvalidModelValue, kInvalidModelValue); - auto const typeKey = GetHighwayTypeKey(*type); + auto const typeKey = GetHighwayTypeKey(type); auto const local = m_highwayBasedInfo.m_localFactors.find(typeKey); if (local != m_highwayBasedInfo.m_localFactors.cend()) { @@ -215,25 +226,22 @@ SpeedKMpH VehicleModel::GetSpeedOnFeatureWithMaxspeed(boost::optional(speed * maxspeedFactor, maxModelSpeed); } -SpeedKMpH VehicleModel::GetSpeedOnFeatureWithoutMaxspeed(boost::optional const & type, +SpeedKMpH VehicleModel::GetSpeedOnFeatureWithoutMaxspeed(HighwayType const & type, SpeedParams const & speedParams) const { CHECK(!speedParams.m_maxspeed.IsValid(), ()); auto const isCityRoad = speedParams.m_inCity; - if (!type) - return m_maxModelSpeed.GetSpeed(isCityRoad); - SpeedKMpH const & maxModelSpeed = m_maxModelSpeed.GetSpeed(isCityRoad); SpeedKMpH speed(kInvalidModelValue, kInvalidModelValue); - auto const local = m_highwayBasedInfo.m_localSpeeds.find(*type); + auto const local = m_highwayBasedInfo.m_localSpeeds.find(type); if (local != m_highwayBasedInfo.m_localSpeeds.cend()) speed = local->second.GetSpeed(isCityRoad); if (speed.m_weight != kInvalidModelValue && speed.m_eta != kInvalidModelValue) return speed; - auto const globalIt = m_highwayBasedInfo.m_globalSpeeds.find(*type); - CHECK(globalIt != m_highwayBasedInfo.m_globalSpeeds.cend(), ("Can't find type in global speeds", *type)); + auto const globalIt = m_highwayBasedInfo.m_globalSpeeds.find(type); + CHECK(globalIt != m_highwayBasedInfo.m_globalSpeeds.cend(), ("Can't find type in global speeds", type)); SpeedKMpH const & global = globalIt->second.GetSpeed(isCityRoad); CHECK_NOT_EQUAL(global.m_weight, kInvalidModelValue, ()); CHECK_NOT_EQUAL(global.m_eta, kInvalidModelValue, ()); @@ -255,7 +263,9 @@ SpeedKMpH VehicleModel::GetTypeSpeed(feature::TypesHolder const & types, boost::optional additionalRoadSpeed; for (uint32_t t : types) { - GetHighwayType(t, hwType); + if (!hwType) + hwType = GetHighwayType(t); + GetSurfaceFactor(t, surfaceFactor); GetAdditionalRoadSpeed(t, isCityRoad, additionalRoadSpeed); } @@ -263,10 +273,11 @@ SpeedKMpH VehicleModel::GetTypeSpeed(feature::TypesHolder const & types, if (additionalRoadSpeed) return *additionalRoadSpeed * surfaceFactor; + auto const resultHwType = *hwType; if (speedParams.m_maxspeed.IsValid()) - return GetSpeedOnFeatureWithMaxspeed(hwType, speedParams) * surfaceFactor; + return GetSpeedOnFeatureWithMaxspeed(resultHwType, speedParams) * surfaceFactor; - return GetSpeedOnFeatureWithoutMaxspeed(hwType, speedParams) * surfaceFactor; + return GetSpeedOnFeatureWithoutMaxspeed(resultHwType, speedParams) * surfaceFactor; } SpeedKMpH VehicleModel::GetSpeedWihtoutMaxspeed(FeatureType & f, @@ -452,6 +463,7 @@ string DebugPrint(HighwayType type) case HighwayType::HighwayPlatform: return "highway-platform"; case HighwayType::RouteShuttleTrain: return "route-shuttle_train"; } + UNREACHABLE(); } } // namespace routing diff --git a/routing_common/vehicle_model.hpp b/routing_common/vehicle_model.hpp index 6c6cd7cec7..3879b557f7 100644 --- a/routing_common/vehicle_model.hpp +++ b/routing_common/vehicle_model.hpp @@ -223,6 +223,8 @@ public: /// @param inCity is true if |f| lies in a city of town. virtual SpeedKMpH GetSpeed(FeatureType & f, SpeedParams const & speedParams) const = 0; + virtual HighwayType GetHighwayType(FeatureType & f) const = 0; + /// @return Maximum model weight speed. /// All speeds which the model returns must be less than or equal to this speed. virtual double GetMaxWeightSpeed() const = 0; @@ -302,6 +304,7 @@ public: /// VehicleModelInterface overrides: SpeedKMpH GetSpeed(FeatureType & f, SpeedParams const & speedParams) const override; + HighwayType GetHighwayType(FeatureType & f) const override; double GetMaxWeightSpeed() const override; bool IsOneWay(FeatureType & f) const override; bool IsRoad(FeatureType & f) const override; @@ -391,14 +394,14 @@ private: SpeedFactor m_factor; }; - void GetHighwayType(uint32_t type, boost::optional & hwType) const; + boost::optional GetHighwayType(uint32_t type) const; void GetSurfaceFactor(uint32_t type, SpeedFactor & factor) const; void GetAdditionalRoadSpeed(uint32_t type, bool isCityRoad, boost::optional & speed) const; - SpeedKMpH GetSpeedOnFeatureWithoutMaxspeed(boost::optional const & type, - SpeedParams const & speedParams) const; - SpeedKMpH GetSpeedOnFeatureWithMaxspeed(boost::optional const & type, - SpeedParams const & speedParams) const; + SpeedKMpH GetSpeedOnFeatureWithoutMaxspeed(HighwayType const & type, + SpeedParams const & speedParams) const; + SpeedKMpH GetSpeedOnFeatureWithMaxspeed(HighwayType const & type, + SpeedParams const & speedParams) const; std::vector::const_iterator FindAdditionalRoadType(uint32_t type) const; diff --git a/track_analyzing/serialization.hpp b/track_analyzing/serialization.hpp index b689fb2c54..1a5ebf49fd 100644 --- a/track_analyzing/serialization.hpp +++ b/track_analyzing/serialization.hpp @@ -32,7 +32,7 @@ public: { } - template + template void Serialize(MwmToMatchedTracks const & mwmToMatchedTracks, Sink & sink) { WriteSize(sink, mwmToMatchedTracks.size()); @@ -58,13 +58,13 @@ public: CHECK(!track.empty(), ()); WriteSize(sink, track.size()); - for (MatchedTrackPoint const & point : track) - Serialize(point.GetSegment(), sink); - std::vector dataPoints; dataPoints.reserve(track.size()); for (MatchedTrackPoint const & point : track) - dataPoints.push_back(point.GetDataPoint()); + { + Serialize(point.GetSegment(), sink); + dataPoints.emplace_back(point.GetDataPoint()); + } std::vector buffer; MemWriter memWriter(buffer); @@ -78,7 +78,7 @@ public: } } - template + template void Deserialize(MwmToMatchedTracks & mwmToMatchedTracks, Source & src) { mwmToMatchedTracks.clear(); @@ -111,8 +111,8 @@ public: std::vector segments; segments.resize(numSegments); - for (size_t iSeg = 0; iSeg < numSegments; ++iSeg) - Deserialize(mwmId, segments[iSeg], src); + for (size_t i = 0; i < numSegments; ++i) + Deserialize(mwmId, segments[i], src); std::vector buffer; auto const bufferSize = ReadSize(src); @@ -130,8 +130,8 @@ public: MatchedTrack & track = tracks[iTrack]; track.reserve(numSegments); - for (size_t iPoint = 0; iPoint < numSegments; ++iPoint) - track.emplace_back(dataPoints[iPoint], segments[iPoint]); + for (size_t i = 0; i < numSegments; ++i) + track.emplace_back(dataPoints[i], segments[i]); } } } @@ -141,19 +141,19 @@ private: static uint8_t constexpr kForward = 0; static uint8_t constexpr kBackward = 1; - template + template static void WriteSize(Sink & sink, size_t size) { WriteVarUint(sink, base::checked_cast(size)); } - template + template static size_t ReadSize(Source & src) { return base::checked_cast(ReadVarUint(src)); } - template + template static void Serialize(routing::Segment const & segment, Sink & sink) { WriteToSink(sink, segment.GetFeatureId()); @@ -162,7 +162,7 @@ private: WriteToSink(sink, direction); } - template + template static void Deserialize(routing::NumMwmId numMwmId, routing::Segment & segment, Source & src) { auto const featureId = ReadPrimitiveFromSource(src); diff --git a/track_analyzing/track_analyzer/CMakeLists.txt b/track_analyzing/track_analyzer/CMakeLists.txt index 2a0b46c5df..bfef337533 100644 --- a/track_analyzing/track_analyzer/CMakeLists.txt +++ b/track_analyzing/track_analyzer/CMakeLists.txt @@ -11,6 +11,8 @@ set( cmd_track.cpp cmd_tracks.cpp cmd_unmatched_tracks.cpp + crossroad_checker.cpp + crossroad_checker.hpp track_analyzer.cpp utils.cpp utils.hpp diff --git a/track_analyzing/track_analyzer/cmd_table.cpp b/track_analyzing/track_analyzer/cmd_table.cpp index 6bf853f934..fc68a0d0d1 100644 --- a/track_analyzing/track_analyzer/cmd_table.cpp +++ b/track_analyzing/track_analyzer/cmd_table.cpp @@ -1,8 +1,12 @@ +#include "track_analyzing/track_analyzer/crossroad_checker.hpp" + #include "track_analyzing/track.hpp" #include "track_analyzing/utils.hpp" #include "routing/city_roads.hpp" #include "routing/geometry.hpp" +#include "routing/index_graph.hpp" +#include "routing/index_graph_loader.hpp" #include "routing/maxspeeds.hpp" #include "routing_common/car_model.hpp" @@ -12,6 +16,7 @@ #include "traffic/speed_groups.hpp" #include "indexer/classificator.hpp" +#include "indexer/data_source.hpp" #include "indexer/feature.hpp" #include "indexer/feature_data.hpp" #include "indexer/features_vector.hpp" @@ -25,19 +30,25 @@ #include "base/assert.hpp" #include "base/file_name_utils.hpp" +#include "base/stl_helpers.hpp" #include "base/sunrise_sunset.hpp" #include "base/timer.hpp" #include +#include #include #include #include #include #include +#include #include +#include #include #include +#include + #include "defines.hpp" using namespace routing; @@ -47,6 +58,7 @@ using namespace track_analyzing; namespace { MaxspeedType constexpr kMaxspeedTopBound = 200; +auto constexpr kValidTrafficValue = traffic::SpeedGroup::G5; string TypeToString(uint32_t type) { @@ -90,15 +102,12 @@ public: { bool operator<(Type const & rhs) const { - if (m_hwType != rhs.m_hwType) - return m_hwType < rhs.m_hwType; - - return m_surfaceType < rhs.m_surfaceType; + return tie(m_hwType, m_surfaceType) < tie(rhs.m_hwType, rhs.m_surfaceType); } bool operator==(Type const & rhs) const { - return m_hwType == rhs.m_hwType && m_surfaceType == rhs.m_surfaceType; + return tie(m_hwType, m_surfaceType) == tie(rhs.m_hwType, m_surfaceType); } bool operator!=(Type const & rhs) const @@ -151,10 +160,8 @@ struct RoadInfo { bool operator==(RoadInfo const & rhs) const { - return m_type == rhs.m_type && - m_maxspeedKMpH == rhs.m_maxspeedKMpH && - m_isCityRoad == rhs.m_isCityRoad && - m_isOneWay == rhs.m_isOneWay; + return tie(m_type, m_maxspeedKMpH, m_isCityRoad, m_isOneWay) == + tie(rhs.m_type, rhs.m_maxspeedKMpH, rhs.m_isCityRoad, rhs.m_isOneWay); } bool operator!=(RoadInfo const & rhs) const @@ -164,16 +171,8 @@ struct RoadInfo bool operator<(RoadInfo const & rhs) const { - if (m_type != rhs.m_type) - return m_type < rhs.m_type; - - if (m_maxspeedKMpH != rhs.m_maxspeedKMpH) - return m_maxspeedKMpH < rhs.m_maxspeedKMpH; - - if (m_isCityRoad != rhs.m_isCityRoad) - return m_isCityRoad < rhs.m_isCityRoad; - - return m_isOneWay < rhs.m_isOneWay; + return tie(m_type, m_maxspeedKMpH, m_isCityRoad, m_isOneWay) < + tie(rhs.m_type, rhs.m_maxspeedKMpH, rhs.m_isCityRoad, rhs.m_isOneWay); } string GetSummary() const @@ -207,18 +206,14 @@ public: bool operator==(MoveType const & rhs) const { - return m_roadInfo == rhs.m_roadInfo && m_speedGroup == rhs.m_speedGroup && m_isDayTime == rhs.m_isDayTime; + return tie(m_roadInfo, m_speedGroup) == tie(rhs.m_roadInfo, rhs.m_speedGroup); } bool operator<(MoveType const & rhs) const { - if (m_roadInfo != rhs.m_roadInfo) - return m_roadInfo < rhs.m_roadInfo; - - if (m_speedGroup != rhs.m_speedGroup) - return m_speedGroup < rhs.m_speedGroup; - - return m_isDayTime < rhs.m_isDayTime; + auto const lhsGroup = base::Underlying(m_speedGroup); + auto const rhsGroup = base::Underlying(rhs.m_speedGroup); + return tie(m_roadInfo, lhsGroup) < tie(rhs.m_roadInfo, rhsGroup); } bool IsValid() const @@ -226,7 +221,7 @@ public: // In order to collect cleaner data we don't use speed group lower than G5. return m_roadInfo.m_type.m_hwType != 0 && m_roadInfo.m_type.m_surfaceType != 0 && - m_speedGroup == traffic::SpeedGroup::G5; + m_speedGroup == kValidTrafficValue; } string GetSummary() const @@ -249,30 +244,40 @@ private: class SpeedInfo final { public: - void Add(double distance, uint64_t time) + void Add(double distance, uint64_t time, IsCrossroadChecker::CrossroadInfo const & crossroads) { m_totalDistance += distance; m_totalTime += time; + IsCrossroadChecker::MergeCrossroads(crossroads, m_crossroads); } - void Add(SpeedInfo const & rhs) { Add(rhs.m_totalDistance, rhs.m_totalTime); } - string GetSummary() const { ostringstream out; - out << m_totalDistance << "," << m_totalTime << "," << CalcSpeedKMpH(m_totalDistance, m_totalTime); + out << m_totalDistance << "," + << m_totalTime << "," + << CalcSpeedKMpH(m_totalDistance, m_totalTime) << ","; + + for (size_t i = 1; i < m_crossroads.size(); ++i) + { + out << m_crossroads[i]; + if (i != m_crossroads.size() - 1) + out << ","; + } + return out.str(); } private: double m_totalDistance = 0.0; uint64_t m_totalTime = 0; + IsCrossroadChecker::CrossroadInfo m_crossroads{}; }; class MoveTypeAggregator final { public: - void Add(MoveType && moveType, MatchedTrack::const_iterator begin, + void Add(MoveType && moveType, IsCrossroadChecker::CrossroadInfo const & crossroads, MatchedTrack::const_iterator begin, MatchedTrack::const_iterator end, Geometry & geometry) { if (begin + 1 >= end) @@ -280,13 +285,7 @@ public: uint64_t const time = (end - 1)->GetDataPoint().m_timestamp - begin->GetDataPoint().m_timestamp; double const length = CalcSubtrackLength(begin, end, geometry); - m_moveInfos[moveType].Add(length, time); - } - - void Add(MoveTypeAggregator const & rhs) - { - for (auto it : rhs.m_moveInfos) - m_moveInfos[it.first].Add(it.second); + m_moveInfos[moveType].Add(length, time, crossroads); } string GetSummary(string const & user, string const & mwm) const @@ -365,10 +364,13 @@ namespace track_analyzing void CmdTagsTable(string const & filepath, string const & trackExtension, StringFilter mwmFilter, StringFilter userFilter) { - cout << "user,mwm,hw type,surface type,maxspeed km/h,is city road,is one way,is day,lat lon,distance,time,mean speed km/h\n"; + cout << "user,mwm,hw type,surface type,maxspeed km/h,is city road,is one way,is day,lat lon,distance,time," + "mean speed km/h,turn from smaller to bigger,turn from bigger to smaller,from link,to link," + "intersection with big,intersection with small,intersection with link\n"; storage::Storage storage; storage.RegisterAllLocalMaps(false /* enableDiffs */); + FrozenDataSource dataSource; auto numMwmIds = CreateNumMwmIds(storage); auto processMwm = [&](string const & mwmName, UserToMatchedTracks const & userToMatchedTracks) { @@ -376,11 +378,29 @@ void CmdTagsTable(string const & filepath, string const & trackExtension, String return; auto const countryName = storage.GetTopmostParentFor(mwmName); + auto const carModelFactory = make_shared(VehicleModelFactory::CountryParentNameGetterFn{}); shared_ptr vehicleModel = - CarModelFactory({}).GetVehicleModelForCountry(mwmName); + carModelFactory->GetVehicleModelForCountry(mwmName); string const mwmFile = GetCurrentVersionMwmFile(storage, mwmName); MatchedTrackPointToMoveType pointToMoveType(FilesContainerR(make_unique(mwmFile)), *vehicleModel); Geometry geometry(GeometryLoader::CreateFromFile(mwmFile, vehicleModel)); + auto const vehicleType = VehicleType::Car; + auto const edgeEstimator = EdgeEstimator::Create(vehicleType, *vehicleModel, nullptr /* trafficStash */); + auto indexGraphLoader = IndexGraphLoader::Create(vehicleType, false /* loadAltitudes */, numMwmIds, + carModelFactory, edgeEstimator, dataSource); + + platform::CountryFile const countryFile(mwmName); + auto localCountryFile = storage.GetLatestLocalFile(countryFile); + CHECK(localCountryFile, ("Can't find latest country file for", countryFile.GetName())); + if (!dataSource.IsLoaded(countryFile)) + { + auto registerResult = dataSource.Register(*localCountryFile); + CHECK_EQUAL(registerResult.second, MwmSet::RegResult::Success, + ("Can't register mwm", countryFile.GetName())); + } + + auto const mwmId = numMwmIds->GetId(countryFile); + IsCrossroadChecker checker(indexGraphLoader->GetIndexGraph(mwmId), geometry); for (auto const & kv : userToMatchedTracks) { @@ -388,23 +408,36 @@ void CmdTagsTable(string const & filepath, string const & trackExtension, String if (userFilter(user)) continue; - for (size_t trackIdx = 0; trackIdx < kv.second.size(); ++trackIdx) + for (auto const & track : kv.second) { - MatchedTrack const & track = kv.second[trackIdx]; if (track.size() <= 1) continue; MoveTypeAggregator aggregator; - for (auto subTrackBegin = track.begin(); subTrackBegin != track.end();) + IsCrossroadChecker::CrossroadInfo info{}; + for (auto subtrackBegin = track.begin(); subtrackBegin != track.end();) { - auto moveType = pointToMoveType.GetMoveType(*subTrackBegin); - auto subTrackEnd = subTrackBegin + 1; - while (subTrackEnd != track.end() && - pointToMoveType.GetMoveType(*subTrackEnd) == moveType) - ++subTrackEnd; + auto moveType = pointToMoveType.GetMoveType(*subtrackBegin); + auto prev = subtrackBegin; + auto end = subtrackBegin + 1; + while (end != track.end() && pointToMoveType.GetMoveType(*end) == moveType) + { + IsCrossroadChecker::MergeCrossroads(checker(prev->GetSegment(), end->GetSegment()), info); + prev = end; + ++end; + } - aggregator.Add(move(moveType), subTrackBegin, subTrackEnd, geometry); - subTrackBegin = subTrackEnd; + // If it's not the end of the track than it could be a crossroad. + IsCrossroadChecker::CrossroadInfo crossroad{}; + if (end != track.end()) + { + crossroad = checker(prev->GetSegment(), end->GetSegment()); + IsCrossroadChecker::MergeCrossroads(crossroad, info); + } + + aggregator.Add(move(moveType), info, subtrackBegin, end, geometry); + subtrackBegin = end; + info = move(crossroad); } auto const summary = aggregator.GetSummary(user, countryName); diff --git a/track_analyzing/track_analyzer/crossroad_checker.cpp b/track_analyzing/track_analyzer/crossroad_checker.cpp new file mode 100644 index 0000000000..689a595d0c --- /dev/null +++ b/track_analyzing/track_analyzer/crossroad_checker.cpp @@ -0,0 +1,162 @@ +#include "track_analyzing/track_analyzer/crossroad_checker.hpp" + +#include "routing/joint.hpp" +#include "routing/segment.hpp" + +#include "routing_common/vehicle_model.hpp" + +#include "base/assert.hpp" + +#include + +using namespace std; + +namespace +{ +using namespace routing; + +bool IsHighwayLink(HighwayType type) +{ + switch (type) + { + case HighwayType::HighwayMotorwayLink: + case HighwayType::HighwayTrunkLink: + case HighwayType::HighwayPrimaryLink: + case HighwayType::HighwaySecondaryLink: + case HighwayType::HighwayTertiaryLink: + return true; + default: + return false; + } + + UNREACHABLE(); +} + +bool IsBigHighway(HighwayType type) +{ + switch (type) + { + case HighwayType::HighwayMotorway: + case HighwayType::HighwayTrunk: + case HighwayType::HighwayPrimary: + case HighwayType::HighwaySecondary: + case HighwayType::HighwayTertiary: + return true; + default: + return false; + } + + UNREACHABLE(); +} + +bool FromSmallerToBigger(HighwayType lhs, HighwayType rhs) +{ + CHECK_NOT_EQUAL(lhs, rhs, ()); + + static std::array constexpr kHighwayTypes = { + HighwayType::HighwayTertiary, + HighwayType::HighwaySecondary, + HighwayType::HighwayPrimary, + HighwayType::HighwayTrunk, + HighwayType::HighwayMotorway + }; + + auto const lhsIt = find(kHighwayTypes.begin(), kHighwayTypes.end(), lhs); + auto const rhsIt = find(kHighwayTypes.begin(), kHighwayTypes.end(), rhs); + if (lhsIt == kHighwayTypes.end() && rhsIt != kHighwayTypes.end()) + return true; + + if (lhsIt != kHighwayTypes.end() && rhsIt == kHighwayTypes.end()) + return false; + + return lhsIt < rhsIt; +} +} // namespace + +namespace routing +{ +IsCrossroadChecker::CrossroadInfo IsCrossroadChecker::operator()(Segment const & current, Segment const & next) const +{ + IsCrossroadChecker::CrossroadInfo ret{}; + if (current == next) + return ret; + + auto const currentSegmentFeatureId = current.GetFeatureId(); + auto const currentSegmentHwType = m_geometry.GetRoad(currentSegmentFeatureId).GetHighwayType(); + auto const nextSegmentFeatureId = next.GetFeatureId(); + auto const nextSegmentHwType = m_geometry.GetRoad(nextSegmentFeatureId).GetHighwayType(); + auto const currentRoadPoint = current.GetRoadPoint(true /* isFront */); + auto const jointId = m_indexGraph.GetJointId(currentRoadPoint); + if (jointId == Joint::kInvalidId) + return ret; + + if (currentSegmentFeatureId != nextSegmentFeatureId && currentSegmentHwType != nextSegmentHwType) + { + // Changing highway type. + if (IsHighwayLink(currentSegmentHwType)) + ++ret[base::Underlying(Type::FromLink)]; + + if (IsHighwayLink(nextSegmentHwType)) + ++ret[base::Underlying(Type::ToLink)]; + + // It's move without links. + if (!ret[base::Underlying(Type::FromLink)] && !ret[base::Underlying(Type::ToLink)]) + { + bool const currentIsBig = IsBigHighway(currentSegmentHwType); + bool const nextIsBig = IsBigHighway(nextSegmentHwType); + if (currentIsBig && !nextIsBig) + { + ++ret[base::Underlying(Type::TurnFromBiggerToSmaller)]; + } + else if (!currentIsBig && nextIsBig) + { + ++ret[base::Underlying(Type::TurnFromSmallerToBigger)]; + } + else if (currentIsBig && nextIsBig) + { + // Both roads are big but one is bigger. + auto const type = FromSmallerToBigger(currentSegmentHwType, nextSegmentHwType) ? + Type::TurnFromSmallerToBigger : Type::TurnFromBiggerToSmaller; + ++ret[base::Underlying(type)]; + } + } + } + + auto const nextRoadPoint = next.GetRoadPoint(false /* isFront */); + m_indexGraph.ForEachPoint(jointId, [&](RoadPoint const & point) { + // Check for already included roads. + auto const pointFeatureId = point.GetFeatureId(); + if (pointFeatureId == currentSegmentFeatureId || pointFeatureId == nextSegmentFeatureId) + return; + + auto const & roadGeometry = m_geometry.GetRoad(pointFeatureId); + auto const pointHwType = roadGeometry.GetHighwayType(); + if (pointHwType == nextSegmentHwType) + { + // Is the same road but parted on different features. + if (roadGeometry.IsEndPointId(point.GetPointId()) && + roadGeometry.IsEndPointId(nextRoadPoint.GetPointId())) + return; + } + + if (IsHighwayLink(pointHwType)) + { + ++ret[base::Underlying(Type::IntersectionWithLink)]; + return; + } + + auto const type = IsBigHighway(pointHwType) ? Type::IntersectionWithBig : Type::IntersectionWithSmall; + ++ret[base::Underlying(type)]; + }); + + return ret; +} + +// static +void IsCrossroadChecker::MergeCrossroads(IsCrossroadChecker::CrossroadInfo const & from, + IsCrossroadChecker::CrossroadInfo & to) +{ + for (size_t i = 0; i < from.size(); ++i) + to[i] += from[i]; +} +} // namespace routing diff --git a/track_analyzing/track_analyzer/crossroad_checker.hpp b/track_analyzing/track_analyzer/crossroad_checker.hpp new file mode 100644 index 0000000000..ca6b9b53be --- /dev/null +++ b/track_analyzing/track_analyzer/crossroad_checker.hpp @@ -0,0 +1,42 @@ +#pragma once + +#include "routing/geometry.hpp" +#include "routing/index_graph.hpp" +#include "routing/segment.hpp" + +#include "base/stl_helpers.hpp" + +#include + +namespace routing +{ +class IsCrossroadChecker +{ +public: + enum class Type + { + None, + TurnFromSmallerToBigger, + TurnFromBiggerToSmaller, + FromLink, + ToLink, + IntersectionWithBig, + IntersectionWithSmall, + IntersectionWithLink, + Count + }; + + using CrossroadInfo = std::array; + + IsCrossroadChecker(IndexGraph & indexGraph, Geometry & geometry) : m_indexGraph(indexGraph), m_geometry(geometry) {} + /// \brief Compares two segments by their highway type to find if there was a crossroad between them. + /// Check if current segment is a joint to find and find all intersections with other roads. + CrossroadInfo operator()(Segment const & current, Segment const & next) const; + + static void MergeCrossroads(CrossroadInfo const & from, CrossroadInfo & to); + +private: + IndexGraph & m_indexGraph; + Geometry & m_geometry; +}; +} // namespace routing