From c9dc04dfd114400caf0c6737ba0e1d9e8c8a937e Mon Sep 17 00:00:00 2001 From: tatiana-kondakova Date: Mon, 25 Dec 2017 18:12:15 +0300 Subject: [PATCH] Save road points access info --- .../generator_tests/road_access_test.cpp | 22 ++-- generator/road_access_generator.cpp | 108 +++++++++++++++--- generator/road_access_generator.hpp | 6 +- routing/index_graph.cpp | 6 +- routing/index_graph.hpp | 2 +- routing/road_access.cpp | 69 +++++------ routing/road_access.hpp | 26 +++-- routing/road_access_serialization.hpp | 56 +++++++-- routing/road_point.hpp | 7 ++ routing/routing_tests/index_graph_tools.cpp | 7 +- routing/routing_tests/road_access_test.cpp | 16 +-- 11 files changed, 226 insertions(+), 99 deletions(-) diff --git a/generator/generator_tests/road_access_test.cpp b/generator/generator_tests/road_access_test.cpp index 1919959190..333880306a 100644 --- a/generator/generator_tests/road_access_test.cpp +++ b/generator/generator_tests/road_access_test.cpp @@ -133,20 +133,20 @@ UNIT_TEST(RoadAccess_Smoke) UNIT_TEST(RoadAccess_AccessPrivate) { - string const roadAccessContent = R"(Car Private 0)"; + string const roadAccessContent = R"(Car Private 0 0)"; string const osmIdsToFeatureIdsContent = R"(0, 0,)"; auto const roadAccessAllTypes = SaveAndLoadRoadAccess(roadAccessContent, osmIdsToFeatureIdsContent); auto const carRoadAccess = roadAccessAllTypes[static_cast(VehicleType::Car)]; - TEST_EQUAL(carRoadAccess.GetSegmentType(Segment(0, 0, 0, false)), RoadAccess::Type::Private, ()); + TEST_EQUAL(carRoadAccess.GetFeatureType(0 /* featureId */), RoadAccess::Type::Private, ()); } UNIT_TEST(RoadAccess_Access_Multiple_Vehicle_Types) { - string const roadAccessContent = R"(Car Private 10 - Car Private 20 - Bicycle No 30 - Car Destination 40)"; + string const roadAccessContent = R"(Car Private 10 0 + Car Private 20 0 + Bicycle No 30 0 + Car Destination 40 0)"; string const osmIdsToFeatureIdsContent = R"(10, 1, 20, 2, 30, 3, @@ -155,11 +155,11 @@ UNIT_TEST(RoadAccess_Access_Multiple_Vehicle_Types) SaveAndLoadRoadAccess(roadAccessContent, osmIdsToFeatureIdsContent); auto const carRoadAccess = roadAccessAllTypes[static_cast(VehicleType::Car)]; auto const bicycleRoadAccess = roadAccessAllTypes[static_cast(VehicleType::Bicycle)]; - TEST_EQUAL(carRoadAccess.GetSegmentType(Segment(0, 1, 0, false)), RoadAccess::Type::Private, ()); - TEST_EQUAL(carRoadAccess.GetSegmentType(Segment(0, 2, 2, true)), RoadAccess::Type::Private, ()); - TEST_EQUAL(carRoadAccess.GetSegmentType(Segment(0, 3, 1, true)), RoadAccess::Type::Yes, ()); - TEST_EQUAL(carRoadAccess.GetSegmentType(Segment(0, 4, 3, false)), RoadAccess::Type::Destination, + TEST_EQUAL(carRoadAccess.GetFeatureType(1 /* featureId */), RoadAccess::Type::Private, ()); + TEST_EQUAL(carRoadAccess.GetFeatureType(2 /* featureId */), RoadAccess::Type::Private, ()); + TEST_EQUAL(carRoadAccess.GetFeatureType(3 /* featureId */), RoadAccess::Type::Yes, ()); + TEST_EQUAL(carRoadAccess.GetFeatureType(4 /* featureId */), RoadAccess::Type::Destination, ()); - TEST_EQUAL(bicycleRoadAccess.GetSegmentType(Segment(0, 3, 0, false)), RoadAccess::Type::No, ()); + TEST_EQUAL(bicycleRoadAccess.GetFeatureType(3 /* featureId */), RoadAccess::Type::No, ()); } } // namespace diff --git a/generator/road_access_generator.cpp b/generator/road_access_generator.cpp index bf4f56105a..d70086bec5 100644 --- a/generator/road_access_generator.cpp +++ b/generator/road_access_generator.cpp @@ -22,6 +22,7 @@ #include "defines.hpp" #include +#include using namespace routing; using namespace std; @@ -53,6 +54,19 @@ TagMapping const kVehicleTagMapping = { {OsmElement::Tag("vehicle", "destination"), RoadAccess::Type::Destination}, }; +TagMapping const kCarBarriersTagMapping = { + {OsmElement::Tag("barrier", "block"), RoadAccess::Type::No}, + {OsmElement::Tag("barrier", "bollard"), RoadAccess::Type::No}, + {OsmElement::Tag("barrier", "chain"), RoadAccess::Type::No}, + {OsmElement::Tag("barrier", "cycle_barrier"), RoadAccess::Type::No}, + {OsmElement::Tag("barrier", "gate"), RoadAccess::Type::Private}, + {OsmElement::Tag("barrier", "jersey_barrier"), RoadAccess::Type::No}, + {OsmElement::Tag("barrier", "lift_gate"), RoadAccess::Type::Private}, + {OsmElement::Tag("barrier", "log"), RoadAccess::Type::No}, + {OsmElement::Tag("barrier", "motorcycle_barrier"), RoadAccess::Type::No}, + {OsmElement::Tag("barrier", "swing_gate"), RoadAccess::Type::Private}, +}; + TagMapping const kPedestrianTagMapping = { {OsmElement::Tag("foot", "yes"), RoadAccess::Type::Yes}, {OsmElement::Tag("foot", "no"), RoadAccess::Type::No}, @@ -67,6 +81,13 @@ TagMapping const kBicycleTagMapping = { {OsmElement::Tag("bicycle", "destination"), RoadAccess::Type::Destination}, }; +TagMapping const kBicycleBarriersTagMapping = { + {OsmElement::Tag("barrier", "cycle_barrier"), RoadAccess::Type::No}, + {OsmElement::Tag("barrier", "turnstile"), RoadAccess::Type::No}, + {OsmElement::Tag("barrier", "kissing_gate"), RoadAccess::Type::Private}, + {OsmElement::Tag("barrier", "gate"), RoadAccess::Type::Private}, +}; + // Allow everything to keep transit section empty. We'll use pedestrian section for // transit + pedestrian combination. TagMapping const kTransitTagMapping = { @@ -96,12 +117,13 @@ bool ParseRoadAccess(string const & roadAccessPath, map const vector privateRoads; - map segmentType[static_cast(VehicleType::Count)]; + map featureType[static_cast(VehicleType::Count)]; + map pointType[static_cast(VehicleType::Count)]; - auto addSegment = [&](Segment const & segment, VehicleType vehicleType, + auto addFeature = [&](uint32_t featureId, VehicleType vehicleType, RoadAccess::Type roadAccessType, uint64_t osmId) { - auto & m = segmentType[static_cast(vehicleType)]; - auto const emplaceRes = m.emplace(segment, roadAccessType); + auto & m = featureType[static_cast(vehicleType)]; + auto const emplaceRes = m.emplace(featureId, roadAccessType); if (!emplaceRes.second && emplaceRes.first->second != roadAccessType) { LOG(LDEBUG, ("Duplicate road access info for OSM way", osmId, "vehicle:", vehicleType, @@ -109,6 +131,17 @@ bool ParseRoadAccess(string const & roadAccessPath, map const } }; + auto addPoint = [&](RoadPoint const & point, VehicleType vehicleType, + RoadAccess::Type roadAccessType) { + auto & m = pointType[static_cast(vehicleType)]; + auto const emplaceRes = m.emplace(point, roadAccessType); + if (!emplaceRes.second && emplaceRes.first->second != roadAccessType) + { + LOG(LDEBUG, ("Duplicate road access info for road point", point, "vehicle:", vehicleType, + "access is:", emplaceRes.first->second, "tried:", roadAccessType)); + } + }; + string line; for (uint32_t lineNo = 1;; ++lineNo) { @@ -128,7 +161,7 @@ bool ParseRoadAccess(string const & roadAccessPath, map const if (!iter) { - LOG(LERROR, ("Error when parsing road access: no road access type", lineNo)); + LOG(LERROR, ("Error when parsing road access: no road access type at line", lineNo, "Line contents:", line)); return false; } RoadAccess::Type roadAccessType; @@ -138,7 +171,15 @@ bool ParseRoadAccess(string const & roadAccessPath, map const uint64_t osmId; if (!iter || !strings::to_uint64(*iter, osmId)) { - LOG(LERROR, ("Error when parsing road access: bad osm id at line", lineNo)); + LOG(LERROR, ("Error when parsing road access: bad osm id at line", lineNo, "Line contents:", line)); + return false; + } + ++iter; + + uint32_t pointIdx; + if (!iter || !strings::to_uint(*iter, pointIdx)) + { + LOG(LERROR, ("Error when parsing road access: bad pointIdx at line", lineNo, "Line contents:", line)); return false; } ++iter; @@ -152,13 +193,14 @@ bool ParseRoadAccess(string const & roadAccessPath, map const uint32_t const featureId = it->second; - addSegment(Segment(kFakeNumMwmId, featureId, 0 /* wildcard segment idx */, - true /* wildcard isForward */), - vehicleType, roadAccessType, osmId); + if (pointIdx == 0) + addFeature(featureId, vehicleType, roadAccessType, osmId); + else + addPoint(RoadPoint(featureId, pointIdx - 1), vehicleType, roadAccessType); } for (size_t i = 0; i < static_cast(VehicleType::Count); ++i) - roadAccessByVehicleType[i].SetSegmentTypes(move(segmentType[i])); + roadAccessByVehicleType[i].SetAccessTypes(move(featureType[i]), move(pointType[i])); return true; } @@ -192,6 +234,8 @@ RoadAccessTagProcessor::RoadAccessTagProcessor(VehicleType vehicleType) m_tagMappings.push_back(&kMotorVehicleTagMapping); m_tagMappings.push_back(&kVehicleTagMapping); m_tagMappings.push_back(&kDefaultTagMapping); + // Apply barrier tags if we have no {vehicle = ...}, {access = ...} etc. + m_tagMappings.push_back(&kCarBarriersTagMapping); break; case VehicleType::Pedestrian: m_tagMappings.push_back(&kPedestrianTagMapping); @@ -201,6 +245,8 @@ RoadAccessTagProcessor::RoadAccessTagProcessor(VehicleType vehicleType) m_tagMappings.push_back(&kBicycleTagMapping); m_tagMappings.push_back(&kVehicleTagMapping); m_tagMappings.push_back(&kDefaultTagMapping); + // Apply barrier tags if we have no {bicycle = ...}, {access = ...} etc. + m_tagMappings.push_back(&kBicycleBarriersTagMapping); break; case VehicleType::Transit: // Use kTransitTagMapping to keep transit section empty. We'll use pedestrian section for @@ -213,22 +259,48 @@ RoadAccessTagProcessor::RoadAccessTagProcessor(VehicleType vehicleType) } } -void RoadAccessTagProcessor::Process(OsmElement const & elem, ofstream & oss) const +void RoadAccessTagProcessor::Process(OsmElement const & elem, ofstream & oss) { - // todo(@m) Add support for non-way elements, such as barrier=gate. + // We will proccess all nodes before ways because of o5m format: + // all nodes are first, then all ways, then all relations. + if (elem.type == OsmElement::EntityType::Node) + { + RoadAccess::Type accessType = GetAccessType(elem); + if (accessType != RoadAccess::Type::Yes) + m_barriers[elem.id] = accessType; + return; + } + if (elem.type != OsmElement::EntityType::Way) return; + // All feature tags. + auto const accessType = GetAccessType(elem); + if (accessType != RoadAccess::Type::Yes) + oss << ToString(m_vehicleType) << " " << ToString(accessType) << " " << elem.id << 0 << endl; + + // Barrier tags. + for (size_t pointIdx = 0; pointIdx < elem.m_nds.size(); ++pointIdx) + { + auto const it = m_barriers.find(elem.m_nds[pointIdx]); + if (it == m_barriers.cend()) + continue; + + // idx == 0 used as wildcard segment Idx, for nodes we store |pointIdx + 1| instead of |pointIdx|. + oss << ToString(m_vehicleType) << " " << ToString(it->second) << " " << elem.id << " " + << pointIdx + 1; + } +} + +RoadAccess::Type RoadAccessTagProcessor::GetAccessType(OsmElement const & elem) const +{ for (auto const tagMapping : m_tagMappings) { auto const accessType = GetAccessTypeFromMapping(elem, tagMapping); if (accessType != RoadAccess::Type::Count) - { - if (accessType != RoadAccess::Type::Yes) - oss << ToString(m_vehicleType) << " " << ToString(accessType) << " " << elem.id << endl; - return; - } + return accessType; } + return RoadAccess::Type::Yes; } // RoadAccessWriter ------------------------------------------------------------ @@ -256,7 +328,7 @@ void RoadAccessWriter::Process(OsmElement const & elem) return; } - for (auto const & p : m_tagProcessors) + for (auto & p : m_tagProcessors) p.Process(elem, m_stream); } diff --git a/generator/road_access_generator.hpp b/generator/road_access_generator.hpp index 0b419ae780..5ebf448758 100644 --- a/generator/road_access_generator.hpp +++ b/generator/road_access_generator.hpp @@ -29,13 +29,17 @@ public: explicit RoadAccessTagProcessor(VehicleType vehicleType); - void Process(OsmElement const & elem, std::ofstream & oss) const; + void Process(OsmElement const & elem, std::ofstream & oss); private: + RoadAccess::Type GetAccessType(OsmElement const & elem) const; + VehicleType m_vehicleType; // Order of tag mappings in m_tagMappings is from more to less specific. // e.g. for car: motorcar, motorvehicle, vehicle, general access tags. std::vector m_tagMappings; + // Tag mapping for barriers. Key is barrier node osm id. + std::map m_barriers; }; class RoadAccessWriter diff --git a/routing/index_graph.cpp b/routing/index_graph.cpp index 658bceb0c3..4ab2291f92 100644 --- a/routing/index_graph.cpp +++ b/routing/index_graph.cpp @@ -160,7 +160,7 @@ void IndexGraph::GetNeighboringEdge(Segment const & from, Segment const & to, bo if (IsRestricted(m_restrictions, from, to, isOutgoing)) return; - if (m_roadAccess.GetSegmentType(to) == RoadAccess::Type::No) + if (m_roadAccess.GetFeatureType(to.GetFeatureId()) == RoadAccess::Type::No) return; RouteWeight const weight = CalcSegmentWeight(isOutgoing ? to : from) + @@ -177,8 +177,8 @@ RouteWeight IndexGraph::GetPenalties(Segment const & u, Segment const & v) int32_t const passThroughPenalty = fromPassThroughAllowed == toPassThroughAllowed ? 0 : 1; // We do not distinguish between RoadAccess::Type::Private and RoadAccess::Type::Destination for now. - bool const fromAccessAllowed = m_roadAccess.GetSegmentType(u) == RoadAccess::Type::Yes; - bool const toAccessAllowed = m_roadAccess.GetSegmentType(v) == RoadAccess::Type::Yes; + bool const fromAccessAllowed = m_roadAccess.GetFeatureType(u.GetFeatureId()) == RoadAccess::Type::Yes; + bool const toAccessAllowed = m_roadAccess.GetFeatureType(v.GetFeatureId()) == RoadAccess::Type::Yes; // Route crosses border of access=yes/access={private, destination} area if |u| and |v| have different // access restrictions. int32_t const accessPenalty = fromAccessAllowed == toAccessAllowed ? 0 : 1; diff --git a/routing/index_graph.hpp b/routing/index_graph.hpp index a448d1659e..a20861de94 100644 --- a/routing/index_graph.hpp +++ b/routing/index_graph.hpp @@ -42,7 +42,7 @@ public: RoadAccess::Type GetAccessType(Segment const & segment) const { - return m_roadAccess.GetSegmentType(segment); + return m_roadAccess.GetFeatureType(segment.GetFeatureId()); } uint32_t GetNumRoads() const { return m_roadIndex.GetSize(); } diff --git a/routing/road_access.cpp b/routing/road_access.cpp index f2d4a8ea38..cf5784c6b0 100644 --- a/routing/road_access.cpp +++ b/routing/road_access.cpp @@ -10,38 +10,51 @@ using namespace std; namespace { string const kNames[] = {"No", "Private", "Destination", "Yes", "Count"}; + +template +void PrintKV(ostringstream & oss, KV const & kvs, size_t maxKVToShow) +{ + size_t i = 0; + for (auto const & kv : kvs) + { + if (i > 0) + oss << ", "; + oss << DebugPrint(kv.first) << " " << DebugPrint(kv.second); + ++i; + if (i == maxKVToShow) + break; + } + if (kvs.size() > maxKVToShow) + oss << ", ..."; +} } // namespace namespace routing { // RoadAccess -------------------------------------------------------------------------------------- -RoadAccess::Type RoadAccess::GetSegmentType(Segment const & segment) const +RoadAccess::Type RoadAccess::GetFeatureType(uint32_t featureId) const { - // todo(@m) This may or may not be too slow. Consider profiling this and using - // a Bloom filter or anything else that is faster than std::map. +// todo(@m) This may or may not be too slow. Consider profiling this and using +// a Bloom filter or anything else that is faster than std::map. + auto const it = m_featureTypes.find(featureId); + if (it != m_featureTypes.cend()) + return it->second; - { - Segment key(kFakeNumMwmId, segment.GetFeatureId(), 0 /* wildcard segment idx */, - true /* wildcard isForward */); - auto const it = m_segmentTypes.find(key); - if (it != m_segmentTypes.end()) - return it->second; - } + return RoadAccess::Type::Yes; +} - { - Segment key(kFakeNumMwmId, segment.GetFeatureId(), segment.GetSegmentIdx() + 1, - segment.IsForward()); - auto const it = m_segmentTypes.find(key); - if (it != m_segmentTypes.end()) - return it->second; - } +RoadAccess::Type RoadAccess::GetPointType(RoadPoint const & point) const +{ + auto const it = m_pointTypes.find(point); + if (it != m_pointTypes.cend()) + return it->second; return RoadAccess::Type::Yes; } bool RoadAccess::operator==(RoadAccess const & rhs) const { - return m_segmentTypes == rhs.m_segmentTypes; + return m_featureTypes == rhs.m_featureTypes && m_pointTypes == rhs.m_pointTypes; } // Functions --------------------------------------------------------------------------------------- @@ -73,21 +86,11 @@ string DebugPrint(RoadAccess const & r) { size_t const kMaxIdsToShow = 10; ostringstream oss; - oss << "RoadAccess ["; - size_t id = 0; - for (auto const & kv : r.GetSegmentTypes()) - { - if (id > 0) - oss << ", "; - oss << DebugPrint(kv.first) << " " << DebugPrint(kv.second); - ++id; - if (id == kMaxIdsToShow) - break; - } - if (r.GetSegmentTypes().size() > kMaxIdsToShow) - oss << ", ..."; - - oss << "]"; + oss << "RoadAccess { FeatureTypes ["; + PrintKV(oss, r.GetFeatureTypes(), kMaxIdsToShow); + oss << "], PointTypes ["; + PrintKV(oss, r.GetPointTypes(), kMaxIdsToShow); + oss << "] }"; return oss.str(); } } // namespace routing diff --git a/routing/road_access.hpp b/routing/road_access.hpp index 4c6e3911fc..f7f42a7fc0 100644 --- a/routing/road_access.hpp +++ b/routing/road_access.hpp @@ -8,6 +8,7 @@ #include #include #include +#include #include namespace routing @@ -40,14 +41,17 @@ public: Count }; - std::map const & GetSegmentTypes() const { return m_segmentTypes; } + std::map const & GetFeatureTypes() const { return m_featureTypes; } + std::map const & GetPointTypes() const { return m_pointTypes; } - Type GetSegmentType(Segment const & segment) const; + Type GetFeatureType(uint32_t featureId) const; + Type GetPointType(RoadPoint const & point) const; - template - void SetSegmentTypes(V && v) + template + void SetAccessTypes(MF && mf, MP && mp) { - m_segmentTypes = std::forward(v); + m_featureTypes = std::forward(mf); + m_pointTypes = std::forward(mp); } void Clear(); @@ -56,14 +60,18 @@ public: bool operator==(RoadAccess const & rhs) const; + template + void SetFeatureTypesForTests(MF && mf) + { + m_featureTypes = std::forward(mf); + } + private: - // todo(@m) Segment's NumMwmId is not used here. Decouple it from - // segment and use only (fid, idx, forward) in the map. - // // If segmentIdx of a key in this map is 0, it means the // entire feature has the corresponding access type. // Otherwise, the information is about the segment with number (segmentIdx-1). - std::map m_segmentTypes; + std::map m_featureTypes; + std::map m_pointTypes; }; std::string ToString(RoadAccess::Type type); diff --git a/routing/road_access_serialization.hpp b/routing/road_access_serialization.hpp index 0364892035..35fe524b29 100644 --- a/routing/road_access_serialization.hpp +++ b/routing/road_access_serialization.hpp @@ -2,6 +2,7 @@ #include "routing/coding.hpp" #include "routing/road_access.hpp" +#include "routing/road_point.hpp" #include "routing/segment.hpp" #include "routing/vehicle_mask.hpp" @@ -19,6 +20,7 @@ #include #include #include +#include #include namespace routing @@ -26,7 +28,8 @@ namespace routing class RoadAccessSerializer final { public: - using RoadAccessTypesMap = std::map; + using RoadAccessTypesFeatureMap = std::map; + using RoadAccessTypesPointMap = std::map; using RoadAccessByVehicleType = std::array(VehicleType::Count)>; RoadAccessSerializer() = delete; @@ -48,7 +51,8 @@ public: for (size_t i = 0; i < static_cast(VehicleType::Count); ++i) { auto const pos = sink.Pos(); - SerializeOneVehicleType(sink, roadAccessByType[i].GetSegmentTypes()); + SerializeOneVehicleType(sink, roadAccessByType[i].GetFeatureTypes(), + roadAccessByType[i].GetPointTypes()); sectionSizes[i] = base::checked_cast(sink.Pos() - pos); } @@ -91,21 +95,32 @@ public: continue; } - RoadAccessTypesMap m; - DeserializeOneVehicleType(src, m); + RoadAccessTypesFeatureMap mf; + RoadAccessTypesPointMap mp; + DeserializeOneVehicleType(src, mf, mp); - roadAccess.SetSegmentTypes(std::move(m)); + roadAccess.SetAccessTypes(std::move(mf), std::move(mp)); } } private: template - static void SerializeOneVehicleType(Sink & sink, RoadAccessTypesMap const & m) + static void SerializeOneVehicleType(Sink & sink, RoadAccessTypesFeatureMap const & mf, + RoadAccessTypesPointMap const & mp) { std::array, static_cast(RoadAccess::Type::Count)> segmentsByRoadAccessType; - for (auto const & kv : m) - segmentsByRoadAccessType[static_cast(kv.second)].push_back(kv.first); + for (auto const & kv : mf) + { + segmentsByRoadAccessType[static_cast(kv.second)].push_back( + Segment(kFakeNumMwmId, kv.first, 0 /* wildcard segmentIdx */, true /* widcard forward */)); + } + // For nodes we store |pointId + 1| because 0 is reserved for wildcard segmentIdx. + for (auto const & kv : mp) + { + segmentsByRoadAccessType[static_cast(kv.second)].push_back( + Segment(kFakeNumMwmId, kv.first.GetFeatureId(), kv.first.GetPointId() + 1, true)); + } for (auto & segs : segmentsByRoadAccessType) { @@ -115,15 +130,34 @@ private: } template - static void DeserializeOneVehicleType(Source & src, RoadAccessTypesMap & m) + static void DeserializeOneVehicleType(Source & src, RoadAccessTypesFeatureMap & mf, + RoadAccessTypesPointMap & mp) { - m.clear(); + mf.clear(); + mp.clear(); for (size_t i = 0; i < static_cast(RoadAccess::Type::Count); ++i) { + // An earlier version allowed blocking any segment of a feature (or the entire feature + // by providing a wildcard segment index). + // At present, we either block the feature entirely or block any of its road points. The + // the serialization code remains the same, although its semantics changes as we now + // work with point indices instead of segment indices. std::vector segs; DeserializeSegments(src, segs); for (auto const & seg : segs) - m[seg] = static_cast(i); + { + if (seg.GetSegmentIdx() == 0) + { + // Wildcard segmentIdx. + mf[seg.GetFeatureId()] = static_cast(i); + } + else + { + // For nodes we store |pointId + 1| because 0 is reserved for wildcard segmentIdx. + mp[RoadPoint(seg.GetFeatureId(), seg.GetSegmentIdx() - 1)] = + static_cast(i); + } + } } } diff --git a/routing/road_point.hpp b/routing/road_point.hpp index 7127eddc6f..ba199c0656 100644 --- a/routing/road_point.hpp +++ b/routing/road_point.hpp @@ -21,6 +21,13 @@ public: uint32_t GetPointId() const { return m_pointId; } + bool operator<(RoadPoint const & rp) const + { + if (m_featureId != rp.m_featureId) + return m_featureId < rp.m_featureId; + return m_pointId < rp.m_pointId; + } + bool operator==(RoadPoint const & rp) const { return m_featureId == rp.m_featureId && m_pointId == rp.m_pointId; diff --git a/routing/routing_tests/index_graph_tools.cpp b/routing/routing_tests/index_graph_tools.cpp index 06ba3b1ef0..ded2ab2290 100644 --- a/routing/routing_tests/index_graph_tools.cpp +++ b/routing/routing_tests/index_graph_tools.cpp @@ -254,18 +254,17 @@ void TestIndexGraphTopology::Builder::BuildJoints() void TestIndexGraphTopology::Builder::BuildGraphFromRequests(vector const & requests) { - map segmentTypes; + map featureTypes; for (auto const & request : requests) { BuildSegmentFromEdge(request); if (request.m_accessType != RoadAccess::Type::Yes) { - segmentTypes[Segment(kFakeNumMwmId, request.m_id, 0 /* wildcard segmentIdx */, true)] = - request.m_accessType; + featureTypes[request.m_id] = request.m_accessType; } } - m_roadAccess.SetSegmentTypes(move(segmentTypes)); + m_roadAccess.SetFeatureTypesForTests(move(featureTypes)); } void TestIndexGraphTopology::Builder::BuildSegmentFromEdge(EdgeRequest const & request) diff --git a/routing/routing_tests/road_access_test.cpp b/routing/routing_tests/road_access_test.cpp index 41a1407f27..4e7dd613d9 100644 --- a/routing/routing_tests/road_access_test.cpp +++ b/routing/routing_tests/road_access_test.cpp @@ -23,21 +23,21 @@ namespace UNIT_TEST(RoadAccess_Serialization) { // Segment is (numMwmId, featureId, segmentIdx, isForward). - map const m0 = { - {Segment(kFakeNumMwmId, 1, 0, false), RoadAccess::Type::No}, - {Segment(kFakeNumMwmId, 2, 2, false), RoadAccess::Type::Private}, + map const m0 = { + {1, RoadAccess::Type::No}, + {2, RoadAccess::Type::Private}, }; - map const m1 = { - {Segment(kFakeNumMwmId, 1, 1, false), RoadAccess::Type::Private}, - {Segment(kFakeNumMwmId, 2, 0, true), RoadAccess::Type::Destination}, + map const m1 = { + {1, RoadAccess::Type::Private}, + {2, RoadAccess::Type::Destination}, }; RoadAccess roadAccessCar; - roadAccessCar.SetSegmentTypes(m0); + roadAccessCar.SetFeatureTypesForTests(m0); RoadAccess roadAccessPedestrian; - roadAccessPedestrian.SetSegmentTypes(m1); + roadAccessPedestrian.SetFeatureTypesForTests(m1); RoadAccessSerializer::RoadAccessByVehicleType roadAccessAllTypes; roadAccessAllTypes[static_cast(VehicleType::Car)] = roadAccessCar;