diff --git a/generator/restriction_writer.cpp b/generator/restriction_writer.cpp index 4d469394f5..18840b0101 100644 --- a/generator/restriction_writer.cpp +++ b/generator/restriction_writer.cpp @@ -201,7 +201,7 @@ std::string DebugPrint(RestrictionWriter::ViaType const & type) { case RestrictionWriter::ViaType::Node: return RestrictionWriter::kNodeString; case RestrictionWriter::ViaType::Way: return RestrictionWriter::kWayString; - case RestrictionWriter::ViaType::Max: CHECK(false, ()); + case RestrictionWriter::ViaType::Max: UNREACHABLE(); } UNREACHABLE(); } diff --git a/routing/index_graph.cpp b/routing/index_graph.cpp index 1f74fcc8fa..80fc84f54b 100644 --- a/routing/index_graph.cpp +++ b/routing/index_graph.cpp @@ -1,3 +1,5 @@ +#include + #include "routing/index_graph.hpp" #include "routing/restrictions_serialization.hpp" @@ -9,6 +11,7 @@ #include "base/assert.hpp" #include "base/checked_cast.hpp" #include "base/exception.hpp" +#include "base/timer.hpp" #include #include @@ -20,20 +23,14 @@ using namespace base; using namespace routing; using namespace std; -bool IsUTurn(Segment const & u, Segment const & v) -{ - return u.GetFeatureId() == v.GetFeatureId() && u.GetSegmentIdx() == v.GetSegmentIdx() && - u.IsForward() != v.IsForward(); -} - -bool IsRestricted(RestrictionVec const & restrictions, Segment const & u, Segment const & v, +/*bool IsRestricted(std::vector> const & restrictions, Segment const & u, Segment const & v, bool isOutgoing) { uint32_t const featureIdFrom = isOutgoing ? u.GetFeatureId() : v.GetFeatureId(); uint32_t const featureIdTo = isOutgoing ? v.GetFeatureId() : u.GetFeatureId(); if (!binary_search(restrictions.cbegin(), restrictions.cend(), - Restriction(Restriction::Type::No, {featureIdFrom, featureIdTo}))) + std::vector({featureIdFrom, featureIdTo}))) { return false; } @@ -42,29 +39,36 @@ bool IsRestricted(RestrictionVec const & restrictions, Segment const & u, Segmen return true; // @TODO(bykoianko) According to current code if a feature id is marked as a feature with - // restrictricted U-turn it's restricted to make a U-turn on the both ends of the feature. - // Generally speaking it's wrong. In osm there's information about the end of the feature - // where the U-turn is restricted. It's necessary to pass the data to mwm and to use it here. - // Please see test LineGraph_RestrictionF1F1No for details. - // - // Another example when it's necessary to be aware about feature end participated in restriction - // is - // *---F1---* - // | | - // *--F3--A B--F4--* - // | | - // *---F2---* - // In case of restriction F1-A-F2 or F1-B-F2 of any type (No, Only) the important information - // is lost. + // restrictricted U-turn it's restricted to make a U-turn on the both ends of the feature. + // Generally speaking it's wrong. In osm there's information about the end of the feature + // where the U-turn is restricted. It's necessary to pass the data to mwm and to use it here. + // Please see test LineGraph_RestrictionF1F1No for details. + // Another example when it's necessary to be aware about feature end participated in restriction + // is + // *---F1---* + // | | + // *--F3--A B--F4--* + // | | + // *---F2---* + // In case of restriction F1-A-F2 or F1-B-F2 of any type (No, Only) the important information + // is lost. return IsUTurn(u, v); -} +}*/ } // namespace namespace routing { +bool IsUTurn(Segment const & u, Segment const & v) +{ + return u.GetFeatureId() == v.GetFeatureId() && u.GetSegmentIdx() == v.GetSegmentIdx() && + u.IsForward() != v.IsForward(); +} + +std::map IndexGraph::kEmptyParentsSegments = {}; + IndexGraph::IndexGraph(shared_ptr geometry, shared_ptr estimator, RoutingOptions routingOptions) - : m_geometry(geometry), + : m_geometry(std::move(geometry)), m_estimator(move(estimator)), m_avoidRoutingOptions(routingOptions) { @@ -77,7 +81,8 @@ bool IndexGraph::IsJoint(RoadPoint const & roadPoint) const return m_roadIndex.GetJointId(roadPoint) != Joint::kInvalidId; } -void IndexGraph::GetEdgeList(Segment const & segment, bool isOutgoing, vector & edges) +void IndexGraph::GetEdgeList(Segment const & segment, bool isOutgoing, vector & edges, + std::map & parents) { RoadPoint const roadPoint = segment.GetRoadPoint(isOutgoing); Joint::Id const jointId = m_roadIndex.GetJointId(roadPoint); @@ -85,12 +90,12 @@ void IndexGraph::GetEdgeList(Segment const & segment, bool isOutgoing, vector const & children, } } -void IndexGraph::GetEdgeList(Segment const & parent, bool isOutgoing, vector & edges, - vector & parentWeights) +void IndexGraph::GetEdgeList(JointSegment const & parentJoint, + Segment const & parent, bool isOutgoing, vector & edges, + vector & parentWeights, std::map & parents) { vector possibleChildren; GetSegmentCandidateForJoint(parent, isOutgoing, possibleChildren); @@ -144,8 +150,9 @@ void IndexGraph::GetEdgeList(Segment const & parent, bool isOutgoing, vector lastPoints; GetLastPointsForJoint(possibleChildren, isOutgoing, lastPoints); - ReconstructJointSegment(parent, possibleChildren, lastPoints, - isOutgoing, edges, parentWeights); + + ReconstructJointSegment(parentJoint, parent, possibleChildren, lastPoints, + isOutgoing, edges, parentWeights, parents); } boost::optional @@ -157,8 +164,9 @@ IndexGraph::GetJointEdgeByLastPoint(Segment const & parent, Segment const & firs vector edges; vector parentWeights; - ReconstructJointSegment(parent, possibleChilds, lastPoints, - isOutgoing, edges, parentWeights); + std::map emptyParents; + ReconstructJointSegment({}, parent, possibleChilds, lastPoints, + isOutgoing, edges, parentWeights, emptyParents); CHECK_LESS_OR_EQUAL(edges.size(), 1, ()); if (edges.size() == 1) @@ -182,7 +190,22 @@ void IndexGraph::Import(vector const & joints) void IndexGraph::SetRestrictions(RestrictionVec && restrictions) { ASSERT(is_sorted(restrictions.cbegin(), restrictions.cend()), ()); - m_restrictions = move(restrictions); + + m_restrictionsForward.clear(); + m_restrictionsBackward.clear(); + + base::HighResTimer timer; + for (auto const & restriction : restrictions) + { + ASSERT(!restriction.empty(), ()); + auto & forward = m_restrictionsForward[restriction.back()]; + forward.emplace_back(restriction.begin(), std::prev(restriction.end())); + std::reverse(forward.back().begin(), forward.back().end()); + + m_restrictionsBackward[restriction.front()].emplace_back(std::next(restriction.begin()), restriction.end()); + } + + LOG(LDEBUG, ("Restrictions are loaded in:", timer.ElapsedNano() / 1e6, "ms")); } void IndexGraph::SetRoadAccess(RoadAccess && roadAccess) { m_roadAccess = move(roadAccess); } @@ -206,7 +229,7 @@ RouteWeight IndexGraph::CalcSegmentWeight(Segment const & segment) } void IndexGraph::GetNeighboringEdges(Segment const & from, RoadPoint const & rp, bool isOutgoing, - vector & edges) + vector & edges, std::map & parents) { RoadGeometry const & road = m_geometry->GetRoad(rp.GetFeatureId()); @@ -222,14 +245,14 @@ void IndexGraph::GetNeighboringEdges(Segment const & from, RoadPoint const & rp, { GetNeighboringEdge(from, Segment(from.GetMwmId(), rp.GetFeatureId(), rp.GetPointId(), isOutgoing), - isOutgoing, edges); + isOutgoing, edges, parents); } if ((!isOutgoing || bidirectional) && rp.GetPointId() > 0) { GetNeighboringEdge( from, Segment(from.GetMwmId(), rp.GetFeatureId(), rp.GetPointId() - 1, !isOutgoing), - isOutgoing, edges); + isOutgoing, edges, parents); } } @@ -270,12 +293,14 @@ void IndexGraph::GetSegmentCandidateForJoint(Segment const & parent, bool isOutg /// \param |parentWeights| - see |IndexGraphStarterJoints::GetEdgeList| method about this argument. /// Shotly - in case of |isOutgoing| == false, method saves here the weights /// from parent to firstChildren. -void IndexGraph::ReconstructJointSegment(Segment const & parent, +void IndexGraph::ReconstructJointSegment(JointSegment const & parentJoint, + Segment const & parent, vector const & firstChildren, vector const & lastPointIds, bool isOutgoing, vector & jointEdges, - vector & parentWeights) + vector & parentWeights, + std::map & parents) { CHECK_EQUAL(firstChildren.size(), lastPointIds.size(), ()); @@ -308,7 +333,8 @@ void IndexGraph::ReconstructJointSegment(Segment const & parent, } if (parent.GetFeatureId() != firstChild.GetFeatureId() && - IsRestricted(m_restrictions, parent, firstChild, isOutgoing)) + IsRestricted(parentJoint, parent.GetFeatureId(), firstChild.GetFeatureId(), + isOutgoing, parents)) { continue; } @@ -360,7 +386,7 @@ void IndexGraph::ReconstructJointSegment(Segment const & parent, } void IndexGraph::GetNeighboringEdge(Segment const & from, Segment const & to, bool isOutgoing, - vector & edges) + vector & edges, std::map & parents) { // Blocking U-turns on internal feature points. RoadPoint const rp = from.GetRoadPoint(isOutgoing); @@ -370,8 +396,11 @@ void IndexGraph::GetNeighboringEdge(Segment const & from, Segment const & to, bo return; } - if (IsRestricted(m_restrictions, from, to, isOutgoing)) + if (from.GetFeatureId() != to.GetFeatureId() && + IsRestricted(from, from.GetFeatureId(), to.GetFeatureId(), isOutgoing, parents)) + { return; + } if (m_roadAccess.GetFeatureType(to.GetFeatureId()) == RoadAccess::Type::No) return; diff --git a/routing/index_graph.hpp b/routing/index_graph.hpp index 1707055f9e..a5aef48fd0 100644 --- a/routing/index_graph.hpp +++ b/routing/index_graph.hpp @@ -19,6 +19,9 @@ #include #include +#include +#include + #include namespace routing diff --git a/routing/restriction_loader.cpp b/routing/restriction_loader.cpp index 6ed6acbc7e..1543dcd501 100644 --- a/routing/restriction_loader.cpp +++ b/routing/restriction_loader.cpp @@ -7,14 +7,15 @@ #include "base/stl_helpers.hpp" +#include +#include +#include +#include #include -using namespace std; - namespace { using namespace routing; - /// \returns if features |r1| and |r2| have a common end returns its joint id. /// If not, returns Joint::kInvalidId. /// \note It's possible that the both ends of |r1| and |r2| have common joint ids. @@ -57,6 +58,7 @@ RestrictionLoader::RestrictionLoader(MwmValue const & mwmValue, IndexGraph const RestrictionVec restrictionsOnly; RestrictionSerializer::Deserialize(m_header, m_restrictions /* restriction No */, restrictionsOnly, src); + ConvertRestrictionsOnlyToNoAndSort(graph, restrictionsOnly, m_restrictions); } catch (Reader::OpenException const & e) @@ -68,34 +70,56 @@ RestrictionLoader::RestrictionLoader(MwmValue const & mwmValue, IndexGraph const } } +bool RestrictionLoader::HasRestrictions() const +{ + return !m_restrictions.empty(); +} + +RestrictionVec && RestrictionLoader::StealRestrictions() +{ + return std::move(m_restrictions); +} + void ConvertRestrictionsOnlyToNoAndSort(IndexGraph const & graph, - RestrictionVec const & restrictionsOnly, + RestrictionVec & restrictionsOnly, RestrictionVec & restrictionsNo) { - for (Restriction const & o : restrictionsOnly) + for (auto & restriction : restrictionsOnly) { - if (o.m_featureIds.size() != 2) + if (std::any_of(restriction.begin(), restriction.end(), + [&graph](auto const & item) + { + return !graph.IsRoad(item); + })) + { continue; + } - if (!graph.IsRoad(o.m_featureIds[0]) || !graph.IsRoad(o.m_featureIds[1])) - continue; + CHECK_GREATER_OR_EQUAL(restriction.size(), 2, ()); + auto const lastFeatureId = *(restriction.end() - 1); + auto const prevLastFeatureId = *(restriction.end() - 2); // Looking for a joint of an intersection of |o| features. Joint::Id const common = - GetCommonEndJoint(graph.GetRoad(o.m_featureIds[0]), graph.GetRoad(o.m_featureIds[1])); + GetCommonEndJoint(graph.GetRoad(prevLastFeatureId), graph.GetRoad(lastFeatureId)); + if (common == Joint::kInvalidId) continue; + // Delete last feature id ("to" member). + restriction.pop_back(); // Adding restriction of type No for all features of joint |common| except for // the second feature of restriction |o|. graph.ForEachPoint(common, [&](RoadPoint const & rp) { - if (rp.GetFeatureId() != o.m_featureIds[1 /* to */]) + if (rp.GetFeatureId() != lastFeatureId) { - restrictionsNo.emplace_back(Restriction::Type::No, - vector({o.m_featureIds[0 /* from */], rp.GetFeatureId()})); + std::vector result(restriction); + result.emplace_back(rp.GetFeatureId()); + restrictionsNo.emplace_back(std::move(result)); } }); } + base::SortUnique(restrictionsNo); } } // namespace routing diff --git a/routing/restriction_loader.hpp b/routing/restriction_loader.hpp index 86513d3d61..1a9d2cfa26 100644 --- a/routing/restriction_loader.hpp +++ b/routing/restriction_loader.hpp @@ -17,8 +17,8 @@ class RestrictionLoader public: explicit RestrictionLoader(MwmValue const & mwmValue, IndexGraph const & graph); - bool HasRestrictions() const { return !m_restrictions.empty(); } - RestrictionVec && StealRestrictions() { return move(m_restrictions); } + bool HasRestrictions() const; + RestrictionVec && StealRestrictions(); private: std::unique_ptr m_reader; @@ -28,6 +28,6 @@ private: }; void ConvertRestrictionsOnlyToNoAndSort(IndexGraph const & graph, - RestrictionVec const & restrictionsOnly, + RestrictionVec & restrictionsOnly, RestrictionVec & restrictionsNo); } // namespace routing diff --git a/routing/restrictions_serialization.cpp b/routing/restrictions_serialization.cpp index 142cd296c6..d3738e4287 100644 --- a/routing/restrictions_serialization.cpp +++ b/routing/restrictions_serialization.cpp @@ -15,12 +15,6 @@ namespace routing // static uint32_t const Restriction::kInvalidFeatureId = numeric_limits::max(); -bool Restriction::IsValid() const -{ - return !m_featureIds.empty() && - find(begin(m_featureIds), end(m_featureIds), kInvalidFeatureId) == end(m_featureIds); -} - bool Restriction::operator==(Restriction const & restriction) const { return m_featureIds == restriction.m_featureIds && m_type == restriction.m_type; @@ -33,7 +27,7 @@ bool Restriction::operator<(Restriction const & restriction) const return m_featureIds < restriction.m_featureIds; } -string ToString(Restriction::Type const & type) +string DebugPrint(Restriction::Type const & type) { switch (type) { @@ -43,13 +37,17 @@ string ToString(Restriction::Type const & type) return "Unknown"; } -string DebugPrint(Restriction::Type const & type) { return ToString(type); } - string DebugPrint(Restriction const & restriction) { ostringstream out; - out << "m_featureIds:[" << ::DebugPrint(restriction.m_featureIds) - << "] m_type:" << DebugPrint(restriction.m_type) << " "; + out << "[" << DebugPrint(restriction.m_type) << "]: {"; + for (size_t i = 0; i < restriction.m_featureIds.size(); ++i) + { + out << restriction.m_featureIds[i]; + if (i + 1 != restriction.m_featureIds.size()) + out << ", "; + } + out << "}"; return out.str(); } } // namespace routing diff --git a/routing/restrictions_serialization.hpp b/routing/restrictions_serialization.hpp index 59f5df89a1..b983d89a98 100644 --- a/routing/restrictions_serialization.hpp +++ b/routing/restrictions_serialization.hpp @@ -13,7 +13,9 @@ #include #include +#include #include +#include #include namespace routing @@ -38,7 +40,6 @@ struct Restriction }; Restriction(Type type, std::vector const & links) : m_featureIds(links), m_type(type) {} - bool IsValid() const; bool operator==(Restriction const & restriction) const; bool operator<(Restriction const & restriction) const; @@ -47,9 +48,9 @@ struct Restriction Type m_type; }; -using RestrictionVec = std::vector; +// Each std::vector contains feature ids of a restriction of type only or no. +using RestrictionVec = std::vector>; -std::string ToString(Restriction::Type const & type); std::string DebugPrint(Restriction::Type const & type); std::string DebugPrint(Restriction const & restriction); @@ -96,8 +97,8 @@ class RestrictionSerializer public: template static void Serialize(RestrictionHeader const & header, - RestrictionVec::const_iterator begin, - RestrictionVec::const_iterator end, Sink & sink) + std::vector::const_iterator begin, + std::vector::const_iterator end, Sink & sink) { auto const firstOnlyIt = begin + header.m_noRestrictionCount; SerializeSingleType(begin, firstOnlyIt, sink); @@ -109,10 +110,8 @@ public: RestrictionVec & restrictionsNo, RestrictionVec & restrictionsOnly, Source & src) { - DeserializeSingleType(Restriction::Type::No, header.m_noRestrictionCount, - restrictionsNo, src); - DeserializeSingleType(Restriction::Type::Only, header.m_onlyRestrictionCount, - restrictionsOnly, src); + DeserializeSingleType(header.m_noRestrictionCount, restrictionsNo, src); + DeserializeSingleType(header.m_onlyRestrictionCount, restrictionsOnly, src); } private: @@ -123,8 +122,8 @@ private: /// \param end is an iterator to the element after the last element to serialize. /// \note All restrictions should have the same type. template - static void SerializeSingleType(RestrictionVec::const_iterator begin, - RestrictionVec::const_iterator end, Sink & sink) + static void SerializeSingleType(std::vector::const_iterator begin, + std::vector::const_iterator end, Sink & sink) { if (begin == end) return; @@ -138,8 +137,7 @@ private: { CHECK_EQUAL(type, begin->m_type, ()); - routing::Restriction const & restriction = *begin; - CHECK(restriction.IsValid(), ()); + Restriction const & restriction = *begin; CHECK_LESS(1, restriction.m_featureIds.size(), ("No sense in zero or one link restrictions.")); @@ -161,8 +159,7 @@ private: } template - static bool DeserializeSingleType(Restriction::Type type, uint32_t count, - RestrictionVec & restrictions, Source & src) + static bool DeserializeSingleType(uint32_t count, RestrictionVec & restrictions, Source & src) { uint32_t prevFirstLinkFeatureId = 0; BitReader bits(src); @@ -176,16 +173,15 @@ private: } size_t const numLinks = biasedLinkNumber + 1 /* number of link is two or more */; - Restriction restriction(type, {} /* links */); - restriction.m_featureIds.resize(numLinks); + std::vector restriction(numLinks); auto const biasedFirstFeatureId = coding::DeltaCoder::Decode(bits); if (biasedFirstFeatureId == 0) { LOG(LERROR, ("Decoded first link restriction feature id delta is zero.")); return false; } - restriction.m_featureIds[0] = - prevFirstLinkFeatureId + base::checked_cast(biasedFirstFeatureId) - 1; + + restriction[0] = prevFirstLinkFeatureId + base::checked_cast(biasedFirstFeatureId) - 1; for (size_t j = 1; j < numLinks; ++j) { auto const biasedDelta = coding::DeltaCoder::Decode(bits); @@ -196,13 +192,14 @@ private: } uint32_t const delta = base::asserted_cast(biasedDelta - 1); - restriction.m_featureIds[j] = - static_cast(bits::ZigZagDecode(delta) + restriction.m_featureIds[j - 1]); + restriction[j] = + static_cast(bits::ZigZagDecode(delta) + restriction[j - 1]); } - prevFirstLinkFeatureId = restriction.m_featureIds[0]; - restrictions.push_back(restriction); + prevFirstLinkFeatureId = restriction[0]; + restrictions.emplace_back(std::move(restriction)); } + return true; } };