diff --git a/generator/camera_info_collector.cpp b/generator/camera_info_collector.cpp index bc6710c56f..0627270f5d 100644 --- a/generator/camera_info_collector.cpp +++ b/generator/camera_info_collector.cpp @@ -24,14 +24,14 @@ CamerasInfoCollector::CamerasInfoCollector(std::string const & dataFilePath, std::string const & camerasInfoPath, std::string const & osmIdsToFeatureIdsPath) { - std::map osmIdToFeatureId; - if (!routing::ParseRoadsOsmIdToFeatureIdMapping(osmIdsToFeatureIdsPath, osmIdToFeatureId)) + std::map> osmIdToFeatureIds; + if (!routing::ParseRoadsOsmIdToFeatureIdMapping(osmIdsToFeatureIdsPath, osmIdToFeatureIds)) { LOG(LCRITICAL, ("An error happened while parsing feature id to osm ids mapping from file:", osmIdsToFeatureIdsPath)); } - if (!ParseIntermediateInfo(camerasInfoPath, osmIdToFeatureId)) + if (!ParseIntermediateInfo(camerasInfoPath, osmIdToFeatureIds)) LOG(LCRITICAL, ("Unable to parse intermediate file(", camerasInfoPath, ") about cameras info")); platform::LocalCountryFile file = platform::LocalCountryFile::MakeTemporary(dataFilePath); @@ -96,8 +96,9 @@ void CamerasInfoCollector::Serialize(FileWriter & writer) const camera.Serialize(writer, prevFeatureId); } -bool CamerasInfoCollector::ParseIntermediateInfo(std::string const & camerasInfoPath, - std::map const & osmIdToFeatureId) +bool CamerasInfoCollector::ParseIntermediateInfo( + std::string const & camerasInfoPath, + std::map> const & osmIdToFeatureIds) { FileReader reader(camerasInfoPath); ReaderSource src(reader); @@ -144,9 +145,14 @@ bool CamerasInfoCollector::ParseIntermediateInfo(std::string const & camerasInfo { ReadPrimitiveFromSource(src, wayOsmId); - auto const it = osmIdToFeatureId.find(base::MakeOsmWay(wayOsmId)); - if (it != osmIdToFeatureId.cend()) - ways.emplace_back(it->second /* featureId */, 0 /* segmentId */, 0 /* coef */); + auto const it = osmIdToFeatureIds.find(base::MakeOsmWay(wayOsmId)); + if (it != osmIdToFeatureIds.cend()) + { + auto const & featureIdsVec = it->second; + // Note. One |wayOsmId| may correspond several feature ids. + for (auto const featureId : featureIdsVec) + ways.emplace_back(featureId, 0 /* segmentId */, 0 /* coef */); + } } auto const speed = base::asserted_cast(maxSpeedKmPH); @@ -282,10 +288,6 @@ std::optional> CamerasInfoCollector::Camera::FindMys ft.ForEachPoint(findPoint, scales::GetUpperScale()); -// CHECK(found, ("Cannot find camera point at the feature:", wayFeatureId, -// "; camera center:", mercator::ToLatLon(m_data.m_center))); - // @TODO(bykoianko) The invariant above is valid. But as a fast fix, let's remove - // all the cameras in case of no feature ends matches. if (!found) { cannotFindMyself = true; @@ -308,8 +310,14 @@ std::optional> CamerasInfoCollector::Camera::FindMys if (cannotFindMyself) { - LOG(LWARNING, ("Cannot find camera point at the feature:", wayFeatureId, - "; camera center:", mercator::ToLatLon(m_data.m_center))); + // Note. Speed camera with coords |m_data.m_center| is not located at any point of |wayFeatureId|. + // It may happens if osm feature corresponds to |wayFeatureId| is split by a mini_roundabout + // or turning_loop. There are two cases: + // * |m_data.m_center| is located at a point of another part of the whole osm feature. In + // that case it will be found on another call of FindMyself() method. + // * |m_data.m_center| coincides with point of mini_roundabout or turning_loop. It means + // there on feature point which coincides with speed camera (|m_data.m_center|). + // These camera (notification) will be lost. return {}; } diff --git a/generator/camera_info_collector.hpp b/generator/camera_info_collector.hpp index d86e0cbcd6..83693157fa 100644 --- a/generator/camera_info_collector.hpp +++ b/generator/camera_info_collector.hpp @@ -80,8 +80,9 @@ private: inline static double constexpr kMaxDistFromCameraToClosestSegmentMeters = 20.0; inline static double constexpr kSearchCameraRadiusMeters = 10.0; - bool ParseIntermediateInfo(std::string const & camerasInfoPath, - std::map const & osmIdToFeatureId); + bool ParseIntermediateInfo( + std::string const & camerasInfoPath, + std::map> const & osmIdToFeatureIds); std::vector m_cameras; }; diff --git a/generator/metalines_builder.cpp b/generator/metalines_builder.cpp index 4944337fb5..a75c4bd2fc 100644 --- a/generator/metalines_builder.cpp +++ b/generator/metalines_builder.cpp @@ -225,8 +225,8 @@ void MetalinesBuilder::MergeInto(MetalinesBuilder & collector) const bool WriteMetalinesSection(std::string const & mwmPath, std::string const & metalinesPath, std::string const & osmIdsToFeatureIdsPath) { - std::map osmIdToFeatureId; - if (!routing::ParseRoadsOsmIdToFeatureIdMapping(osmIdsToFeatureIdsPath, osmIdToFeatureId)) + std::map> osmIdToFeatureIds; + if (!routing::ParseRoadsOsmIdToFeatureIdMapping(osmIdsToFeatureIdsPath, osmIdToFeatureIds)) return false; FileReader reader(metalinesPath); @@ -243,12 +243,14 @@ bool WriteMetalinesSection(std::string const & mwmPath, std::string const & meta for (auto const wayId : ways) { // We get a negative wayId when a feature direction should be reversed. - auto fid = osmIdToFeatureId.find(base::MakeOsmWay(std::abs(wayId))); - if (fid == osmIdToFeatureId.cend()) + auto fids = osmIdToFeatureIds.find(base::MakeOsmWay(std::abs(wayId))); + if (fids == osmIdToFeatureIds.cend()) break; // Keeping the sign for the feature direction. - int32_t const featureId = static_cast(fid->second); + // @TODO(bykoianko) All the feature ids should be used instead of |fids->second[0]|. + CHECK(!fids->second.empty(), ()); + int32_t const featureId = static_cast(fids->second.front()); featureIds.push_back(wayId > 0 ? featureId : -featureId); } diff --git a/generator/restriction_collector.cpp b/generator/restriction_collector.cpp index 8ea83b7d40..5e156d54e7 100644 --- a/generator/restriction_collector.cpp +++ b/generator/restriction_collector.cpp @@ -52,7 +52,7 @@ RestrictionCollector::RestrictionCollector(std::string const & osmIdsToFeatureId std::unique_ptr && graph) : m_indexGraph(std::move(graph)) { - CHECK(ParseRoadsOsmIdToFeatureIdMapping(osmIdsToFeatureIdPath, m_osmIdToFeatureId), + CHECK(ParseRoadsOsmIdToFeatureIdMapping(osmIdsToFeatureIdPath, m_osmIdToFeatureIds), ("An error happened while parsing feature id to " "osm ids mapping from file:", osmIdsToFeatureIdPath)); } @@ -62,7 +62,7 @@ bool RestrictionCollector::Process(std::string const & restrictionPath) CHECK(m_indexGraph, ()); SCOPE_GUARD(clean, [this]() { - m_osmIdToFeatureId.clear(); + m_osmIdToFeatureIds.clear(); m_restrictions.clear(); }); @@ -311,16 +311,18 @@ bool RestrictionCollector::AddRestriction(m2::PointD const & coords, std::vector featureIds(osmIds.size()); for (size_t i = 0; i < osmIds.size(); ++i) { - auto const result = m_osmIdToFeatureId.find(osmIds[i]); - if (result == m_osmIdToFeatureId.cend()) + auto const result = m_osmIdToFeatureIds.find(osmIds[i]); + if (result == m_osmIdToFeatureIds.cend()) { - // It could happend near mwm border when one of a restriction lines is not included in mwm + // It could happens near mwm border when one of a restriction lines is not included in mwm // but the restriction is included. return false; } // Only one feature id is found for |osmIds[i]|. - featureIds[i] = result->second; + // @TODO(bykoianko) All the feature ids should be used instead of |result->second.front()|. + CHECK(!result->second.empty(), ()); + featureIds[i] = result->second.front(); } if (!IsRestrictionValid(restrictionType, coords, featureIds)) @@ -332,7 +334,7 @@ bool RestrictionCollector::AddRestriction(m2::PointD const & coords, void RestrictionCollector::AddFeatureId(uint32_t featureId, base::GeoObjectId osmId) { - ::routing::AddFeatureId(osmId, featureId, m_osmIdToFeatureId); + ::routing::AddFeatureId(osmId, featureId, m_osmIdToFeatureIds); } void FromString(std::string const & str, Restriction::Type & type) diff --git a/generator/restriction_collector.hpp b/generator/restriction_collector.hpp index 9cfa268615..225df06a5d 100644 --- a/generator/restriction_collector.hpp +++ b/generator/restriction_collector.hpp @@ -91,7 +91,7 @@ private: std::vector const & osmIds); std::vector m_restrictions; - std::map m_osmIdToFeatureId; + std::map> m_osmIdToFeatureIds; std::unique_ptr m_indexGraph; diff --git a/generator/road_access_generator.cpp b/generator/road_access_generator.cpp index d97b8aa552..52ebe079b2 100644 --- a/generator/road_access_generator.cpp +++ b/generator/road_access_generator.cpp @@ -163,7 +163,7 @@ set const kHighwaysWhereIgnoreBarriersWithoutAccess = { }; bool ParseRoadAccess(string const & roadAccessPath, - map const & osmIdToFeatureId, + map> const & osmIdToFeatureIds, RoadAccessCollector::RoadAccessByVehicleType & roadAccessByVehicleType) { ifstream stream(roadAccessPath); @@ -240,14 +240,16 @@ bool ParseRoadAccess(string const & roadAccessPath, } ++iter; - auto const it = osmIdToFeatureId.find(base::MakeOsmWay(osmId)); + auto const it = osmIdToFeatureIds.find(base::MakeOsmWay(osmId)); // Even though this osm element has a tag that is interesting for us, // we have not created a feature from it. Possible reasons: // no primary tag, unsupported type, etc. - if (it == osmIdToFeatureId.cend()) + if (it == osmIdToFeatureIds.cend()) continue; - uint32_t const featureId = it->second; + // @TODO(bykoianko) All the feature ids should be used instead of |it->second.front()|. + CHECK(!it->second.empty(), ()); + uint32_t const featureId = it->second.front(); if (pointIdx == 0) addFeature(featureId, vehicleType, roadAccessType, osmId); @@ -262,7 +264,8 @@ bool ParseRoadAccess(string const & roadAccessPath, } void ParseRoadAccessConditional( - string const & roadAccessPath, map const & osmIdToFeatureId, + string const & roadAccessPath, + map> const & osmIdToFeatureIds, RoadAccessCollector::RoadAccessByVehicleType & roadAccessByVehicleType) { ifstream stream(roadAccessPath); @@ -312,10 +315,12 @@ void ParseRoadAccessConditional( if (conditional.IsEmpty()) continue; - auto const it = osmIdToFeatureId.find(base::MakeOsmWay(osmId)); - if (it == osmIdToFeatureId.end()) + auto const it = osmIdToFeatureIds.find(base::MakeOsmWay(osmId)); + if (it == osmIdToFeatureIds.end()) continue; - uint32_t const featureId = it->second; + // @TODO(bykoianko) All the feature ids should be used instead of |it->second.front()|. + CHECK(!osmIdToFeatureIds.empty(), ()); + uint32_t const featureId = it->second.front(); wayToAccessConditional[static_cast(vehicleType)].emplace(featureId, move(conditional)); } @@ -643,8 +648,8 @@ void RoadAccessWriter::MergeInto(RoadAccessWriter & collector) const RoadAccessCollector::RoadAccessCollector(string const & dataFilePath, string const & roadAccessPath, string const & osmIdsToFeatureIdsPath) { - map osmIdToFeatureId; - if (!ParseRoadsOsmIdToFeatureIdMapping(osmIdsToFeatureIdsPath, osmIdToFeatureId)) + map> osmIdToFeatureIds; + if (!ParseRoadsOsmIdToFeatureIdMapping(osmIdsToFeatureIdsPath, osmIdToFeatureIds)) { LOG(LWARNING, ("An error happened while parsing feature id to osm ids mapping from file:", osmIdsToFeatureIdsPath)); @@ -653,7 +658,7 @@ RoadAccessCollector::RoadAccessCollector(string const & dataFilePath, string con } RoadAccessCollector::RoadAccessByVehicleType roadAccessByVehicleType; - if (!ParseRoadAccess(roadAccessPath, osmIdToFeatureId, roadAccessByVehicleType)) + if (!ParseRoadAccess(roadAccessPath, osmIdToFeatureIds, roadAccessByVehicleType)) { LOG(LWARNING, ("An error happened while parsing road access from file:", roadAccessPath)); m_valid = false; @@ -661,7 +666,7 @@ RoadAccessCollector::RoadAccessCollector(string const & dataFilePath, string con } auto const roadAccessConditional = roadAccessPath + ROAD_ACCESS_CONDITIONAL_EXT; - ParseRoadAccessConditional(roadAccessConditional, osmIdToFeatureId, roadAccessByVehicleType); + ParseRoadAccessConditional(roadAccessConditional, osmIdToFeatureIds, roadAccessByVehicleType); m_valid = true; m_roadAccessByVehicleType.swap(roadAccessByVehicleType); diff --git a/generator/routing_helpers.cpp b/generator/routing_helpers.cpp index ad49a372b9..4f01a48e59 100644 --- a/generator/routing_helpers.cpp +++ b/generator/routing_helpers.cpp @@ -27,19 +27,18 @@ bool ForEachRoadFromFile(string const & filename, ToDo && toDo) namespace routing { void AddFeatureId(base::GeoObjectId osmId, uint32_t featureId, - map & osmIdToFeatureId) + map> & osmIdToFeatureIds) { - // Failing to insert here usually means that two features were created - // from one osm id, for example an area and its boundary. - osmIdToFeatureId.emplace(osmId, featureId); + osmIdToFeatureIds[osmId].push_back(featureId); } -bool ParseRoadsOsmIdToFeatureIdMapping(string const & osmIdsToFeatureIdPath, - map & osmIdToFeatureId) +bool ParseRoadsOsmIdToFeatureIdMapping( + string const & osmIdsToFeatureIdPath, + map> & osmIdToFeatureIds) { return ForEachRoadFromFile(osmIdsToFeatureIdPath, [&](uint32_t featureId, base::GeoObjectId osmId) { - AddFeatureId(osmId, featureId, osmIdToFeatureId); + AddFeatureId(osmId, featureId, osmIdToFeatureIds); }); } diff --git a/generator/routing_helpers.hpp b/generator/routing_helpers.hpp index 56b54001f3..9257c1c690 100644 --- a/generator/routing_helpers.hpp +++ b/generator/routing_helpers.hpp @@ -5,24 +5,30 @@ #include #include #include +#include namespace routing { -// Adds feature id and corresponding |osmId| to |osmIdToFeatureId|. +// Adds |featureId| and corresponding |osmId| to |osmIdToFeatureId|. // Note. In general, one |featureId| may correspond to several osm ids. -// But for a road feature |featureId| corresponds to exactly one osm id. +// Or on the contrary one osm id may correspond to several feature ids. It may happens for example +// when an area and its boundary may correspond to the same osm id. +// As for road features a road |osmId| may correspond to several feature ids if +// the |osmId| is split by a mini_roundabout or a turning_loop. void AddFeatureId(base::GeoObjectId osmId, uint32_t featureId, - std::map & osmIdToFeatureId); + std::map> & osmIdToFeatureIds); // Parses comma separated text file with line in following format: // , , , and so -// on +// on. It may contain several line with the same feature ids. // For example: // 137999, 5170186, // 138000, 5170209, 5143342, // 138001, 5170228, -bool ParseRoadsOsmIdToFeatureIdMapping(std::string const & osmIdsToFeatureIdPath, - std::map & osmIdToFeatureId); +// 137999, 5170197, +bool ParseRoadsOsmIdToFeatureIdMapping( + std::string const & osmIdsToFeatureIdPath, + std::map> & osmIdToFeatureIds); bool ParseRoadsFeatureIdToOsmIdMapping(std::string const & osmIdsToFeatureIdPath, std::map & featureIdToOsmId); } // namespace routing