From f61f58ffcb76905218aa6ff4689922e4c8b8b015 Mon Sep 17 00:00:00 2001 From: Lev Dragunov Date: Tue, 3 Feb 2015 12:51:21 +0300 Subject: [PATCH] Cross mwm routing pull comments fixes and refactoring --- .../RoutingAlgorithms/ManyToManyRouting.h | 2 +- routing/osrm_data_facade.hpp | 2 + routing/osrm_router.cpp | 404 ++++++++++-------- routing/osrm_router.hpp | 14 +- routing/routing_tests/routing_tests.pro | 2 +- 5 files changed, 232 insertions(+), 192 deletions(-) diff --git a/3party/osrm/osrm-backend/RoutingAlgorithms/ManyToManyRouting.h b/3party/osrm/osrm-backend/RoutingAlgorithms/ManyToManyRouting.h index bf4631b3a7..ab73150738 100644 --- a/3party/osrm/osrm-backend/RoutingAlgorithms/ManyToManyRouting.h +++ b/3party/osrm/osrm-backend/RoutingAlgorithms/ManyToManyRouting.h @@ -209,7 +209,7 @@ template class ManyToManyRouting final : public BasicRouting { for (auto edge : super::facade->GetAdjacentEdgeRange(node)) { - const auto &data = super::facade->GetEdgeData(edge,node); + const auto &data = super::facade->GetEdgeData(edge, node); const bool direction_flag = (forward_direction ? data.forward : data.backward); if (direction_flag) { diff --git a/routing/osrm_data_facade.hpp b/routing/osrm_data_facade.hpp index 882bef51b0..ef1307d3bc 100644 --- a/routing/osrm_data_facade.hpp +++ b/routing/osrm_data_facade.hpp @@ -91,6 +91,7 @@ public: virtual EdgeDataT & GetEdgeData(const EdgeID e) { static EdgeDataT res; + ASSERT(false, ("Maps me routing facade do not supports this edge unpacking method")); return res; } @@ -110,6 +111,7 @@ public: virtual EdgeDataT & GetEdgeData(const EdgeID e) const { static EdgeDataT res; + ASSERT(false, ("Maps me routing facade do not supports this edge unpacking method")); return res; } diff --git a/routing/osrm_router.cpp b/routing/osrm_router.cpp index a0dd815e41..13764d24fe 100644 --- a/routing/osrm_router.cpp +++ b/routing/osrm_router.cpp @@ -26,8 +26,6 @@ #include "../3party/osrm/osrm-backend/RoutingAlgorithms/ShortestPathRouting.h" #include "../3party/osrm/osrm-backend/RoutingAlgorithms/ManyToManyRouting.h" -#include "../std/queue.hpp" - namespace routing { @@ -453,7 +451,14 @@ void OsrmRouter::CalculateRouteAsync(ReadyCallback const & callback) try { - code = CalculateRouteImpl(startPt, startDr, finalPt, route); + try + { + code = CalculateRouteImpl(startPt, startDr, finalPt, route); + } + catch (OsrmRouter::ResultCode e) + { + code = e; + } switch (code) { case StartPointNotFound: @@ -500,38 +505,19 @@ bool IsRouteExist(RawRouteData const & r) } MultiroutingTaskPointT::const_iterator OsrmRouter::FilterWeightsMatrix(MultiroutingTaskPointT const & sources, MultiroutingTaskPointT const & targets, - std::vector &weightMatrix, bool const filterSource) + std::vector &weightMatrix) { - if (filterSource) - { - for (size_t i=0; i-1; --k) - { - weightMatrix.erase(weightMatrix.begin()+k*targets.size(), weightMatrix.begin()+k*targets.size()+i); - weightMatrix.erase(weightMatrix.begin()+k*targets.size()+i+1, weightMatrix.begin()+(k+1)*targets.size()); - } - ASSERT(weightMatrix.size()==sources.size(), ()); - return targets.cbegin() + i; - } - return targets.cend(); - } + return sources.cend(); } void OsrmRouter::FindWeightsMatrix(MultiroutingTaskPointT const & sources, MultiroutingTaskPointT const & targets, @@ -749,22 +735,74 @@ OsrmRouter::ResultCode OsrmRouter::MakeRouteFromCrossesPath(CheckedPathT const & return OsrmRouter::NoError; } -OsrmRouter::ResultCode OsrmRouter::CalculateRouteImpl(m2::PointD const & startPt, m2::PointD const & startDr, m2::PointD const & finalPt, Route & route) -{ - RoutingMappingPtrT startMapping; - RoutingMappingPtrT finalMapping; - try +class OsrmRouter::LastCrossFinder { + CrossRoutingContext const & m_targetContext; + string const m_mwmName; + MultiroutingTaskPointT m_sources; + FeatureGraphNode m_target; + vector m_weights; + +public: + LastCrossFinder(RoutingMappingPtrT & mapping, FeatureGraphNodeVecT const & targetTask):m_targetContext(mapping->dataFacade.getRoutingContext()), m_mwmName(mapping->GetName()) { - startMapping = m_indexManager.GetMappingByPoint(startPt, m_pIndex); - finalMapping = m_indexManager.GetMappingByPoint(finalPt, m_pIndex); - } - catch (OsrmRouter::ResultCode e) - { - return e; + auto income_iterators = m_targetContext.GetIngoingIterators(); + MultiroutingTaskPointT targets; + m_sources.resize(distance(income_iterators.first, income_iterators.second)); + + for (auto i = income_iterators.first; i weights; + OsrmRouter::FindWeightsMatrix(m_sources, targets, mapping->dataFacade, weights); + + for (size_t i=0; i-1; --k) + { + weights.erase(weights.begin()+k*targets.size(), weights.begin()+k*targets.size()+i); + weights.erase(weights.begin()+k*targets.size()+i+1, weights.begin()+(k+1)*targets.size()); + } + ASSERT(weights.size()==m_sources.size(), ()); + m_target = targets[i]; + m_weights.swap(weights); + return; + } + throw OsrmRouter::EndPointNotFound; } + bool MakeLastCrossSegment(size_t const incomeNodeId, OsrmRouter::RoutePathCross & outCrossTask) + { + auto income_iterators = m_targetContext.GetIngoingIterators(); + + auto const it = find(income_iterators.first, income_iterators.second, incomeNodeId); + ASSERT(it != income_iterators.second, ()); + + const size_t targetNumber = distance(income_iterators.first, it); + const EdgeWeight targetWeight = m_weights[targetNumber]; + if (targetWeight == INVALID_EDGE_WEIGHT) + { + outCrossTask.weight = INVALID_EDGE_WEIGHT; + return false; + } + outCrossTask = {m_mwmName, m_sources[targetNumber], m_target, targetWeight}; + return targetWeight != INVALID_EDGE_WEIGHT; + } + +}; + + + +OsrmRouter::ResultCode OsrmRouter::CalculateRouteImpl(m2::PointD const & startPt, m2::PointD const & startDr, m2::PointD const & finalPt, Route & route) +{ + RoutingMappingPtrT startMapping = m_indexManager.GetMappingByPoint(startPt, m_pIndex); + RoutingMappingPtrT targetMapping = m_indexManager.GetMappingByPoint(finalPt, m_pIndex); + MappingGuard startMappingGuard(startMapping); - MappingGuard finalMappingGuard(finalMapping); + MappingGuard finalMappingGuard(targetMapping); UNUSED_VALUE(startMappingGuard); UNUSED_VALUE(finalMappingGuard); @@ -779,7 +817,7 @@ OsrmRouter::ResultCode OsrmRouter::CalculateRouteImpl(m2::PointD const & startPt { if (finalPt != m_CachedTargetPoint) { - ResultCode const code = FindPhantomNodes(finalMapping->GetName(), finalPt, m2::PointD::Zero(), m_CachedTargetTask, MAX_NODE_CANDIDATES, finalMapping); + ResultCode const code = FindPhantomNodes(targetMapping->GetName(), finalPt, m2::PointD::Zero(), m_CachedTargetTask, MAX_NODE_CANDIDATES, targetMapping); if (code != NoError) return code; m_CachedTargetPoint = finalPt; @@ -791,7 +829,8 @@ OsrmRouter::ResultCode OsrmRouter::CalculateRouteImpl(m2::PointD const & startPt // 4. Find route. RawRoutingResultT routingResult; - if (startMapping->GetName() == finalMapping->GetName()) + // 4.1 Single mwm case + if (startMapping->GetName() == targetMapping->GetName()) { LOG(LINFO, ("Single mwm routing case")); if (!FindSingleRoute(startTask, m_CachedTargetTask, startMapping->dataFacade, routingResult)) @@ -819,195 +858,185 @@ OsrmRouter::ResultCode OsrmRouter::CalculateRouteImpl(m2::PointD const & startPt return NoError; } - else + else //4.2 Different mwm case { LOG(LINFO, ("Different mwm routing case")); + // Load source data CrossRoutingContext const & startContext = startMapping->dataFacade.getRoutingContext(); auto out_iterators = startContext.GetOutgoingIterators(); - CrossRoutingContext const & targetContext = finalMapping->dataFacade.getRoutingContext(); - auto income_iterators = targetContext.GetIngoingIterators(); MultiroutingTaskPointT sources, targets(distance(out_iterators.first, out_iterators.second)); - MultiroutingTaskPointT income_sources(distance(income_iterators.first, income_iterators.second)), income_targets; + for (auto i = out_iterators.first; ifirst, targets[distance(out_iterators.first, i)]); - for (auto i = income_iterators.first; i weights, target_weights; + vector weights; FindWeightsMatrix(sources, targets, startMapping->dataFacade, weights); - FindWeightsMatrix(income_sources, income_targets, finalMapping->dataFacade, target_weights); - auto sourceNodeIter = FilterWeightsMatrix(sources, targets,weights, true); - auto targetNodeIter = FilterWeightsMatrix(income_sources, income_targets, target_weights, false); + auto sourceNodeIter = FilterWeightsMatrix(sources, targets,weights); if (sourceNodeIter == sources.cend()) return StartPointNotFound; - if (targetNodeIter == targets.cend()) - return EndPointNotFound; + + if (m_requestCancel) + return Cancelled; + + //Load target data + LastCrossFinder targetFinder(targetMapping, m_CachedTargetTask); EdgeWeight finalWeight = INVALID_EDGE_WEIGHT; CheckedPathT finalPath; - struct pathChecker{ - bool operator() (CheckedPathT const & a, CheckedPathT const & b) const { - return getPathWeight(b), pathChecker> crossTasks; + RoutingTaskQueueT crossTasks; CheckedOutsT checkedOuts; - for (size_t j=0; jsecond); + RoutingMappingPtrT nextMapping; + try { - string const & nextMwm = startContext.getOutgoingMwmName((out_iterators.first+j)->second); - RoutingMappingPtrT nextMapping; - try + nextMapping = m_indexManager.GetMappingByName(nextMwm, m_pIndex); + } + catch (OsrmRouter::ResultCode e) + { + continue; //There is no outgoing mwm + } + MappingGuard nextMappingGuard(nextMapping); + UNUSED_VALUE(nextMappingGuard); + size_t tNodeId = (out_iterators.first+j)->first; + size_t nextNodeId = FindNextMwmNode(startMapping, tNodeId, nextMapping); + if(nextNodeId == INVALID_NODE_ID) + continue; + checkedOuts.insert(make_pair(tNodeId, startMapping->GetName())); + CheckedPathT tmpPath; + tmpPath.push_back({startMapping->GetName(), *sourceNodeIter, targets[j], weights[j]}); + if (nextMwm != targetMapping->GetName()) + { + + FeatureGraphNode tmpNode; + OsrmRouter::GenerateRoutingTaskFromNodeId(nextNodeId, tmpNode); + tmpPath.push_back({nextMapping->GetName(), tmpNode, tmpNode, 0}); + crossTasks.push(tmpPath); + } + else + { + OsrmRouter::RoutePathCross targetCross; + if(targetFinder.MakeLastCrossSegment(nextNodeId, targetCross)) { - nextMapping = m_indexManager.GetMappingByName(nextMwm, m_pIndex); - } - catch (OsrmRouter::ResultCode e) - { - continue; //There is no outgoing mwm - } - MappingGuard nextMappingGuard(nextMapping); - UNUSED_VALUE(nextMappingGuard); - size_t tNodeId = (out_iterators.first+j)->first; - size_t nextNodeId = FindNextMwmNode(startMapping, tNodeId, nextMapping); - if(nextNodeId == INVALID_NODE_ID) - continue; - checkedOuts.insert(make_pair(tNodeId, startMapping->GetName())); - if (nextMwm != finalMapping->GetName()) - { - CheckedPathT tmpPath; - FeatureGraphNode tmpNode; - OsrmRouter::GenerateRoutingTaskFromNodeId(nextNodeId, tmpNode); - tmpPath.push_back({startMapping->GetName(), *sourceNodeIter, targets[j], weights[j]}); - tmpPath.push_back({nextMapping->GetName(), tmpNode, tmpNode, 0}); - crossTasks.push(tmpPath); - } - else - { - auto const it = find(income_iterators.first, income_iterators.second, nextNodeId); - ASSERT(it != income_iterators.second, ()); - const size_t targetNumber = distance(income_iterators.first, it); - const EdgeWeight targetWeight = target_weights[targetNumber]; - if (targetWeight == INVALID_EDGE_WEIGHT) - continue; - const EdgeWeight newWeight = weights[j] + targetWeight; - if (newWeightGetName(), *sourceNodeIter, targets[j], weights[j]}); - finalPath.push_back({finalMapping->GetName(), income_sources[targetNumber], *targetNodeIter, targetWeight}); + tmpPath.push_back(targetCross); + finalWeight = weights[j] + targetCross.weight; + finalPath.swap(tmpPath); } } } } + } - if (crossTasks.size()) + // Process tasks from tasks queu + if (crossTasks.size()) + { + while (getPathWeight(crossTasks.top())dataFacade.getRoutingContext(); - auto current_in_iterators = currentContext.GetIngoingIterators(); - auto current_out_iterators = currentContext.GetOutgoingIterators(); + currentMapping = m_indexManager.GetMappingByName(cross.mwmName, m_pIndex); + } + catch (OsrmRouter::ResultCode e) + { + continue; //There is no outgoing mwm + } + MappingGuard currentMappingGuard(currentMapping); + UNUSED_VALUE(currentMappingGuard); + CrossRoutingContext const & currentContext = currentMapping->dataFacade.getRoutingContext(); + auto current_in_iterators = currentContext.GetIngoingIterators(); + auto current_out_iterators = currentContext.GetOutgoingIterators(); - //find income number - auto const iit = find(current_in_iterators.first, current_in_iterators.second, cross.startNode.m_node.forward_node_id); - ASSERT(iit != current_in_iterators.second, ()); + //find income number + auto const iit = find(current_in_iterators.first, current_in_iterators.second, cross.startNode.m_node.forward_node_id); + ASSERT(iit != current_in_iterators.second, ()); - //find outs - for (auto oit=current_out_iterators.first; oit!= current_out_iterators.second; ++oit) + //find outs + for (auto oit=current_out_iterators.first; oit!= current_out_iterators.second; ++oit) + { + const EdgeWeight outWeight = currentContext.getAdjacencyCost(iit, oit); + if (outWeight != INVALID_EDGE_WEIGHT) { - const EdgeWeight outWeight = currentContext.getAdjacencyCost(iit, oit); - if (outWeight != INVALID_EDGE_WEIGHT) + if (getPathWeight(topTask)+outWeight < finalWeight) { - if (getPathWeight(topTask)+outWeight < finalWeight) + if (m_requestCancel) + return Cancelled; + string const & nextMwm = currentContext.getOutgoingMwmName(oit->second); + RoutingMappingPtrT nextMapping; + try { - string const & nextMwm = currentContext.getOutgoingMwmName(oit->second); - RoutingMappingPtrT nextMapping; - try - { - nextMapping = m_indexManager.GetMappingByName(nextMwm, m_pIndex); - } - catch (OsrmRouter::ResultCode e) - { - continue; //There is no outgoing mwm - } + nextMapping = m_indexManager.GetMappingByName(nextMwm, m_pIndex); + } + catch (OsrmRouter::ResultCode e) + { + continue; //There is no outgoing mwm + } - MappingGuard nextMappingGuard(nextMapping); - UNUSED_VALUE(nextMappingGuard); - size_t const tNodeId = oit->first; - auto const outNode = make_pair(tNodeId, currentMapping->GetName()); - if (checkedOuts.find(outNode)!=checkedOuts.end()) - continue; - checkedOuts.insert(outNode); - size_t nextNodeId = FindNextMwmNode(currentMapping, tNodeId, nextMapping); - if(nextNodeId == INVALID_NODE_ID) - continue; - if (nextMwm != finalMapping->GetName()) + MappingGuard nextMappingGuard(nextMapping); + UNUSED_VALUE(nextMappingGuard); + size_t const tNodeId = oit->first; + auto const outNode = make_pair(tNodeId, currentMapping->GetName()); + if (checkedOuts.find(outNode)!=checkedOuts.end()) + continue; + checkedOuts.insert(outNode); + size_t nextNodeId = FindNextMwmNode(currentMapping, tNodeId, nextMapping); + if(nextNodeId == INVALID_NODE_ID) + continue; + CheckedPathT tmpPath(topTask); + FeatureGraphNode startNode, stopNode; + OsrmRouter::GenerateRoutingTaskFromNodeId(*iit, startNode); + OsrmRouter::GenerateRoutingTaskFromNodeId(tNodeId, stopNode); + tmpPath.pop_back(); + tmpPath.push_back({currentMapping->GetName(), startNode, stopNode, outWeight}); + if (nextMwm != targetMapping->GetName()) + { + FeatureGraphNode tmpNode; + OsrmRouter::GenerateRoutingTaskFromNodeId(nextNodeId, tmpNode); + tmpPath.push_back({nextMapping->GetName(), tmpNode, tmpNode, 0}); + crossTasks.emplace(tmpPath); + } + else + { + OsrmRouter::RoutePathCross targetCross; + if(targetFinder.MakeLastCrossSegment(nextNodeId, targetCross)) { - CheckedPathT tmpPath(topTask); - FeatureGraphNode startNode, stopNode, tmpNode; - OsrmRouter::GenerateRoutingTaskFromNodeId(nextNodeId, tmpNode); - OsrmRouter::GenerateRoutingTaskFromNodeId(*iit, startNode); - OsrmRouter::GenerateRoutingTaskFromNodeId(tNodeId, stopNode); - tmpPath.pop_back(); - tmpPath.push_back({currentMapping->GetName(), startNode, stopNode, outWeight}); - tmpPath.push_back({nextMapping->GetName(), tmpNode, tmpNode, 0}); - crossTasks.push(tmpPath); - } - else - { - auto const it = find(income_iterators.first, income_iterators.second, nextNodeId); - ASSERT(it != income_iterators.second, ()); - const size_t targetNumber = distance(income_iterators.first, it); - const EdgeWeight targetWeight = target_weights[targetNumber]; - if (targetWeight == INVALID_EDGE_WEIGHT) - continue; - const EdgeWeight newWeight = getPathWeight(topTask) + outWeight + targetWeight; - if (newWeightGetName(), startNode, stopNode, outWeight}); - finalPath.push_back({nextMwm, income_sources[targetNumber], *targetNodeIter, targetWeight}); + finalPath.swap(tmpPath); } } } } } - if (!crossTasks.size()) - break; } + if (!crossTasks.size()) + break; } + } + // 5. Make generate answer if (finalWeight < INVALID_EDGE_WEIGHT) return MakeRouteFromCrossesPath(finalPath, route); else @@ -1505,7 +1534,6 @@ void OsrmRouter::GetTurnDirection(PathData const & node1, if (nodesSz == 0) { - ASSERT(false, ()); return; } diff --git a/routing/osrm_router.hpp b/routing/osrm_router.hpp index ad8b51254f..67ef03dba9 100644 --- a/routing/osrm_router.hpp +++ b/routing/osrm_router.hpp @@ -11,6 +11,7 @@ #include "../std/function.hpp" #include "../std/numeric.hpp" #include "../std/atomic.hpp" +#include "../std/queue.hpp" #include "../3party/osrm/osrm-backend/DataStructures/QueryEdge.h" #include "../3party/osrm/osrm-backend/DataStructures/RawRouteData.h" @@ -83,7 +84,7 @@ struct RoutingMapping void Unmap() { --map_counter; - if (map_counter<1 && mapping.IsMapped()) + if (map_counter < 1 && mapping.IsMapped()) mapping.Unmap(); } @@ -268,14 +269,23 @@ private: EdgeWeight weight; }; typedef vector CheckedPathT; + class LastCrossFinder; static EdgeWeight getPathWeight(CheckedPathT const & path) { return accumulate(path.begin(), path.end(), 0, [](EdgeWeight sum, RoutePathCross const & elem){return sum+elem.weight;}); } + struct pathChecker{ + bool operator() (CheckedPathT const & a, CheckedPathT const & b) const { + return getPathWeight(b), pathChecker> RoutingTaskQueueT; + MultiroutingTaskPointT::const_iterator FilterWeightsMatrix(MultiroutingTaskPointT const & sources, MultiroutingTaskPointT const & targets, - std::vector &weightMatrix, bool const filterSource); + std::vector &weightMatrix); /*! * \brief Makes route (points turns and other annotations) and submits it to @route class diff --git a/routing/routing_tests/routing_tests.pro b/routing/routing_tests/routing_tests.pro index 12567d540d..08ee21dab2 100644 --- a/routing/routing_tests/routing_tests.pro +++ b/routing/routing_tests/routing_tests.pro @@ -21,5 +21,5 @@ SOURCES += \ ../../testing/testingmain.cpp \ osrm_router_test.cpp \ vehicle_model_test.cpp \ - cross_routing_tests.cpp + cross_routing_tests.cpp \