From 5c45a68ecb84bda9e4071de91a61d35f1f103c32 Mon Sep 17 00:00:00 2001 From: Vladimir Byko-Ianko Date: Mon, 25 Dec 2017 14:40:01 +0300 Subject: [PATCH] Removing osrm. cpp, hpp, CMake. --- CMakeLists.txt | 1 - defines.hpp | 9 - generator/CMakeLists.txt | 1 - generator/generator_tests/CMakeLists.txt | 1 - generator/generator_tool/CMakeLists.txt | 1 - .../srtm_coverage_checker/CMakeLists.txt | 46 -- .../srtm_coverage_checker.cpp | 152 ------ map/map_tests/CMakeLists.txt | 1 - map/style_tests/CMakeLists.txt | 1 - .../openlr_assessment_tool/CMakeLists.txt | 1 - qt/CMakeLists.txt | 1 - routing/CMakeLists.txt | 17 - routing/async_router.cpp | 5 +- routing/cross_mwm_graph.cpp | 17 +- routing/cross_mwm_graph.hpp | 15 +- routing/cross_mwm_index_graph.hpp | 1 - routing/cross_mwm_osrm_graph.cpp | 263 --------- routing/cross_mwm_osrm_graph.hpp | 90 ---- routing/cross_mwm_road_graph.cpp | 441 --------------- routing/cross_mwm_road_graph.hpp | 201 ------- routing/cross_mwm_router.cpp | 107 ---- routing/cross_mwm_router.hpp | 45 -- routing/cross_routing_context.cpp | 196 ------- routing/cross_routing_context.hpp | 173 ------ routing/index_router.cpp | 5 +- routing/index_router.hpp | 2 - routing/online_absent_fetcher.hpp | 4 +- routing/osrm2feature_map.cpp | 505 ------------------ routing/osrm2feature_map.hpp | 206 ------- routing/osrm_data_facade.hpp | 312 ----------- routing/osrm_engine.cpp | 94 ---- routing/osrm_engine.hpp | 74 --- routing/router.hpp | 2 +- routing/routing_benchmarks/CMakeLists.txt | 1 - .../routing_consistency_tests/CMakeLists.txt | 1 - .../routing_integration_tests/CMakeLists.txt | 2 - .../cross_section_tests.cpp | 229 -------- routing/routing_mapping.cpp | 217 -------- routing/routing_mapping.hpp | 126 ----- routing/routing_session.hpp | 2 + routing/routing_tests/CMakeLists.txt | 6 - routing/routing_tests/async_router_test.cpp | 1 + routing/routing_tests/cross_routing_tests.cpp | 176 ------ routing/routing_tests/osrm_router_test.cpp | 261 --------- .../routing_tests/routing_mapping_test.cpp | 126 ----- routing/turns_generator.cpp | 5 +- routing/turns_generator.hpp | 2 - .../assessment_tool/CMakeLists.txt | 1 - .../storage_integration_tests/CMakeLists.txt | 1 - storage/storage_tests/CMakeLists.txt | 1 - track_analyzing/track_analyzer/CMakeLists.txt | 1 - 51 files changed, 28 insertions(+), 4121 deletions(-) delete mode 100644 generator/srtm_coverage_checker/CMakeLists.txt delete mode 100644 generator/srtm_coverage_checker/srtm_coverage_checker.cpp delete mode 100644 routing/cross_mwm_osrm_graph.cpp delete mode 100644 routing/cross_mwm_osrm_graph.hpp delete mode 100644 routing/cross_mwm_road_graph.cpp delete mode 100644 routing/cross_mwm_road_graph.hpp delete mode 100644 routing/cross_mwm_router.cpp delete mode 100644 routing/cross_mwm_router.hpp delete mode 100644 routing/cross_routing_context.cpp delete mode 100644 routing/cross_routing_context.hpp delete mode 100644 routing/osrm2feature_map.cpp delete mode 100644 routing/osrm2feature_map.hpp delete mode 100644 routing/osrm_data_facade.hpp delete mode 100644 routing/osrm_engine.cpp delete mode 100644 routing/osrm_engine.hpp delete mode 100644 routing/routing_integration_tests/cross_section_tests.cpp delete mode 100644 routing/routing_mapping.cpp delete mode 100644 routing/routing_mapping.hpp delete mode 100644 routing/routing_tests/cross_routing_tests.cpp delete mode 100644 routing/routing_tests/osrm_router_test.cpp delete mode 100644 routing/routing_tests/routing_mapping_test.cpp diff --git a/CMakeLists.txt b/CMakeLists.txt index 1f6ae7c7e0..1dcd44bb90 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -370,7 +370,6 @@ add_subdirectory(3party/protobuf) add_subdirectory(3party/liboauthcpp) add_subdirectory(3party/pugixml) add_subdirectory(3party/succinct) -add_subdirectory(3party/osrm) add_subdirectory(3party/gflags) add_subdirectory(3party/bsdiff-courgette) add_subdirectory(base) diff --git a/defines.hpp b/defines.hpp index 70de66e7f6..4dbbf06569 100644 --- a/defines.hpp +++ b/defines.hpp @@ -45,15 +45,6 @@ #define TRANSIT_FILE_TAG "transit" #define UGC_FILE_TAG "ugc" -#define ROUTING_MATRIX_FILE_TAG "mercedes" -#define ROUTING_EDGEDATA_FILE_TAG "daewoo" -#define ROUTING_EDGEID_FILE_TAG "infinity" -#define ROUTING_SHORTCUTS_FILE_TAG "skoda" -#define ROUTING_CROSS_CONTEXT_TAG "chrysler" - -#define ROUTING_FTSEG_FILE_TAG "ftseg" -#define ROUTING_NODEIND_TO_FTSEGIND_FILE_TAG "node2ftseg" - #define READY_FILE_EXTENSION ".ready" #define RESUME_FILE_EXTENSION ".resume" #define DOWNLOADING_FILE_EXTENSION ".downloading" diff --git a/generator/CMakeLists.txt b/generator/CMakeLists.txt index 6484804fa8..e5080773ab 100644 --- a/generator/CMakeLists.txt +++ b/generator/CMakeLists.txt @@ -113,4 +113,3 @@ add_subdirectory(generator_tool) add_subdirectory(booking_quality_check) add_subdirectory(feature_segments_checker) add_subdirectory(restaurants_info) -add_subdirectory(srtm_coverage_checker) diff --git a/generator/generator_tests/CMakeLists.txt b/generator/generator_tests/CMakeLists.txt index 3375b1cdc4..8fc3691b1b 100644 --- a/generator/generator_tests/CMakeLists.txt +++ b/generator/generator_tests/CMakeLists.txt @@ -54,7 +54,6 @@ omim_link_libraries( agg jansson protobuf - osrm stats_client minizip succinct diff --git a/generator/generator_tool/CMakeLists.txt b/generator/generator_tool/CMakeLists.txt index 9f304ec107..a9b91ca993 100644 --- a/generator/generator_tool/CMakeLists.txt +++ b/generator/generator_tool/CMakeLists.txt @@ -30,7 +30,6 @@ omim_link_libraries( protobuf mwm_diff bsdiff - osrm stats_client minizip succinct diff --git a/generator/srtm_coverage_checker/CMakeLists.txt b/generator/srtm_coverage_checker/CMakeLists.txt deleted file mode 100644 index 5427912aab..0000000000 --- a/generator/srtm_coverage_checker/CMakeLists.txt +++ /dev/null @@ -1,46 +0,0 @@ -project(srtm_coverage_checker) - -include_directories( - ${OMIM_ROOT}/3party/gflags/src - ${OMIM_ROOT}/3party/jansson/src -) - -set( - SRC - srtm_coverage_checker.cpp -) - -omim_add_executable(${PROJECT_NAME} ${SRC}) - -omim_link_libraries( - ${PROJECT_NAME} - generator - map - routing - routing_common - search - storage - ugc - traffic - indexer - editor - platform - geometry - coding - base - agg - gflags - icu - jansson - minizip - oauthcpp - opening_hours - osrm - protobuf - pugixml - stats_client - succinct - ${LIBZ} -) - -link_qt5_core(${PROJECT_NAME}) diff --git a/generator/srtm_coverage_checker/srtm_coverage_checker.cpp b/generator/srtm_coverage_checker/srtm_coverage_checker.cpp deleted file mode 100644 index c7a71c238b..0000000000 --- a/generator/srtm_coverage_checker/srtm_coverage_checker.cpp +++ /dev/null @@ -1,152 +0,0 @@ -#include "generator/srtm_parser.hpp" - -#include "map/feature_vec_model.hpp" - -#include "routing/routing_integration_tests/routing_test_tools.hpp" - -#include "indexer/feature_altitude.hpp" - -#include "coding/file_name_utils.hpp" - -#include "platform/country_file.hpp" -#include "platform/local_country_file_utils.hpp" - -#include "base/logging.hpp" - -#include - -#include - -#include "3party/gflags/src/gflags/gflags.h" - -DEFINE_string(srtm_path, "", "Path to directory with SRTM files"); -DEFINE_string(mwm_path, "", "Path to mwm files (writable dir)"); - -// TODO: remove when routing_test_tools becomes a library. -namespace -{ -void ChangeMaxNumberOfOpenFiles(size_t n) -{ - struct rlimit rlp; - getrlimit(RLIMIT_NOFILE, &rlp); - rlp.rlim_cur = n; - setrlimit(RLIMIT_NOFILE, &rlp); -} - -shared_ptr CreateFeaturesFetcher(vector const & localFiles) -{ - size_t const maxOpenFileNumber = 1024; - ChangeMaxNumberOfOpenFiles(maxOpenFileNumber); - shared_ptr featuresFetcher(new model::FeaturesFetcher); - featuresFetcher->InitClassificator(); - - for (LocalCountryFile const & localFile : localFiles) - { - auto p = featuresFetcher->RegisterMap(localFile); - if (p.second != MwmSet::RegResult::Success) - { - ASSERT(false, ("Can't register", localFile)); - return nullptr; - } - } - return featuresFetcher; -} -} // namespace - -int main(int argc, char * argv[]) -{ - google::SetUsageMessage("SRTM coverage checker."); - google::ParseCommandLineFlags(&argc, &argv, true); - - Platform & platform = GetPlatform(); - if (!FLAGS_mwm_path.empty()) - platform.SetWritableDirForTests(FLAGS_mwm_path); - - if (FLAGS_srtm_path.empty()) - { - LOG(LERROR, ("SRTM files directory is not specified.")); - return -1; - } - - LOG(LINFO, ("writable dir =", platform.WritableDir())); - LOG(LINFO, ("srtm dir =", FLAGS_srtm_path)); - - vector localFiles; - platform::FindAllLocalMapsAndCleanup(numeric_limits::max() /* latestVersion */, - localFiles); - - auto fetcher = CreateFeaturesFetcher(localFiles); - generator::SrtmTileManager manager(FLAGS_srtm_path); - - for (auto & file : localFiles) - { - file.SyncWithDisk(); - if (file.GetFiles() != MapOptions::MapWithCarRouting) - { - LOG(LINFO, ("Warning! Routing file not found for:", file.GetCountryName())); - continue; - } - - FilesMappingContainer container(file.GetPath(MapOptions::CarRouting)); - if (!container.IsExist(ROUTING_FTSEG_FILE_TAG)) - { - LOG(LINFO, ("Warning! Mwm file has not routing ftseg section:", file.GetCountryName())); - continue; - } - - routing::TDataFacade dataFacade; - dataFacade.Load(container); - - OsrmFtSegMapping segMapping; - segMapping.Load(container, file); - segMapping.Map(container); - - size_t all = 0; - size_t good = 0; - - for (TOsrmNodeId i = 0; i < dataFacade.GetNumberOfNodes(); ++i) - { - buffer_vector buffer; - segMapping.ForEachFtSeg(i, MakeBackInsertFunctor(buffer)); - - vector path; - for (size_t k = 0; k < buffer.size(); ++k) - { - auto const & segment = buffer[k]; - if (!segment.IsValid()) - continue; - // Load data from drive. - FeatureType ft; - Index::FeaturesLoaderGuard loader( - fetcher->GetIndex(), fetcher->GetIndex().GetMwmIdByCountryFile(file.GetCountryFile())); - if (!loader.GetFeatureByIndex(segment.m_fid, ft)) - continue; - ft.ParseGeometry(FeatureType::BEST_GEOMETRY); - - // Get points in proper direction. - auto const startIdx = segment.m_pointStart; - auto const endIdx = segment.m_pointEnd; - for (auto idx = std::min(startIdx, endIdx); idx <= std::max(startIdx, endIdx); ++idx) - path.push_back(ft.GetPoint(idx)); - - all += path.size(); - for (auto const & point : path) - { - auto const height = manager.GetHeight(MercatorBounds::ToLatLon(point)); - if (height != feature::kInvalidAltitude) - good++; - } - } - } - - auto const bad = all - good; - auto const percent = all == 0 ? 0.0 : bad * 100.0 / all; - if (percent > 10.0) - { - LOG(LINFO, ("Huge error rate in:", file.GetCountryName(), "good:", good, "bad:", bad, "all:", - all, "%:", percent)); - } - } - - return 0; -} diff --git a/map/map_tests/CMakeLists.txt b/map/map_tests/CMakeLists.txt index 661288b733..49ba5101fc 100644 --- a/map/map_tests/CMakeLists.txt +++ b/map/map_tests/CMakeLists.txt @@ -57,7 +57,6 @@ omim_link_libraries( expat protobuf jansson - osrm stats_client minizip succinct diff --git a/map/style_tests/CMakeLists.txt b/map/style_tests/CMakeLists.txt index 656b5bff66..1c927fd503 100644 --- a/map/style_tests/CMakeLists.txt +++ b/map/style_tests/CMakeLists.txt @@ -18,7 +18,6 @@ omim_link_libraries( map traffic indexer - osrm pugixml opening_hours editor diff --git a/openlr/openlr_match_quality/openlr_assessment_tool/CMakeLists.txt b/openlr/openlr_match_quality/openlr_assessment_tool/CMakeLists.txt index f5d9a5bbc4..358890e139 100644 --- a/openlr/openlr_match_quality/openlr_assessment_tool/CMakeLists.txt +++ b/openlr/openlr_match_quality/openlr_assessment_tool/CMakeLists.txt @@ -67,7 +67,6 @@ omim_link_libraries( oauthcpp opening_hours openlr - osrm protobuf pugixml sdf_image diff --git a/qt/CMakeLists.txt b/qt/CMakeLists.txt index fc1b56e5a3..d65cb1d702 100644 --- a/qt/CMakeLists.txt +++ b/qt/CMakeLists.txt @@ -84,7 +84,6 @@ omim_link_libraries( agg jansson protobuf - osrm stats_client minizip succinct diff --git a/routing/CMakeLists.txt b/routing/CMakeLists.txt index d15ac0b0b4..34133a191a 100644 --- a/routing/CMakeLists.txt +++ b/routing/CMakeLists.txt @@ -3,8 +3,6 @@ project(routing) include_directories( . ${OMIM_ROOT}/3party/jansson/src - ${OMIM_ROOT}/3party/osrm/osrm-backend/include - ${OMIM_ROOT}/3party/osrm/osrm-backend/third_party ) set( @@ -31,14 +29,6 @@ set( cross_mwm_graph.hpp cross_mwm_ids.hpp cross_mwm_index_graph.hpp - cross_mwm_osrm_graph.cpp - cross_mwm_osrm_graph.hpp - cross_mwm_road_graph.cpp - cross_mwm_road_graph.hpp - cross_mwm_router.cpp - cross_mwm_router.hpp - cross_routing_context.cpp - cross_routing_context.hpp directions_engine.cpp directions_engine.hpp edge_estimator.cpp @@ -77,11 +67,6 @@ set( online_absent_fetcher.hpp online_cross_fetcher.cpp online_cross_fetcher.hpp - osrm2feature_map.cpp - osrm2feature_map.hpp - osrm_data_facade.hpp - osrm_engine.cpp - osrm_engine.hpp pedestrian_directions.cpp pedestrian_directions.hpp restriction_loader.cpp @@ -113,8 +98,6 @@ set( routing_exceptions.hpp routing_helpers.cpp routing_helpers.hpp - routing_mapping.cpp - routing_mapping.hpp routing_result_graph.hpp routing_session.cpp routing_session.hpp diff --git a/routing/async_router.cpp b/routing/async_router.cpp index d45e7e591c..297f90e3b9 100644 --- a/routing/async_router.cpp +++ b/routing/async_router.cpp @@ -2,14 +2,17 @@ #include "platform/platform.hpp" +#include "geometry/mercator.hpp" + #include "base/logging.hpp" #include "base/macros.hpp" #include "base/string_utils.hpp" #include "base/timer.hpp" -#include "geometry/mercator.hpp" +#include "std/functional.hpp" using namespace std; +using namespace std::placeholders; namespace routing { diff --git a/routing/cross_mwm_graph.cpp b/routing/cross_mwm_graph.cpp index 82b134996b..2bcad69f3c 100644 --- a/routing/cross_mwm_graph.cpp +++ b/routing/cross_mwm_graph.cpp @@ -49,7 +49,7 @@ CrossMwmGraph::CrossMwmGraph(shared_ptr numMwmIds, shared_ptr> numMwmTree, shared_ptr vehicleModelFactory, VehicleType vehicleType, CourntryRectFn const & countryRectFn, - Index & index, RoutingIndexManager & indexManager) + Index & index) : m_index(index) , m_numMwmIds(numMwmIds) , m_numMwmTree(numMwmTree) @@ -57,7 +57,6 @@ CrossMwmGraph::CrossMwmGraph(shared_ptr numMwmIds, , m_countryRectFn(countryRectFn) , m_crossMwmIndexGraph(index, numMwmIds, vehicleType) , m_crossMwmTransitGraph(index, numMwmIds, VehicleType::Transit) - , m_crossMwmOsrmGraph(numMwmIds, indexManager) { CHECK(m_numMwmIds, ()); CHECK(m_vehicleModelFactory, ()); @@ -74,7 +73,7 @@ bool CrossMwmGraph::IsTransition(Segment const & s, bool isOutgoing) } return CrossMwmSectionExists(s.GetMwmId()) ? m_crossMwmIndexGraph.IsTransition(s, isOutgoing) - : m_crossMwmOsrmGraph.IsTransition(s, isOutgoing); + : false; } void CrossMwmGraph::FindBestTwins(NumMwmId sMwmId, bool isOutgoing, FeatureType const & ft, m2::PointD const & point, @@ -234,14 +233,12 @@ void CrossMwmGraph::GetOutgoingEdgeList(Segment const & s, vector & return; } - return CrossMwmSectionExists(s.GetMwmId()) - ? m_crossMwmIndexGraph.GetOutgoingEdgeList(s, edges) - : m_crossMwmOsrmGraph.GetEdgeList(s, true /* isOutgoing */, edges); + if (CrossMwmSectionExists(s.GetMwmId())) + m_crossMwmIndexGraph.GetOutgoingEdgeList(s, edges); } void CrossMwmGraph::Clear() { - m_crossMwmOsrmGraph.Clear(); m_crossMwmIndexGraph.Clear(); m_crossMwmTransitGraph.Clear(); } @@ -251,8 +248,10 @@ TransitionPoints CrossMwmGraph::GetTransitionPoints(Segment const & s, bool isOu CHECK(!TransitGraph::IsTransitSegment(s), ("Geometry index based twins search is not supported for transit.")); - return CrossMwmSectionExists(s.GetMwmId()) ? m_crossMwmIndexGraph.GetTransitionPoints(s, isOutgoing) - : m_crossMwmOsrmGraph.GetTransitionPoints(s, isOutgoing); + if (CrossMwmSectionExists(s.GetMwmId())) + m_crossMwmIndexGraph.GetTransitionPoints(s, isOutgoing); + + return TransitionPoints(); // No transition points. } CrossMwmGraph::MwmStatus CrossMwmGraph::GetMwmStatus(NumMwmId numMwmId, diff --git a/routing/cross_mwm_graph.hpp b/routing/cross_mwm_graph.hpp index 0f4da9266a..cfc2d667b5 100644 --- a/routing/cross_mwm_graph.hpp +++ b/routing/cross_mwm_graph.hpp @@ -2,7 +2,7 @@ #include "routing/cross_mwm_ids.hpp" #include "routing/cross_mwm_index_graph.hpp" -#include "routing/cross_mwm_osrm_graph.hpp" +#include "routing/router.hpp" #include "routing/segment.hpp" #include "routing/vehicle_mask.hpp" @@ -37,8 +37,7 @@ public: CrossMwmGraph(std::shared_ptr numMwmIds, shared_ptr> numMwmTree, std::shared_ptr vehicleModelFactory, VehicleType vehicleType, - CourntryRectFn const & countryRectFn, Index & index, - RoutingIndexManager & indexManager); + CourntryRectFn const & countryRectFn, Index & index); /// \brief Transition segment is a segment which is crossed by mwm border. That means /// start and finish of such segment have to lie in different mwms. If a segment is @@ -90,9 +89,7 @@ public: void Clear(); - // \returns nontransit transitions for mwm with id |numMwmId|. - // Should be used with CrossMwmIndexGraph only. - // @todo(bykoianko): rewrite comment and check after CrossMwmOsrm removal. + // \returns transitions for mwm with id |numMwmId| for CrossMwmIndexGraph. std::vector const & GetTransitions(NumMwmId numMwmId, bool isEnter) { CHECK(CrossMwmSectionExists(numMwmId), ("Should be used in LeapsOnly mode only. LeapsOnly mode requires CrossMwmIndexGraph.")); @@ -111,11 +108,10 @@ private: bool m_exactMatchFound; }; - /// \returns points of |s|. |s| should be a transition segment of mwm with an OSRM cross-mwm sections or - /// with an index graph cross-mwm section. + /// \returns points of |s|. |s| should be a transition segment of mwm with an index graph cross-mwm section. /// \param s is a transition segment of type |isOutgoing|. /// \note the result of the method is returned by value because the size of the vector is usually - /// one or very small in rare cases in OSRM. + /// one or very small. TransitionPoints GetTransitionPoints(Segment const & s, bool isOutgoing); MwmStatus GetMwmStatus(NumMwmId numMwmId, std::string const & sectionName) const; @@ -159,7 +155,6 @@ private: CourntryRectFn const & m_countryRectFn; CrossMwmIndexGraph m_crossMwmIndexGraph; CrossMwmIndexGraph m_crossMwmTransitGraph; - CrossMwmOsrmGraph m_crossMwmOsrmGraph; }; string DebugPrint(CrossMwmGraph::MwmStatus status); diff --git a/routing/cross_mwm_index_graph.hpp b/routing/cross_mwm_index_graph.hpp index 400b40c22c..6cfc51b02e 100644 --- a/routing/cross_mwm_index_graph.hpp +++ b/routing/cross_mwm_index_graph.hpp @@ -2,7 +2,6 @@ #include "routing/cross_mwm_connector.hpp" #include "routing/cross_mwm_connector_serialization.hpp" -#include "routing/cross_mwm_road_graph.hpp" #include "routing/fake_feature_ids.hpp" #include "routing/routing_exceptions.hpp" #include "routing/segment.hpp" diff --git a/routing/cross_mwm_osrm_graph.cpp b/routing/cross_mwm_osrm_graph.cpp deleted file mode 100644 index d7aa3a1fb8..0000000000 --- a/routing/cross_mwm_osrm_graph.cpp +++ /dev/null @@ -1,263 +0,0 @@ -#include "routing/cross_mwm_osrm_graph.hpp" -#include "routing/cross_mwm_road_graph.hpp" - -#include "base/stl_helpers.hpp" - -using namespace routing; -using namespace std; - -namespace -{ -struct NodeIds -{ - TOsrmNodeId m_directNodeId; - TOsrmNodeId m_reverseNodeId; -}; - -OsrmMappingTypes::FtSeg GetFtSeg(Segment const & segment) -{ - return OsrmMappingTypes::FtSeg(segment.GetFeatureId(), segment.GetMinPointId(), - segment.GetMaxPointId()); -} - -/// \returns a pair of direct and reverse(backward) node id by |segment|. -/// The first member of the pair is direct node id and the second is reverse node id. -NodeIds GetDirectAndReverseNodeId(OsrmFtSegMapping const & segMapping, Segment const & segment) -{ - OsrmFtSegMapping::TFtSegVec const ftSegs = {GetFtSeg(segment)}; - OsrmFtSegMapping::OsrmNodesT osrmNodes; - segMapping.GetOsrmNodes(ftSegs, osrmNodes); - CHECK_LESS(osrmNodes.size(), 2, ()); - if (osrmNodes.empty()) - return {INVALID_NODE_ID, INVALID_NODE_ID}; - - NodeIds const forwardNodeIds = {osrmNodes.begin()->second.first, - osrmNodes.begin()->second.second}; - NodeIds const backwardNodeIds = {osrmNodes.begin()->second.second, - osrmNodes.begin()->second.first}; - - return segment.IsForward() ? forwardNodeIds : backwardNodeIds; -} - -bool GetFirstValidSegment(OsrmFtSegMapping const & segMapping, NumMwmId numMwmId, - TWrittenNodeId nodeId, Segment & segment) -{ - auto const range = segMapping.GetSegmentsRange(nodeId); - for (size_t segmentIndex = range.first; segmentIndex != range.second; ++segmentIndex) - { - OsrmMappingTypes::FtSeg seg; - // The meaning of node id in osrm is an edge between two joints. - // So, it's possible to consider the first valid segment from the range which returns by - // GetSegmentsRange(). - segMapping.GetSegmentByIndex(segmentIndex, seg); - - if (seg.IsValid()) - { - CHECK_NOT_EQUAL(seg.m_pointStart, seg.m_pointEnd, ()); - segment = Segment(numMwmId, seg.m_fid, min(seg.m_pointStart, seg.m_pointEnd), seg.IsForward()); - return true; - } - } - LOG(LDEBUG, ("No valid segments in the range returned by OsrmFtSegMapping::GetSegmentsRange(", - nodeId, "). Num mwm id:", numMwmId)); - return false; -} - -void FillTransitionSegments(OsrmFtSegMapping const & segMapping, TWrittenNodeId nodeId, - NumMwmId numMwmId, ms::LatLon const & latLon, - map> & transitionSegments) -{ - Segment key; - if (!GetFirstValidSegment(segMapping, numMwmId, nodeId, key)) - return; - - transitionSegments[key].push_back(latLon); -} - -void AddSegmentEdge(NumMwmIds const & numMwmIds, OsrmFtSegMapping const & segMapping, - CrossWeightedEdge const & osrmEdge, bool isOutgoing, NumMwmId numMwmId, - vector & edges) -{ - BorderCross const & target = osrmEdge.GetTarget(); - CrossNode const & crossNode = isOutgoing ? target.fromNode : target.toNode; - - if (!crossNode.mwmId.IsAlive()) - return; - - NumMwmId const crossNodeMwmId = - numMwmIds.GetId(crossNode.mwmId.GetInfo()->GetLocalFile().GetCountryFile()); - CHECK_EQUAL(numMwmId, crossNodeMwmId, ()); - - Segment segment; - if (!GetFirstValidSegment(segMapping, crossNodeMwmId, crossNode.node, segment)) - return; - - // OSRM and AStar have different car models, therefore AStar heuristic doesn't work for OSRM node - // ids (edges). - // This factor makes index graph (used in AStar) edge weight smaller than node ids weight. - // - // As a result large cross mwm routes with connectors works as Dijkstra, but short and medium - // routes - // without connectors works as AStar. - // Most of routes don't use leaps, therefore it is important to keep AStar performance. - double constexpr kAstarHeuristicFactor = 100000; - //// Osrm multiplies seconds to 10, so we need to divide it back. - double constexpr kOSRMWeightToSecondsMultiplier = 0.1; - edges.emplace_back( - segment, RouteWeight::FromCrossMwmWeight( - osrmEdge.GetWeight() * kOSRMWeightToSecondsMultiplier * kAstarHeuristicFactor)); -} -} // namespace - -namespace routing -{ -CrossMwmOsrmGraph::CrossMwmOsrmGraph(shared_ptr numMwmIds, - RoutingIndexManager & indexManager) - : m_numMwmIds(numMwmIds), m_indexManager(indexManager) -{ - Clear(); -} - -CrossMwmOsrmGraph::~CrossMwmOsrmGraph() {} - -bool CrossMwmOsrmGraph::IsTransition(Segment const & s, bool isOutgoing) -{ - TransitionSegments const & t = GetSegmentMaps(s.GetMwmId()); - - if (isOutgoing) - return t.m_outgoing.count(s) != 0; - return t.m_ingoing.count(s) != 0; -} - -void CrossMwmOsrmGraph::GetEdgeList(Segment const & s, bool isOutgoing, vector & edges) -{ - // Note. According to cross-mwm OSRM sections there is a node id that could be ingoing and - // outgoing - // at the same time. For example in Berlin mwm on Nordlicher Berliner Ring (A10) near crossing - // with A11 there's such node id. It's an extremely rare case. There're probably several such node - // id for the whole Europe. Such cases are not processed in WorldGraph::GetEdgeList() for the time - // being. To prevent filling |edges| with twins instead of leap edges and vice versa in - // WorldGraph::GetEdgeList(). - // CrossMwmGraph::GetEdgeList() does not fill |edges| if |s| is a transition segment which - // corresponces node id described above. - if (IsTransition(s, isOutgoing)) - return; - - auto const fillEdgeList = [&](TRoutingMappingPtr const & mapping) { - vector borderCrosses; - GetBorderCross(mapping, s, isOutgoing, borderCrosses); - - for (BorderCross const & v : borderCrosses) - { - vector adj; - if (isOutgoing) - m_crossMwmGraph->GetOutgoingEdgesList(v, adj); - else - m_crossMwmGraph->GetIngoingEdgesList(v, adj); - - for (CrossWeightedEdge const & edge : adj) - AddSegmentEdge(*m_numMwmIds, mapping->m_segMapping, edge, isOutgoing, s.GetMwmId(), edges); - } - }; - - LoadWith(s.GetMwmId(), fillEdgeList); - my::SortUnique(edges); -} - -void CrossMwmOsrmGraph::Clear() -{ - m_crossMwmGraph = make_unique(m_indexManager); - m_transitionCache.clear(); - m_mappingGuards.clear(); -} - -TransitionPoints CrossMwmOsrmGraph::GetTransitionPoints(Segment const & s, bool isOutgoing) -{ - vector const & latLons = - isOutgoing ? GetOutgoingTransitionPoints(s) : GetIngoingTransitionPoints(s); - TransitionPoints points; - points.reserve(latLons.size()); - for (auto const & latLon : latLons) - points.push_back(MercatorBounds::FromLatLon(latLon)); - return points; -} - -CrossMwmOsrmGraph::TransitionSegments const & CrossMwmOsrmGraph::LoadSegmentMaps(NumMwmId numMwmId) -{ - auto it = m_transitionCache.find(numMwmId); - if (it != m_transitionCache.cend()) - return it->second; - - auto const fillAllTransitionSegments = [&](TRoutingMappingPtr const & mapping) { - TransitionSegments transitionSegments; - mapping->m_crossContext.ForEachOutgoingNode([&](OutgoingCrossNode const & node) { - FillTransitionSegments(mapping->m_segMapping, node.m_nodeId, numMwmId, node.m_point, - transitionSegments.m_outgoing); - }); - mapping->m_crossContext.ForEachIngoingNode([&](IngoingCrossNode const & node) { - FillTransitionSegments(mapping->m_segMapping, node.m_nodeId, numMwmId, node.m_point, - transitionSegments.m_ingoing); - }); - auto const p = m_transitionCache.emplace(numMwmId, move(transitionSegments)); - UNUSED_VALUE(p); - ASSERT(p.second, ("Mwm num id:", numMwmId, "has been inserted before. Country file name:", - mapping->GetCountryName())); - it = p.first; - }; - - LoadWith(numMwmId, fillAllTransitionSegments); - return it->second; -} - -void CrossMwmOsrmGraph::GetBorderCross(TRoutingMappingPtr const & mapping, Segment const & s, - bool isOutgoing, vector & borderCrosses) -{ - // ingoing edge - NodeIds const nodeIdsTo = GetDirectAndReverseNodeId(mapping->m_segMapping, s); - - vector const & transitionPoints = - isOutgoing ? GetIngoingTransitionPoints(s) : GetOutgoingTransitionPoints(s); - CHECK(!transitionPoints.empty(), ("Segment:", s, ", isOutgoing:", isOutgoing)); - - // If |isOutgoing| == true |nodes| is "to" cross nodes, otherwise |nodes| is "from" cross nodes. - vector nodes; - for (ms::LatLon const & p : transitionPoints) - nodes.emplace_back(nodeIdsTo.m_directNodeId, nodeIdsTo.m_reverseNodeId, mapping->GetMwmId(), p); - - for (CrossNode const & node : nodes) - { - // If |isOutgoing| == true |otherSideNodes| is "from" cross nodes, otherwise - // |otherSideNodes| is "to" cross nodes. - BorderCross bc; - if (isOutgoing) - bc.toNode = node; - else - bc.fromNode = node; - borderCrosses.push_back(bc); - } -} - -CrossMwmOsrmGraph::TransitionSegments const & CrossMwmOsrmGraph::GetSegmentMaps(NumMwmId numMwmId) -{ - auto it = m_transitionCache.find(numMwmId); - if (it == m_transitionCache.cend()) - return LoadSegmentMaps(numMwmId); - return it->second; -} - -vector const & CrossMwmOsrmGraph::GetIngoingTransitionPoints(Segment const & s) -{ - auto const & ingoingSeg = GetSegmentMaps(s.GetMwmId()).m_ingoing; - auto const it = ingoingSeg.find(s); - CHECK(it != ingoingSeg.cend(), ("Segment:", s)); - return it->second; -} - -vector const & CrossMwmOsrmGraph::GetOutgoingTransitionPoints(Segment const & s) -{ - auto const & outgoingSeg = GetSegmentMaps(s.GetMwmId()).m_outgoing; - auto const it = outgoingSeg.find(s); - CHECK(it != outgoingSeg.cend(), ("Segment:", s)); - return it->second; -} -} // namespace routing diff --git a/routing/cross_mwm_osrm_graph.hpp b/routing/cross_mwm_osrm_graph.hpp deleted file mode 100644 index 6a215bedc2..0000000000 --- a/routing/cross_mwm_osrm_graph.hpp +++ /dev/null @@ -1,90 +0,0 @@ -#pragma once - -#include "routing/routing_mapping.hpp" -#include "routing/segment.hpp" -#include "routing/transition_points.hpp" - -#include "routing_common/num_mwm_id.hpp" - -#include "geometry/latlon.hpp" - -#include "platform/country_file.hpp" - -#include -#include -#include - -namespace routing -{ -struct BorderCross; -class CrossMwmRoadGraph; - -using TransitionPoints = buffer_vector; - -class CrossMwmOsrmGraph final -{ -public: - CrossMwmOsrmGraph(std::shared_ptr numMwmIds, RoutingIndexManager & indexManager); - ~CrossMwmOsrmGraph(); - - bool IsTransition(Segment const & s, bool isOutgoing); - void GetEdgeList(Segment const & s, bool isOutgoing, std::vector & edges); - void Clear(); - TransitionPoints GetTransitionPoints(Segment const & s, bool isOutgoing); - -private: - struct TransitionSegments - { - std::map> m_ingoing; - std::map> m_outgoing; - }; - - /// \brief Inserts all ingoing and outgoing transition segments of mwm with |numMwmId| - /// to |m_transitionCache|. It works for OSRM section. - /// \returns Reference to TransitionSegments corresponded to |numMwmId|. - TransitionSegments const & LoadSegmentMaps(NumMwmId numMwmId); - - /// \brief Fills |borderCrosses| of mwm with |mapping| according to |s|. - /// \param mapping if |isOutgoing| == true |mapping| is mapping ingoing (from) border cross. - /// If |isOutgoing| == false |mapping| is mapping outgoing (to) border cross. - /// \note |s| and |isOutgoing| params have the same restrictions which described in - /// GetEdgeList() method. - void GetBorderCross(TRoutingMappingPtr const & mapping, Segment const & s, bool isOutgoing, - std::vector & borderCrosses); - - TransitionSegments const & GetSegmentMaps(NumMwmId numMwmId); - std::vector const & GetIngoingTransitionPoints(Segment const & s); - std::vector const & GetOutgoingTransitionPoints(Segment const & s); - - template - void LoadWith(NumMwmId numMwmId, Fn && fn) - { - platform::CountryFile const & countryFile = m_numMwmIds->GetFile(numMwmId); - TRoutingMappingPtr mapping = m_indexManager.GetMappingByName(countryFile.GetName()); - CHECK(mapping, ("No routing mapping file for countryFile:", countryFile)); - CHECK(mapping->IsValid(), ("Mwm:", countryFile, "was not loaded.")); - - auto const it = m_mappingGuards.find(numMwmId); - if (it == m_mappingGuards.cend()) - { - m_mappingGuards[numMwmId] = make_unique(mapping); - mapping->LoadCrossContext(); - } - - fn(mapping); - } - - std::shared_ptr m_numMwmIds; - - // OSRM based cross-mwm information. - RoutingIndexManager & m_indexManager; - /// \note According to the constructor CrossMwmRoadGraph is initialized with RoutingIndexManager - /// &. - /// But then it is copied by value to CrossMwmRoadGraph::RoutingIndexManager m_indexManager. - /// It means that there're two copies of RoutingIndexManager in CrossMwmGraph. - std::unique_ptr m_crossMwmGraph; - - std::map m_transitionCache; - std::map> m_mappingGuards; -}; -} // namespace routing diff --git a/routing/cross_mwm_road_graph.cpp b/routing/cross_mwm_road_graph.cpp deleted file mode 100644 index 26be45db2d..0000000000 --- a/routing/cross_mwm_road_graph.cpp +++ /dev/null @@ -1,441 +0,0 @@ -#include "routing/cross_mwm_road_graph.hpp" - -#include "routing/cross_mwm_router.hpp" - -#include "geometry/distance_on_sphere.hpp" - -#include - -namespace -{ -using namespace routing; - -inline bool IsValidEdgeWeight(EdgeWeight const & w) { return w != INVALID_EDGE_WEIGHT; } -template -class ClosestNodeFinder -{ -public: - ClosestNodeFinder(CrossNode const & node, double & minDistance, Node & resultingCrossNode) - : m_node(node), m_minDistance(minDistance), m_resultingCrossNode(resultingCrossNode) - { - } - - void operator()(Node const & crossNode) const - { - if (crossNode.m_nodeId != m_node.node) - return; - - double const dist = ms::DistanceOnEarth(m_node.point, crossNode.m_point); - if (dist < m_minDistance) - { - m_minDistance = dist; - m_resultingCrossNode = crossNode; - } - } - -private: - CrossNode const & m_node; - double & m_minDistance; - Node & m_resultingCrossNode; -}; - -double GetAdjacencyCost(CrossRoutingContextReader const & currentContext, - IngoingCrossNode const & ingoingCrossNode, - OutgoingCrossNode const & outgoingCrossNode) -{ - return currentContext.GetAdjacencyCost(ingoingCrossNode, outgoingCrossNode); -} - -double GetAdjacencyCost(CrossRoutingContextReader const & currentContext, - OutgoingCrossNode const & outgoingCrossNode, - IngoingCrossNode const & ingoingCrossNode) -{ - return GetAdjacencyCost(currentContext, ingoingCrossNode, outgoingCrossNode); -} - -template -class EdgesFiller -{ -public: - EdgesFiller(TRoutingMappingPtr const & currentMapping, - CrossRoutingContextReader const & currentContext, SourceNode const & startingNode, - CrossMwmRoadGraph const & crossMwmGraph, vector & adj) - : m_currentMapping(currentMapping) - , m_currentContext(currentContext) - , m_startingNode(startingNode) - , m_crossMwmGraph(crossMwmGraph) - , m_adj(adj) - { - } - - void operator()(TargetNode const & node) const - { - TWrittenEdgeWeight const outWeight = GetAdjacencyCost(m_currentContext, m_startingNode, node); - if (outWeight != kInvalidContextEdgeWeight && outWeight != 0) - { - vector const & targets = - m_crossMwmGraph.ConstructBorderCross(m_currentMapping, node); - for (auto const & target : targets) - { - if (target.toNode.IsValid()) - m_adj.emplace_back(target, outWeight); - } - } - } - -private: - TRoutingMappingPtr const & m_currentMapping; - CrossRoutingContextReader const & m_currentContext; - SourceNode const & m_startingNode; - CrossMwmRoadGraph const & m_crossMwmGraph; - vector & m_adj; -}; - -bool ForEachNodeNearPoint(CrossRoutingContextReader const & currentContext, - ms::LatLon const & point, - ClosestNodeFinder const & findingNode) -{ - return currentContext.ForEachIngoingNodeNearPoint(point, findingNode); -} - -bool ForEachNodeNearPoint(CrossRoutingContextReader const & currentContext, - ms::LatLon const & point, - ClosestNodeFinder const & findingNode) -{ - return currentContext.ForEachOutgoingNodeNearPoint(point, findingNode); -} - -template -bool FindCrossNode(CrossRoutingContextReader const & currentContext, CrossNode const & crossNode, - Node & node) -{ - double constexpr kInvalidDistance = std::numeric_limits::max(); - double minDistance = kInvalidDistance; - ClosestNodeFinder findingNode(crossNode, minDistance, node); - CHECK(ForEachNodeNearPoint(currentContext, crossNode.point, findingNode), ()); - if (minDistance == kInvalidDistance) - { - LOG(LWARNING, ("Cross node is not found. Point:", crossNode.point)); - return false; - } - return true; -} - -template -vector const & ConstructBorderCrossImpl( - TWrittenNodeId nodeId, TRoutingMappingPtr const & currentMapping, - unordered_map, - CrossMwmRoadGraph::Hash> const & cachedNextNodes, - Fn && borderCrossConstructor) -{ - auto const key = make_pair(nodeId, currentMapping->GetMwmId()); - auto const it = cachedNextNodes.find(key); - if (it != cachedNextNodes.end()) - return it->second; - borderCrossConstructor(key); - return cachedNextNodes.find(key)->second; -} -} // namespace - -namespace routing -{ -IRouter::ResultCode CrossMwmRoadGraph::SetStartNode(CrossNode const & startNode) -{ - ASSERT(startNode.mwmId.IsAlive(), ()); - // TODO (ldragunov) make cancellation if necessary - TRoutingMappingPtr startMapping = m_indexManager.GetMappingById(startNode.mwmId); - if (!startMapping->IsValid()) - return IRouter::ResultCode::StartPointNotFound; - MappingGuard startMappingGuard(startMapping); - UNUSED_VALUE(startMappingGuard); - startMapping->LoadCrossContext(); - - // Load source data. - vector outgoingNodes; - - startMapping->m_crossContext.ForEachOutgoingNode([&outgoingNodes](OutgoingCrossNode const & node) - { - outgoingNodes.push_back(node); - }); - - size_t const outSize = outgoingNodes.size(); - // Can't find the route if there are no routes outside source map. - if (!outSize) - return IRouter::RouteNotFound; - - // Generate routing task from one source to several targets. - TRoutingNodes sources(1), targets; - targets.reserve(outSize); - for (auto const & node : outgoingNodes) - { - targets.emplace_back(node.m_nodeId, false /* isStartNode */, startNode.mwmId); - targets.back().segmentPoint = MercatorBounds::FromLatLon(node.m_point); - } - sources[0] = FeatureGraphNode(startNode.node, startNode.reverseNode, true /* isStartNode */, - startNode.mwmId); - - vector weights; - FindWeightsMatrix(sources, targets, startMapping->m_dataFacade, weights); - if (find_if(weights.begin(), weights.end(), &IsValidEdgeWeight) == weights.end()) - return IRouter::StartPointNotFound; - vector dummyEdges; - for (size_t i = 0; i < outSize; ++i) - { - if (IsValidEdgeWeight(weights[i])) - { - vector const & nextCrosses = - ConstructBorderCross(startMapping, outgoingNodes[i]); - - for (auto const & nextCross : nextCrosses) - { - if (nextCross.toNode.IsValid()) - dummyEdges.emplace_back(nextCross, weights[i]); - } - } - } - - m_virtualEdges.insert(make_pair(startNode, dummyEdges)); - return IRouter::NoError; -} - -void CrossMwmRoadGraph::AddVirtualEdge(IngoingCrossNode const & node, CrossNode const & finalNode, - EdgeWeight weight) -{ - CrossNode start(node.m_nodeId, finalNode.mwmId, node.m_point); - vector dummyEdges; - dummyEdges.emplace_back(BorderCross(finalNode, finalNode), weight); - m_virtualEdges.insert(make_pair(start, dummyEdges)); -} - -IRouter::ResultCode CrossMwmRoadGraph::SetFinalNode(CrossNode const & finalNode) -{ - ASSERT(finalNode.mwmId.IsAlive(), ()); - TRoutingMappingPtr finalMapping = m_indexManager.GetMappingById(finalNode.mwmId); - if (!finalMapping->IsValid()) - return IRouter::ResultCode::EndPointNotFound; - MappingGuard finalMappingGuard(finalMapping); - UNUSED_VALUE(finalMappingGuard); - finalMapping->LoadCrossContext(); - - // Load source data. - vector ingoingNodes; - finalMapping->m_crossContext.ForEachIngoingNode([&ingoingNodes](IngoingCrossNode const & node) - { - ingoingNodes.push_back(node); - }); - size_t const ingoingSize = ingoingNodes.size(); - // If there is no routes inside target map. - if (ingoingSize == 0) - return IRouter::RouteNotFound; - - // Generate routing task from one source to several targets. - TRoutingNodes sources, targets(1); - sources.reserve(ingoingSize); - - for (auto const & node : ingoingNodes) - { - // Case with a target node at the income mwm node. - if (node.m_nodeId == finalNode.node) - { - AddVirtualEdge(node, finalNode, 0 /* no weight */); - return IRouter::NoError; - } - sources.emplace_back(node.m_nodeId, true /* isStartNode */, finalNode.mwmId); - sources.back().segmentPoint = MercatorBounds::FromLatLon(node.m_point); - } - vector weights; - - targets[0] = FeatureGraphNode(finalNode.node, finalNode.reverseNode, false /* isStartNode */, - finalNode.mwmId); - FindWeightsMatrix(sources, targets, finalMapping->m_dataFacade, weights); - if (find_if(weights.begin(), weights.end(), &IsValidEdgeWeight) == weights.end()) - return IRouter::EndPointNotFound; - for (size_t i = 0; i < ingoingSize; ++i) - { - if (IsValidEdgeWeight(weights[i])) - { - AddVirtualEdge(ingoingNodes[i], finalNode, weights[i]); - } - } - return IRouter::NoError; -} - -bool CrossMwmRoadGraph::ConstructBorderCrossByOutgoingImpl( - OutgoingCrossNode const & startNode, TRoutingMappingPtr const & currentMapping, - vector & crosses) const -{ - auto const fromCross = CrossNode(startNode.m_nodeId, currentMapping->GetMwmId(), startNode.m_point); - string const & nextMwm = currentMapping->m_crossContext.GetOutgoingMwmName(startNode); - TRoutingMappingPtr nextMapping = m_indexManager.GetMappingByName(nextMwm); - // If we haven't this routing file, we skip this path. - if (!nextMapping->IsValid()) - return false; - ASSERT(crosses.empty(), ()); - nextMapping->LoadCrossContext(); - nextMapping->m_crossContext.ForEachIngoingNodeNearPoint( - startNode.m_point, [&](IngoingCrossNode const & node) { - if (node.m_nodeId == INVALID_NODE_ID) - return; - - if (!node.m_point.EqualDxDy( - startNode.m_point, kMwmCrossingNodeEqualityMeters * MercatorBounds::degreeInMetres)) - return; - - crosses.emplace_back(fromCross, - CrossNode(node.m_nodeId, nextMapping->GetMwmId(), node.m_point)); - }); - return !crosses.empty(); -} - -bool CrossMwmRoadGraph::ConstructBorderCrossByIngoingImpl(IngoingCrossNode const & startNode, - TRoutingMappingPtr const & currentMapping, - vector & crosses) const -{ - ASSERT(crosses.empty(), ()); - auto const toCross = CrossNode(startNode.m_nodeId, currentMapping->GetMwmId(), startNode.m_point); - vector const & neighboringMwms = currentMapping->m_crossContext.GetNeighboringMwmList(); - string const & currentMwm = currentMapping->GetMwmId().GetInfo()->GetCountryName(); - // Note. There's no field |m_ingoingIndex| in class IngoingCrossNode. Because this - // index is not saved in osrm routing section. So we need to write a workaround and - // to check all neighboring mwms. - for (string const & prevMwm : neighboringMwms) - { - TRoutingMappingPtr prevMapping = m_indexManager.GetMappingByName(prevMwm); - if (!prevMapping->IsValid()) - continue; - - prevMapping->LoadCrossContext(); - prevMapping->m_crossContext.ForEachOutgoingNodeNearPoint( - startNode.m_point, [&](OutgoingCrossNode const & node) { - if (node.m_nodeId == INVALID_NODE_ID) - return; - - if (prevMapping->m_crossContext.GetOutgoingMwmName(node) != currentMwm) - return; - - if (!node.m_point.EqualDxDy(startNode.m_point, kMwmCrossingNodeEqualityMeters * - MercatorBounds::degreeInMetres)) - return; - - crosses.emplace_back(CrossNode(node.m_nodeId, prevMapping->GetMwmId(), node.m_point), - toCross); - }); - } - return !crosses.empty(); -} - -vector const & CrossMwmRoadGraph::ConstructBorderCross( - TRoutingMappingPtr const & currentMapping, OutgoingCrossNode const & node) const -{ - return ConstructBorderCrossImpl(node.m_nodeId, currentMapping, m_cachedNextNodesByOutgoing, - [&](std::pair const & key) { - vector crosses; - ConstructBorderCrossByOutgoingImpl(node, currentMapping, - crosses); - m_cachedNextNodesByOutgoing[key] = move(crosses); - }); -} - -vector const & CrossMwmRoadGraph::ConstructBorderCross( - TRoutingMappingPtr const & currentMapping, IngoingCrossNode const & node) const -{ - return ConstructBorderCrossImpl(node.m_nodeId, currentMapping, m_cachedNextNodesByIngoing, - [&](std::pair const & key) { - vector crosses; - ConstructBorderCrossByIngoingImpl(node, currentMapping, - crosses); - m_cachedNextNodesByIngoing[key] = move(crosses); - }); -} - -void CrossMwmRoadGraph::GetEdgesList(BorderCross const & v, bool isOutgoing, - vector & adj) const -{ - // Check for virtual edges. - adj.clear(); - - // Note. Code below processes virtual edges. This code does not work properly if isOutgoing == - // false. - // At the same time when this method is called with isOutgoing == false |m_virtualEdges| is empty. - if (!isOutgoing && !m_virtualEdges.empty()) - { - NOTIMPLEMENTED(); - return; - } - - auto const it = m_virtualEdges.find(v.toNode); - if (it != m_virtualEdges.end()) - { - adj.insert(adj.end(), it->second.begin(), it->second.end()); - // For last map we need to load virtual shortcuts and real cross roads. It takes to account case - // when we have a path from the mwm border to the point inside the map throuh another map. - // See Ust-Katav test for more. - if (it->second.empty() || !it->second.front().GetTarget().toNode.isVirtual) - return; - } - - // Loading cross routing section. - TRoutingMappingPtr currentMapping = - m_indexManager.GetMappingById(isOutgoing ? v.toNode.mwmId : v.fromNode.mwmId); - ASSERT(currentMapping->IsValid(), ()); - currentMapping->LoadCrossContext(); - currentMapping->FreeFileIfPossible(); - - CrossRoutingContextReader const & currentContext = currentMapping->m_crossContext; - - if (isOutgoing) - { - IngoingCrossNode ingoingNode; - if (!FindCrossNode(currentContext, v.toNode, ingoingNode)) - return; - if (!ingoingNode.IsValid()) - return; - - currentContext.ForEachOutgoingNode(EdgesFiller( - currentMapping, currentContext, ingoingNode, *this, adj)); - } - else - { - OutgoingCrossNode outgoingNode; - if (!FindCrossNode(currentContext, v.fromNode, outgoingNode)) - return; - if (!outgoingNode.IsValid()) - return; - - currentContext.ForEachIngoingNode(EdgesFiller( - currentMapping, currentContext, outgoingNode, *this, adj)); - } -} - -double CrossMwmRoadGraph::HeuristicCostEstimate(BorderCross const & v, BorderCross const & w) const -{ - // Simple travel time heuristic works worse than simple Dijkstra's algorithm, represented by - // always 0 heuristics estimation. - return 0; -} - -void ConvertToSingleRouterTasks(vector const & graphCrosses, - FeatureGraphNode const & startGraphNode, - FeatureGraphNode const & finalGraphNode, TCheckedPath & route) -{ - route.clear(); - for (size_t i = 0; i + 1 < graphCrosses.size(); ++i) - { - ASSERT_EQUAL(graphCrosses[i].toNode.mwmId, graphCrosses[i + 1].fromNode.mwmId, ()); - route.emplace_back(graphCrosses[i].toNode.node, - MercatorBounds::FromLatLon(graphCrosses[i].toNode.point), - graphCrosses[i + 1].fromNode.node, - MercatorBounds::FromLatLon(graphCrosses[i + 1].fromNode.point), - graphCrosses[i].toNode.mwmId); - } - - if (route.empty()) - return; - - route.front().startNode = startGraphNode; - route.back().finalNode = finalGraphNode; - ASSERT_EQUAL(route.front().startNode.mwmId, route.front().finalNode.mwmId, ()); - ASSERT_EQUAL(route.back().startNode.mwmId, route.back().finalNode.mwmId, ()); -} - -} // namespace routing diff --git a/routing/cross_mwm_road_graph.hpp b/routing/cross_mwm_road_graph.hpp deleted file mode 100644 index 0b9d3bd741..0000000000 --- a/routing/cross_mwm_road_graph.hpp +++ /dev/null @@ -1,201 +0,0 @@ -#pragma once - -#include "routing/cross_mwm_router.hpp" -#include "routing/osrm_engine.hpp" -#include "routing/router.hpp" - -#include "indexer/index.hpp" - -#include "geometry/latlon.hpp" -#include "geometry/point2d.hpp" - -#include "base/macros.hpp" - -#include "std/functional.hpp" -#include "std/unordered_map.hpp" - -namespace routing -{ -/// OSRM graph node representation with graph mwm name and border crossing point. -struct CrossNode final -{ - NodeID node; - NodeID reverseNode; - Index::MwmId mwmId; - ms::LatLon point; - bool isVirtual; - - CrossNode(NodeID node, NodeID reverse, Index::MwmId const & id, ms::LatLon const & point) - : node(node), reverseNode(reverse), mwmId(id), point(point), isVirtual(false) - { - } - - CrossNode(NodeID node, Index::MwmId const & id, ms::LatLon const & point) - : node(node), reverseNode(INVALID_NODE_ID), mwmId(id), point(point), isVirtual(false) - { - } - - CrossNode() - : node(INVALID_NODE_ID) - , reverseNode(INVALID_NODE_ID) - , point(ms::LatLon::Zero()) - , isVirtual(false) - { - } - - inline bool IsValid() const { return node != INVALID_NODE_ID; } - - inline bool operator==(CrossNode const & a) const - { - return node == a.node && mwmId == a.mwmId && isVirtual == a.isVirtual; - } - - inline bool operator<(CrossNode const & a) const - { - if (a.node != node) - return node < a.node; - - if (isVirtual != a.isVirtual) - return isVirtual < a.isVirtual; - - return mwmId < a.mwmId; - } -}; - -inline string DebugPrint(CrossNode const & t) -{ - ostringstream out; - out << "CrossNode [ node: " << t.node << ", map: " << t.mwmId.GetInfo()->GetCountryName()<< " ]"; - return out.str(); -} - -/// Representation of border crossing. Contains node on previous map and node on next map. -struct BorderCross final -{ - CrossNode fromNode; - CrossNode toNode; - - BorderCross(CrossNode const & from, CrossNode const & to) : fromNode(from), toNode(to) {} - BorderCross() = default; - - // TODO(bykoianko) Consider using fields |fromNode| and |toNode| in operator== and operator<. - inline bool operator==(BorderCross const & a) const { return toNode == a.toNode; } - inline bool operator!=(BorderCross const & a) const { return !(*this == a); } - inline bool operator<(BorderCross const & a) const { return toNode < a.toNode; } -}; - -inline string DebugPrint(BorderCross const & t) -{ - ostringstream out; - out << "Border cross from: " << DebugPrint(t.fromNode) << " to: " << DebugPrint(t.toNode) << "\n"; - return out.str(); -} - -/// A class which represents an cross mwm weighted edge used by CrossMwmRoadGraph. -class CrossWeightedEdge final -{ -public: - CrossWeightedEdge(BorderCross const & target, double weight) : m_target(target), m_weight(weight) - { - } - - BorderCross const & GetTarget() const { return m_target; } - double GetWeight() const { return m_weight; } - - bool operator==(CrossWeightedEdge const & a) const - { - return m_target == a.m_target && m_weight == a.m_weight; - } - - bool operator<(CrossWeightedEdge const & a) const - { - if (m_target != a.m_target) - return m_target < a.m_target; - return m_weight < a.m_weight; - } - -private: - BorderCross m_target; - double m_weight; -}; - -/// A graph used for cross mwm routing in an astar algorithms. -class CrossMwmRoadGraph final -{ -public: - using CachingKey = pair; - using Vertex = BorderCross; - using Edge = CrossWeightedEdge; - using Weight = double; - - struct Hash - { - size_t operator()(CachingKey const & p) const - { - return hash()(p.first) ^ hash()(p.second.GetInfo()->GetCountryName()); - } - }; - - explicit CrossMwmRoadGraph(RoutingIndexManager & indexManager) : m_indexManager(indexManager) {} - void GetOutgoingEdgesList(BorderCross const & v, vector & adj) const - { - GetEdgesList(v, true /* isOutgoing */, adj); - } - - void GetIngoingEdgesList(BorderCross const & v, vector & adj) const - { - GetEdgesList(v, false /* isOutgoing */, adj); - } - - double HeuristicCostEstimate(BorderCross const & v, BorderCross const & w) const; - - IRouter::ResultCode SetStartNode(CrossNode const & startNode); - IRouter::ResultCode SetFinalNode(CrossNode const & finalNode); - - vector const & ConstructBorderCross(TRoutingMappingPtr const & currentMapping, - OutgoingCrossNode const & node) const; - vector const & ConstructBorderCross(TRoutingMappingPtr const & currentMapping, - IngoingCrossNode const & node) const; - -private: - // Pure function to construct boder cross by outgoing cross node. - bool ConstructBorderCrossByOutgoingImpl(OutgoingCrossNode const & startNode, - TRoutingMappingPtr const & currentMapping, - vector & cross) const; - bool ConstructBorderCrossByIngoingImpl(IngoingCrossNode const & startNode, - TRoutingMappingPtr const & currentMapping, - vector & crosses) const; - - /*! - * Adds a virtual edge to the graph so that it is possible to represent - * the final segment of the path that leads from the map's border - * to finalNode. Addition of such virtual edges for the starting node is - * inlined elsewhere. - */ - void AddVirtualEdge(IngoingCrossNode const & node, CrossNode const & finalNode, - EdgeWeight weight); - void GetEdgesList(BorderCross const & v, bool isOutgoing, vector & adj) const; - - map > m_virtualEdges; - - mutable RoutingIndexManager m_indexManager; - - // @TODO(bykoianko) Consider removing key work mutable. - mutable unordered_map, Hash> m_cachedNextNodesByIngoing; - mutable unordered_map, Hash> m_cachedNextNodesByOutgoing; -}; - -//-------------------------------------------------------------------------------------------------- -// Helper functions. -//-------------------------------------------------------------------------------------------------- - -/*! - * \brief Convertor from CrossMwmRoadGraph to cross mwm route task. - * \warning It's assumed that the first and the last BorderCrosses are always virtual and represents - * routing inside mwm. - */ -void ConvertToSingleRouterTasks(vector const & graphCrosses, - FeatureGraphNode const & startGraphNode, - FeatureGraphNode const & finalGraphNode, TCheckedPath & route); - -} // namespace routing diff --git a/routing/cross_mwm_router.cpp b/routing/cross_mwm_router.cpp deleted file mode 100644 index 2ceaa7091d..0000000000 --- a/routing/cross_mwm_router.cpp +++ /dev/null @@ -1,107 +0,0 @@ -#include "routing/cross_mwm_router.hpp" - -#include "routing/base/astar_algorithm.hpp" -#include "routing/base/routing_result.hpp" -#include "routing/cross_mwm_road_graph.hpp" - -#include "base/timer.hpp" - -namespace routing -{ - -namespace -{ -/// Function to run AStar Algorithm from the base. -IRouter::ResultCode CalculateRoute(BorderCross const & startPos, BorderCross const & finalPos, - CrossMwmRoadGraph & roadGraph, RouterDelegate const & delegate, - RoutingResult & route) -{ - using Algorithm = AStarAlgorithm; - - Algorithm::OnVisitedVertexCallback onVisitedVertex = - [&delegate](BorderCross const & cross, BorderCross const & /* target */) { - delegate.OnPointCheck(MercatorBounds::FromLatLon(cross.fromNode.point)); - }; - - Algorithm::Params params(roadGraph, startPos, finalPos, nullptr /* prevRoute */, delegate, - onVisitedVertex, {} /* checkLengthCallback */); - - my::HighResTimer timer(true); - Algorithm::Result const result = Algorithm().FindPath(params, route); - LOG(LINFO, ("Duration of the cross MWM path finding", timer.ElapsedNano())); - switch (result) - { - case Algorithm::Result::OK: - ASSERT_EQUAL(route.m_path.front(), startPos, ()); - ASSERT_EQUAL(route.m_path.back(), finalPos, ()); - return IRouter::NoError; - case Algorithm::Result::NoPath: return IRouter::RouteNotFound; - case Algorithm::Result::Cancelled: return IRouter::Cancelled; - } - return IRouter::RouteNotFound; -} -} // namespace - -IRouter::ResultCode CalculateCrossMwmPath(TRoutingNodes const & startGraphNodes, - TRoutingNodes const & finalGraphNodes, - RoutingIndexManager & indexManager, - double & cost, - RouterDelegate const & delegate, TCheckedPath & route) -{ - CrossMwmRoadGraph roadGraph(indexManager); - FeatureGraphNode startGraphNode, finalGraphNode; - CrossNode startNode, finalNode; - - // Finding start node. - IRouter::ResultCode code = IRouter::StartPointNotFound; - for (FeatureGraphNode const & start : startGraphNodes) - { - startNode = CrossNode(start.node.forward_node_id, start.node.reverse_node_id, start.mwmId, - MercatorBounds::ToLatLon(start.segmentPoint)); - code = roadGraph.SetStartNode(startNode); - if (code == IRouter::NoError) - { - startGraphNode = start; - break; - } - if (delegate.IsCancelled()) - return IRouter::Cancelled; - } - if (code != IRouter::NoError) - return IRouter::StartPointNotFound; - - // Finding final node. - code = IRouter::EndPointNotFound; - for (FeatureGraphNode const & final : finalGraphNodes) - { - finalNode = CrossNode(final.node.reverse_node_id, final.node.forward_node_id, final.mwmId, - MercatorBounds::ToLatLon(final.segmentPoint)); - finalNode.isVirtual = true; - code = roadGraph.SetFinalNode(finalNode); - if (code == IRouter::NoError) - { - finalGraphNode = final; - break; - } - if (delegate.IsCancelled()) - return IRouter::Cancelled; - } - if (code != IRouter::NoError) - return IRouter::EndPointNotFound; - - // Finding path through maps. - RoutingResult tempRoad; - code = CalculateRoute({startNode, startNode}, {finalNode, finalNode}, roadGraph, delegate, tempRoad); - cost = tempRoad.m_distance; - if (code != IRouter::NoError) - return code; - if (delegate.IsCancelled()) - return IRouter::Cancelled; - - // Final path conversion to output type. - ConvertToSingleRouterTasks(tempRoad.m_path, startGraphNode, finalGraphNode, route); - - return route.empty() ? IRouter::RouteNotFound : IRouter::NoError; -} - -} // namespace routing diff --git a/routing/cross_mwm_router.hpp b/routing/cross_mwm_router.hpp deleted file mode 100644 index c5b19780f9..0000000000 --- a/routing/cross_mwm_router.hpp +++ /dev/null @@ -1,45 +0,0 @@ -#pragma once - -#include "routing/osrm_engine.hpp" -#include "routing/router.hpp" -#include "routing/routing_mapping.hpp" - -#include "std/string.hpp" -#include "std/vector.hpp" - -namespace routing -{ -/*! - * \brief The RoutePathCross struct contains information neaded to describe path inside single map. - */ -struct RoutePathCross -{ - FeatureGraphNode startNode; /**< start graph node representation */ - FeatureGraphNode finalNode; /**< end graph node representation */ - - RoutePathCross(NodeID const startNodeId, m2::PointD const & startPoint, NodeID const finalNodeId, m2::PointD const & finalPoint, Index::MwmId const & id) - : startNode(startNodeId, true /* isStartNode */, id), - finalNode(finalNodeId, false /* isStartNode*/, id) - { - startNode.segmentPoint = startPoint; - finalNode.segmentPoint = finalPoint; - } -}; - -using TCheckedPath = vector; - -/*! - * \brief CalculateCrossMwmPath function for calculating path through several maps. - * \param startGraphNodes The vector of starting routing graph nodes. - * \param finalGraphNodes The vector of final routing graph nodes. - * \param route Storage for the result records about crossing maps. - * \param indexManager Manager for getting indexes of new countries. - * \param cost Found path cost. - * \param RoutingVisualizerFn Debug visualization function. - * \return NoError if the path exists, error code otherwise. - */ -IRouter::ResultCode CalculateCrossMwmPath(TRoutingNodes const & startGraphNodes, - TRoutingNodes const & finalGraphNodes, - RoutingIndexManager & indexManager, double & cost, - RouterDelegate const & delegate, TCheckedPath & route); -} // namespace routing diff --git a/routing/cross_routing_context.cpp b/routing/cross_routing_context.cpp deleted file mode 100644 index 312fbde201..0000000000 --- a/routing/cross_routing_context.cpp +++ /dev/null @@ -1,196 +0,0 @@ -#include "routing/cross_routing_context.hpp" - -#include "coding/point_to_integer.hpp" - -namespace -{ -uint32_t constexpr kCoordBits = POINT_COORD_BITS; -} // namespace - -namespace routing -{ -void OutgoingCrossNode::Save(Writer & w) const -{ - uint64_t point = PointToInt64(m2::PointD(m_point.lon, m_point.lat), kCoordBits); - char buff[sizeof(m_nodeId) + sizeof(point) + sizeof(m_outgoingIndex)]; - *reinterpret_cast(&buff[0]) = m_nodeId; - *reinterpret_cast(&(buff[sizeof(m_nodeId)])) = point; - *reinterpret_cast(&(buff[sizeof(m_nodeId) + sizeof(point)])) = m_outgoingIndex; - w.Write(buff, sizeof(buff)); -} - -size_t OutgoingCrossNode::Load(const Reader & r, size_t pos, size_t adjacencyIndex) -{ - char buff[sizeof(m_nodeId) + sizeof(uint64_t) + sizeof(m_outgoingIndex)]; - r.Read(pos, buff, sizeof(buff)); - m_nodeId = *reinterpret_cast(&buff[0]); - m2::PointD bufferPoint = Int64ToPoint(*reinterpret_cast(&(buff[sizeof(m_nodeId)])), kCoordBits); - m_point = ms::LatLon(bufferPoint.y, bufferPoint.x); - m_outgoingIndex = *reinterpret_cast(&(buff[sizeof(m_nodeId) + sizeof(uint64_t)])); - m_adjacencyIndex = adjacencyIndex; - return pos + sizeof(buff); -} - -void IngoingCrossNode::Save(Writer & w) const -{ - uint64_t point = PointToInt64(m2::PointD(m_point.lon, m_point.lat), kCoordBits); - char buff[sizeof(m_nodeId) + sizeof(point)]; - *reinterpret_cast(&buff[0]) = m_nodeId; - *reinterpret_cast(&(buff[sizeof(m_nodeId)])) = point; - w.Write(buff, sizeof(buff)); -} - -size_t IngoingCrossNode::Load(const Reader & r, size_t pos, size_t adjacencyIndex) -{ - char buff[sizeof(m_nodeId) + sizeof(uint64_t)]; - r.Read(pos, buff, sizeof(buff)); - m_nodeId = *reinterpret_cast(&buff[0]); - m2::PointD bufferPoint = Int64ToPoint(*reinterpret_cast(&(buff[sizeof(m_nodeId)])), kCoordBits); - m_point = ms::LatLon(bufferPoint.y, bufferPoint.x); - m_adjacencyIndex = adjacencyIndex; - return pos + sizeof(buff); -} - -void CrossRoutingContextReader::Load(Reader const & r) -{ - size_t pos = 0; - - uint32_t size, ingoingSize; - r.Read(pos, &ingoingSize, sizeof(ingoingSize)); - pos += sizeof(ingoingSize); - - for (size_t i = 0; i < ingoingSize; ++i) - { - IngoingCrossNode node; - pos = node.Load(r, pos, i); - m_ingoingIndex.Add(node); - } - - r.Read(pos, &size, sizeof(size)); - pos += sizeof(size); - m_outgoingNodes.resize(size); - - for (size_t i = 0; i < size; ++i) - { - pos = m_outgoingNodes[i].Load(r, pos, i); - m_outgoingIndex.Add(m_outgoingNodes[i]); - } - - size_t adjacencySize = ingoingSize * m_outgoingNodes.size(); - size_t const adjMatrixSize = sizeof(TWrittenEdgeWeight) * adjacencySize; - m_adjacencyMatrix.resize(adjacencySize); - r.Read(pos, &m_adjacencyMatrix[0], adjMatrixSize); - pos += adjMatrixSize; - - uint32_t strsize; - r.Read(pos, &strsize, sizeof(strsize)); - pos += sizeof(strsize); - for (uint32_t i = 0; i < strsize; ++i) - { - r.Read(pos, &size, sizeof(size)); - pos += sizeof(size); - vector buffer(size); - r.Read(pos, &buffer[0], size); - m_neighborMwmList.push_back(string(&buffer[0], size)); - pos += size; - } -} - -const string & CrossRoutingContextReader::GetOutgoingMwmName( - OutgoingCrossNode const & outgoingNode) const -{ - ASSERT(outgoingNode.m_outgoingIndex < m_neighborMwmList.size(), - ("Routing context out of size mwm name index:", outgoingNode.m_outgoingIndex, - m_neighborMwmList.size())); - return m_neighborMwmList[outgoingNode.m_outgoingIndex]; -} - -TWrittenEdgeWeight CrossRoutingContextReader::GetAdjacencyCost( - IngoingCrossNode const & ingoing, OutgoingCrossNode const & outgoing) const -{ - if (ingoing.m_adjacencyIndex == kInvalidAdjacencyIndex || - outgoing.m_adjacencyIndex == kInvalidAdjacencyIndex) - return kInvalidContextEdgeWeight; - - size_t cost_index = m_outgoingNodes.size() * ingoing.m_adjacencyIndex + outgoing.m_adjacencyIndex; - return cost_index < m_adjacencyMatrix.size() ? m_adjacencyMatrix[cost_index] : kInvalidContextEdgeWeight; -} - -m2::RectD CrossRoutingContextReader::GetMwmCrossingNodeEqualityRect(ms::LatLon const & point) const -{ - double constexpr kMwmCrossingNodeEqualityDegrees = kMwmCrossingNodeEqualityMeters * MercatorBounds::degreeInMetres; - return m2::RectD(point.lat - kMwmCrossingNodeEqualityDegrees, - point.lon - kMwmCrossingNodeEqualityDegrees, - point.lat + kMwmCrossingNodeEqualityDegrees, - point.lon + kMwmCrossingNodeEqualityDegrees); -} - -void CrossRoutingContextWriter::Save(Writer & w) const -{ - uint32_t size = static_cast(m_ingoingNodes.size()); - w.Write(&size, sizeof(size)); - for (auto const & node : m_ingoingNodes) - node.Save(w); - - size = static_cast(m_outgoingNodes.size()); - w.Write(&size, sizeof(size)); - - for (auto const & node : m_outgoingNodes) - node.Save(w); - - CHECK(m_adjacencyMatrix.size() == m_outgoingNodes.size() * m_ingoingNodes.size(), ()); - w.Write(&m_adjacencyMatrix[0], sizeof(m_adjacencyMatrix[0]) * m_adjacencyMatrix.size()); - - size = static_cast(m_neighborMwmList.size()); - w.Write(&size, sizeof(size)); - for (string const & neighbor : m_neighborMwmList) - { - size = static_cast(neighbor.size()); - w.Write(&size, sizeof(size)); - w.Write(neighbor.c_str(), neighbor.size()); - } -} - -void CrossRoutingContextWriter::AddIngoingNode(TWrittenNodeId const nodeId, ms::LatLon const & point) -{ - size_t const adjIndex = m_ingoingNodes.size(); - m_ingoingNodes.emplace_back(nodeId, point, adjIndex); -} - -void CrossRoutingContextWriter::AddOutgoingNode(TWrittenNodeId const nodeId, string const & targetMwm, - ms::LatLon const & point) -{ - size_t const adjIndex = m_outgoingNodes.size(); - auto it = find(m_neighborMwmList.begin(), m_neighborMwmList.end(), targetMwm); - if (it == m_neighborMwmList.end()) - it = m_neighborMwmList.insert(m_neighborMwmList.end(), targetMwm); - m_outgoingNodes.emplace_back(nodeId, distance(m_neighborMwmList.begin(), it), point, adjIndex); -} - -void CrossRoutingContextWriter::ReserveAdjacencyMatrix() -{ - m_adjacencyMatrix.resize(m_ingoingNodes.size() * m_outgoingNodes.size(), - kInvalidContextEdgeWeight); -} - -void CrossRoutingContextWriter::SetAdjacencyCost(IngoingEdgeIteratorT ingoing, - OutgoingEdgeIteratorT outgoing, - TWrittenEdgeWeight value) -{ - size_t const index = m_outgoingNodes.size() * ingoing->m_adjacencyIndex + outgoing->m_adjacencyIndex; - ASSERT_LESS(index, m_adjacencyMatrix.size(), ()); - m_adjacencyMatrix[index] = value; -} - -pair CrossRoutingContextWriter::GetIngoingIterators() - const -{ - return make_pair(m_ingoingNodes.cbegin(), m_ingoingNodes.cend()); -} - -pair CrossRoutingContextWriter::GetOutgoingIterators() - const -{ - return make_pair(m_outgoingNodes.cbegin(), m_outgoingNodes.cend()); -} -} diff --git a/routing/cross_routing_context.hpp b/routing/cross_routing_context.hpp deleted file mode 100644 index 69be60be80..0000000000 --- a/routing/cross_routing_context.hpp +++ /dev/null @@ -1,173 +0,0 @@ -#pragma once - -#include "coding/file_container.hpp" - -#include "geometry/latlon.hpp" -#include "geometry/mercator.hpp" -#include "geometry/point2d.hpp" -#include "geometry/rect2d.hpp" -#include "geometry/tree4d.hpp" - -#include "std/function.hpp" -#include "std/string.hpp" -#include "std/vector.hpp" - -namespace routing -{ -using TWrittenNodeId = uint32_t; -using TWrittenEdgeWeight = uint32_t; - -TWrittenEdgeWeight constexpr kInvalidContextEdgeNodeId = std::numeric_limits::max(); -TWrittenEdgeWeight constexpr kInvalidContextEdgeWeight = std::numeric_limits::max(); -size_t constexpr kInvalidAdjacencyIndex = numeric_limits::max(); -double constexpr kMwmCrossingNodeEqualityMeters = 80; - -struct IngoingCrossNode -{ - ms::LatLon m_point; - TWrittenNodeId m_nodeId; - size_t m_adjacencyIndex; - - IngoingCrossNode() - : m_point(ms::LatLon::Zero()) - , m_nodeId(kInvalidContextEdgeNodeId) - , m_adjacencyIndex(kInvalidAdjacencyIndex) - { - } - - IngoingCrossNode(TWrittenNodeId nodeId, ms::LatLon const & point, size_t const adjacencyIndex) - : m_point(point), m_nodeId(nodeId), m_adjacencyIndex(adjacencyIndex) - { - } - - bool IsValid() const { return m_nodeId != kInvalidContextEdgeNodeId; } - - void Save(Writer & w) const; - - size_t Load(Reader const & r, size_t pos, size_t adjacencyIndex); - - m2::RectD const GetLimitRect() const { return m2::RectD(m_point.lat, m_point.lon, m_point.lat, m_point.lon); } -}; - -struct OutgoingCrossNode -{ - ms::LatLon m_point; - TWrittenNodeId m_nodeId; - unsigned char m_outgoingIndex; - size_t m_adjacencyIndex; - - OutgoingCrossNode() - : m_point(ms::LatLon::Zero()) - , m_nodeId(kInvalidContextEdgeNodeId) - , m_outgoingIndex(0) - , m_adjacencyIndex(kInvalidAdjacencyIndex) - { - } - - OutgoingCrossNode(TWrittenNodeId nodeId, size_t const index, ms::LatLon const & point, - size_t const adjacencyIndex) - : m_point(point) - , m_nodeId(nodeId) - , m_outgoingIndex(static_cast(index)) - , m_adjacencyIndex(adjacencyIndex) - { - } - - bool IsValid() const { return m_nodeId != kInvalidContextEdgeNodeId; } - - void Save(Writer & w) const; - - size_t Load(Reader const & r, size_t pos, size_t adjacencyIndex); - - m2::RectD const GetLimitRect() const { return m2::RectD(m_point.lat, m_point.lon, m_point.lat, m_point.lon); } -}; - -using IngoingEdgeIteratorT = vector::const_iterator; -using OutgoingEdgeIteratorT = vector::const_iterator; - -/// Reader class from cross context section in mwm.routing file -class CrossRoutingContextReader -{ - vector m_outgoingNodes; - vector m_neighborMwmList; - vector m_adjacencyMatrix; - m4::Tree m_ingoingIndex; - m4::Tree m_outgoingIndex; - -public: - void Load(Reader const & r); - - const string & GetOutgoingMwmName(OutgoingCrossNode const & outgoingNode) const; - - TWrittenEdgeWeight GetAdjacencyCost(IngoingCrossNode const & ingoing, - OutgoingCrossNode const & outgoing) const; - - vector const & GetNeighboringMwmList() const { return m_neighborMwmList; } - - m2::RectD GetMwmCrossingNodeEqualityRect(ms::LatLon const & point) const; - - template - bool ForEachIngoingNodeNearPoint(ms::LatLon const & point, Fn && fn) const - { - bool found = false; - m_ingoingIndex.ForEachInRect(GetMwmCrossingNodeEqualityRect(point), - [&found, &fn](IngoingCrossNode const & node) { - fn(node); - found = true; - }); - return found; - } - - template - bool ForEachOutgoingNodeNearPoint(ms::LatLon const & point, Fn && fn) const - { - bool found = false; - m_outgoingIndex.ForEachInRect(GetMwmCrossingNodeEqualityRect(point), - [&found, &fn](OutgoingCrossNode const & node) { - fn(node); - found = true; - }); - return found; - } - - template - void ForEachIngoingNode(TFunctor f) const - { - m_ingoingIndex.ForEach(f); - } - - template - void ForEachOutgoingNode(TFunctor f) const - { - for_each(m_outgoingNodes.cbegin(), m_outgoingNodes.cend(), f); - } -}; - -/// Helper class to generate cross context section in mwm.routing file -class CrossRoutingContextWriter -{ - vector m_ingoingNodes; - vector m_outgoingNodes; - vector m_adjacencyMatrix; - vector m_neighborMwmList; - - size_t GetIndexInAdjMatrix(IngoingEdgeIteratorT ingoing, OutgoingEdgeIteratorT outgoing) const; - -public: - void Save(Writer & w) const; - - void AddIngoingNode(TWrittenNodeId const nodeId, ms::LatLon const & point); - - void AddOutgoingNode(TWrittenNodeId const nodeId, string const & targetMwm, - ms::LatLon const & point); - - void ReserveAdjacencyMatrix(); - - void SetAdjacencyCost(IngoingEdgeIteratorT ingoing, OutgoingEdgeIteratorT outgoin, - TWrittenEdgeWeight value); - - pair GetIngoingIterators() const; - - pair GetOutgoingIterators() const; -}; -} diff --git a/routing/index_router.cpp b/routing/index_router.cpp index f6b85fb298..681d767d80 100644 --- a/routing/index_router.cpp +++ b/routing/index_router.cpp @@ -305,7 +305,6 @@ IndexRouter::IndexRouter(VehicleType vehicleType, bool loadAltitudes, , m_numMwmIds(move(numMwmIds)) , m_numMwmTree(move(numMwmTree)) , m_trafficStash(CreateTrafficStash(m_vehicleType, m_numMwmIds, trafficCache)) - , m_indexManager(countryFileFn, m_index) , m_roadGraph(m_index, vehicleType == VehicleType::Pedestrian || vehicleType == VehicleType::Transit ? IRoadGraph::Mode::IgnoreOnewayTag @@ -693,13 +692,15 @@ unique_ptr IndexRouter::MakeWorldGraph() auto crossMwmGraph = make_unique( m_numMwmIds, m_numMwmTree, m_vehicleModelFactory, m_vehicleType == VehicleType::Transit ? VehicleType::Pedestrian : m_vehicleType, - m_countryRectFn, m_index, m_indexManager); + m_countryRectFn, m_index); auto indexGraphLoader = IndexGraphLoader::Create( m_vehicleType == VehicleType::Transit ? VehicleType::Pedestrian : m_vehicleType, m_loadAltitudes, m_numMwmIds, m_vehicleModelFactory, m_estimator, m_index); if (m_vehicleType != VehicleType::Transit) + { return make_unique(move(crossMwmGraph), move(indexGraphLoader), m_estimator); + } auto transitGraphLoader = TransitGraphLoader::Create(m_index, m_numMwmIds, m_estimator); return make_unique(move(crossMwmGraph), move(indexGraphLoader), move(transitGraphLoader), m_estimator); diff --git a/routing/index_router.hpp b/routing/index_router.hpp index 8f01547dc9..c31a5f2c85 100644 --- a/routing/index_router.hpp +++ b/routing/index_router.hpp @@ -7,7 +7,6 @@ #include "routing/features_road_graph.hpp" #include "routing/joint.hpp" #include "routing/router.hpp" -#include "routing/routing_mapping.hpp" #include "routing/segmented_route.hpp" #include "routing/world_graph.hpp" @@ -123,7 +122,6 @@ private: std::shared_ptr m_numMwmIds; std::shared_ptr> m_numMwmTree; std::shared_ptr m_trafficStash; - RoutingIndexManager m_indexManager; FeaturesRoadGraph m_roadGraph; std::shared_ptr m_estimator; diff --git a/routing/online_absent_fetcher.hpp b/routing/online_absent_fetcher.hpp index 70fdfde64e..f05c267e98 100644 --- a/routing/online_absent_fetcher.hpp +++ b/routing/online_absent_fetcher.hpp @@ -1,7 +1,7 @@ #pragma once #include "routing/checkpoints.hpp" -#include "routing/routing_mapping.hpp" +#include "routing/router.hpp" #include "geometry/point2d.hpp" @@ -9,7 +9,7 @@ #include "std/string.hpp" #include "std/unique_ptr.hpp" - +#include "std/vector.hpp" namespace routing { diff --git a/routing/osrm2feature_map.cpp b/routing/osrm2feature_map.cpp deleted file mode 100644 index 965aed39ac..0000000000 --- a/routing/osrm2feature_map.cpp +++ /dev/null @@ -1,505 +0,0 @@ -#include "routing/osrm2feature_map.hpp" - -#include "indexer/data_header.hpp" - -#include "platform/local_country_file_utils.hpp" - -#include "coding/file_name_utils.hpp" -#include "coding/internal/file_data.hpp" -#include "coding/read_write_utils.hpp" -#include "coding/varint.hpp" - -#include "base/assert.hpp" -#include "base/logging.hpp" -#include "base/math.hpp" -#include "base/scope_guard.hpp" - -#include "std/fstream.hpp" -#include "std/sstream.hpp" -#include "std/unordered_map.hpp" - -#include "defines.hpp" - -#include "3party/succinct/mapper.hpp" - -using platform::CountryIndexes; - -namespace -{ -const routing::TNodesList kEmptyList; -} // namespace - -namespace routing -{ - -namespace OsrmMappingTypes -{ -FtSeg::FtSeg(uint32_t fid, uint32_t ps, uint32_t pe) - : m_fid(fid), - m_pointStart(static_cast(ps)), - m_pointEnd(static_cast(pe)) -{ - CHECK_NOT_EQUAL(ps, pe, ()); - CHECK_EQUAL(m_pointStart, ps, ()); - CHECK_EQUAL(m_pointEnd, pe, ()); -} - -FtSeg::FtSeg(uint64_t x) - : m_fid(x & 0xFFFFFFFF), m_pointStart(x >> 48), m_pointEnd((x >> 32) & 0xFFFF) -{ -} - -uint64_t FtSeg::Store() const -{ - return (uint64_t(m_pointStart) << 48) + (uint64_t(m_pointEnd) << 32) + uint64_t(m_fid); -} - -bool FtSeg::Merge(FtSeg const & other) -{ - if (other.m_fid != m_fid) - return false; - - if (IsForward() != other.IsForward()) - return false; - - auto const s1 = min(m_pointStart, m_pointEnd); - auto const e1 = max(m_pointStart, m_pointEnd); - auto const s2 = min(other.m_pointStart, other.m_pointEnd); - auto const e2 = max(other.m_pointStart, other.m_pointEnd); - - if (my::IsIntersect(s1, e1, s2, e2)) - { - m_pointStart = min(s1, s2); - m_pointEnd = max(e1, e2); - if (!other.IsForward()) - swap(m_pointStart, m_pointEnd); - - return true; - } - else - return false; -} - -bool FtSeg::IsIntersect(FtSeg const & other) const -{ - if (other.m_fid != m_fid) - return false; - - auto const s1 = min(m_pointStart, m_pointEnd); - auto const e1 = max(m_pointStart, m_pointEnd); - auto const s2 = min(other.m_pointStart, other.m_pointEnd); - auto const e2 = max(other.m_pointStart, other.m_pointEnd); - - return my::IsIntersect(s1, e1, s2, e2); -} - -string DebugPrint(FtSeg const & seg) -{ - stringstream ss; - ss << "{ fID = " << seg.m_fid << - ", pStart = " << seg.m_pointStart << - ", pEnd = " << seg.m_pointEnd << " }"; - return ss.str(); -} - -string DebugPrint(SegOffset const & off) -{ - stringstream ss; - ss << "{ " << off.m_nodeId << ", " << off.m_offset << " }"; - return ss.str(); -} - -bool IsInside(FtSeg const & bigSeg, FtSeg const & smallSeg) -{ - ASSERT_EQUAL(bigSeg.m_fid, smallSeg.m_fid, ()); - ASSERT_EQUAL(smallSeg.m_pointEnd - smallSeg.m_pointStart, 1, ()); - - auto segmentLeft = min(bigSeg.m_pointStart, bigSeg.m_pointEnd); - auto segmentRight = max(bigSeg.m_pointStart, bigSeg.m_pointEnd); - - return (segmentLeft <= smallSeg.m_pointStart && segmentRight >= smallSeg.m_pointEnd); -} - -FtSeg SplitSegment(FtSeg const & segment, FtSeg const & splitter) -{ - FtSeg resultSeg; - resultSeg.m_fid = segment.m_fid; - - if (segment.m_pointStart < segment.m_pointEnd) - { - resultSeg.m_pointStart = segment.m_pointStart; - resultSeg.m_pointEnd = splitter.m_pointEnd; - } - else - { - resultSeg.m_pointStart = segment.m_pointStart; - resultSeg.m_pointEnd = splitter.m_pointStart; - } - return resultSeg; -} -} // namespace OsrmMappingTypes - -void OsrmFtSegMapping::Clear() -{ - m_offsets.clear(); - m_handle.Unmap(); -} - -void OsrmFtSegMapping::Load(FilesMappingContainer & cont, platform::LocalCountryFile const & localFile) -{ - Clear(); - - { - ReaderSource src(cont.GetReader(ROUTING_NODEIND_TO_FTSEGIND_FILE_TAG)); - uint32_t const count = ReadVarUint(src); - if (count == 0) - return; - m_offsets.resize(count); - for (uint32_t i = 0; i < count; ++i) - { - m_offsets[i].m_nodeId = ReadVarUint(src); - m_offsets[i].m_offset = ReadVarUint(src); - } - } - - m_backwardIndex.Construct(*this, m_offsets.back().m_nodeId, cont, localFile); -} - -void OsrmFtSegMapping::Map(FilesMappingContainer & cont) -{ - m_handle.Assign(cont.Map(ROUTING_FTSEG_FILE_TAG)); - ASSERT(m_handle.IsValid(), ()); - succinct::mapper::map(m_segments, m_handle.GetData()); -} - -void OsrmFtSegMapping::Unmap() -{ - m_handle.Unmap(); -} - -bool OsrmFtSegMapping::IsMapped() const -{ - return m_handle.IsValid(); -} - -void OsrmFtSegMapping::DumpSegmentsByFID(uint32_t fID) const -{ -#ifdef DEBUG - for (size_t i = 0; i < m_segments.size(); ++i) - { - OsrmMappingTypes::FtSeg s(m_segments[i]); - if (s.m_fid == fID) - LOG(LDEBUG, (s)); - } -#endif -} - -void OsrmFtSegMapping::DumpSegmentByNode(TOsrmNodeId nodeId) const -{ -#ifdef DEBUG - ForEachFtSeg(nodeId, [] (OsrmMappingTypes::FtSeg const & s) - { - LOG(LDEBUG, (s)); - }); -#endif -} - -void OsrmFtSegMapping::GetOsrmNodes(TFtSegVec const & segments, OsrmNodesT & res) const -{ - auto addResFn = [&] (uint64_t seg, TOsrmNodeId nodeId, bool forward) - { - auto it = res.insert({ seg, { forward ? nodeId : INVALID_NODE_ID, - forward ? INVALID_NODE_ID : nodeId } }); - if (it.second) - return false; - else - { - if (forward) - { - ASSERT_EQUAL(it.first->second.first, INVALID_NODE_ID, ()); - it.first->second.first = nodeId; - } - else - { - ASSERT_EQUAL(it.first->second.second, INVALID_NODE_ID, ()); - it.first->second.second = nodeId; - } - } - return true; - }; - - for (auto it = segments.begin(); it != segments.end(); ++it) - { - OsrmMappingTypes::FtSeg const & seg = *it; - - TNodesList const & nodeIds = m_backwardIndex.GetNodeIdByFid(seg.m_fid); - - for (uint32_t nodeId : nodeIds) - { - auto const & range = GetSegmentsRange(nodeId); - for (size_t i = range.first; i != range.second; ++i) - { - OsrmMappingTypes::FtSeg const s(m_segments[i]); - if (s.m_fid != seg.m_fid) - continue; - - if (s.m_pointStart <= s.m_pointEnd) - { - if (seg.m_pointStart >= s.m_pointStart && seg.m_pointEnd <= s.m_pointEnd) - { - if (addResFn(seg.Store(), nodeId, true)) - { - break; - } - } - } - else - { - if (seg.m_pointStart >= s.m_pointEnd && seg.m_pointEnd <= s.m_pointStart) - { - if (addResFn(seg.Store(), nodeId, false)) - { - break; - } - } - } - } - } - } -} - -void OsrmFtSegMapping::GetSegmentByIndex(size_t idx, OsrmMappingTypes::FtSeg & seg) const -{ - ASSERT_LESS(idx, m_segments.size(), ()); - OsrmMappingTypes::FtSeg(m_segments[idx]).Swap(seg); -} - -pair OsrmFtSegMapping::GetSegmentsRange(TOsrmNodeId nodeId) const -{ - SegOffsetsT::const_iterator it = lower_bound(m_offsets.begin(), m_offsets.end(), OsrmMappingTypes::SegOffset(nodeId, 0), - [] (OsrmMappingTypes::SegOffset const & o, OsrmMappingTypes::SegOffset const & val) - { - return (o.m_nodeId < val.m_nodeId); - }); - - size_t const index = distance(m_offsets.begin(), it); - size_t const start = (index > 0 ? m_offsets[index - 1].m_offset + nodeId : nodeId); - - if (index < m_offsets.size() && m_offsets[index].m_nodeId == nodeId) - return make_pair(start, m_offsets[index].m_offset + nodeId + 1); - else - return make_pair(start, start + 1); -} - -TOsrmNodeId OsrmFtSegMapping::GetNodeId(uint32_t segInd) const -{ - SegOffsetsT::const_iterator it = lower_bound(m_offsets.begin(), m_offsets.end(), OsrmMappingTypes::SegOffset(segInd, 0), - [] (OsrmMappingTypes::SegOffset const & o, OsrmMappingTypes::SegOffset const & val) - { - return (o.m_nodeId + o.m_offset < val.m_nodeId); - }); - - size_t const index = distance(m_offsets.begin(), it); - uint32_t const prevOffset = index > 0 ? m_offsets[index-1].m_offset : 0; - - if ((index < m_offsets.size()) && - (segInd >= prevOffset + m_offsets[index].m_nodeId) && - (segInd <= m_offsets[index].m_offset + m_offsets[index].m_nodeId)) - { - return m_offsets[index].m_nodeId; - } - - return (segInd - prevOffset); -} - -OsrmFtSegMappingBuilder::OsrmFtSegMappingBuilder() - : m_lastOffset(0) -{ -} - -void OsrmFtSegMappingBuilder::Append(TOsrmNodeId nodeId, FtSegVectorT const & data) -{ - size_t const count = data.size(); - - if (count == 0) - { - m_buffer.emplace_back(OsrmMappingTypes::FtSeg(kInvalidFid, 0, 1).Store()); - } - else - { - for (size_t i = 0; i < count; ++i) - m_buffer.emplace_back(data[i].Store()); - } - - if (count > 1) - { - m_lastOffset += (data.size() - 1); - - uint32_t const off = static_cast(m_lastOffset); - CHECK_EQUAL(m_lastOffset, off, ()); - - m_offsets.emplace_back(OsrmMappingTypes::SegOffset(nodeId, off)); - } -} -void OsrmFtSegMappingBuilder::Save(FilesContainerW & cont) const -{ - { - FileWriter writer = cont.GetWriter(ROUTING_NODEIND_TO_FTSEGIND_FILE_TAG); - uint32_t const count = static_cast(m_offsets.size()); - WriteVarUint(writer, count); - for (uint32_t i = 0; i < count; ++i) - { - WriteVarUint(writer, m_offsets[i].m_nodeId); - WriteVarUint(writer, m_offsets[i].m_offset); - } - - // Write padding to make next elias_fano start address multiple of 4. - writer.WritePaddingByEnd(4); - } - - string const fName = cont.GetFileName() + "." ROUTING_FTSEG_FILE_TAG; - MY_SCOPE_GUARD(deleteFileGuard, bind(&FileWriter::DeleteFileX, cref(fName))); - - succinct::elias_fano_compressed_list compressed(m_buffer); - succinct::mapper::freeze(compressed, fName.c_str()); - cont.Write(fName, ROUTING_FTSEG_FILE_TAG); -} - -void OsrmFtSegBackwardIndex::Save(string const & nodesFileName, string const & bitsFileName) -{ - { - LOG(LINFO, ("Saving routing nodes index to ", nodesFileName)); - string const nodesFileNameTmp = nodesFileName + EXTENSION_TMP; - FileWriter nodesFile(nodesFileNameTmp); - WriteVarUint(nodesFile, static_cast(m_nodeIds.size())); - for (auto const bucket : m_nodeIds) - { - rw::WriteVectorOfPOD(nodesFile, bucket); - } - my::RenameFileX(nodesFileNameTmp, nodesFileName); - } - { - LOG(LINFO, ("Saving features routing bits to ", bitsFileName)); - string const bitsFileNameTmp = bitsFileName + EXTENSION_TMP; - succinct::mapper::freeze(m_rankIndex, bitsFileNameTmp.c_str()); - my::RenameFileX(bitsFileNameTmp, bitsFileName); - } -} - -bool OsrmFtSegBackwardIndex::Load(string const & nodesFileName, string const & bitsFileName) -{ - Platform & pl = GetPlatform(); - if (!pl.IsFileExistsByFullPath(nodesFileName) || !pl.IsFileExistsByFullPath(bitsFileName)) - return false; - - m_mappedBits.reset(new MmapReader(bitsFileName)); - - { - FileReader nodesFile(nodesFileName); - ReaderSource nodesSource(nodesFile); - uint32_t size = ReadVarUint(nodesSource); - m_nodeIds.resize(size); - for (uint32_t i = 0; i < size; ++i) - { - rw::ReadVectorOfPOD(nodesSource, m_nodeIds[i]); - } - } - succinct::mapper::map(m_rankIndex, reinterpret_cast(m_mappedBits->Data())); - - return true; -} - -void OsrmFtSegBackwardIndex::Construct(OsrmFtSegMapping & mapping, uint32_t maxNodeId, - FilesMappingContainer & routingFile, - platform::LocalCountryFile const & localFile) -{ - Clear(); - - feature::DataHeader header(localFile.GetPath(MapOptions::Map)); - m_oldFormat = header.GetFormat() < version::Format::v5; - if (m_oldFormat) - LOG(LINFO, ("Using old format index for", localFile.GetCountryName())); - - CountryIndexes::PreparePlaceOnDisk(localFile); - string const bitsFileName = CountryIndexes::GetPath(localFile, CountryIndexes::Index::Bits); - string const nodesFileName = CountryIndexes::GetPath(localFile, CountryIndexes::Index::Nodes); - - m_table = feature::FeaturesOffsetsTable::CreateIfNotExistsAndLoad(localFile); - - if (Load(bitsFileName, nodesFileName)) - return; - - LOG(LINFO, ("Backward routing index is absent! Creating new one.")); - mapping.Map(routingFile); - - // Generate temporary index to speedup processing - unordered_map temporaryBackwardIndex; - for (uint32_t i = 0; i < maxNodeId; ++i) - { - auto indexes = mapping.GetSegmentsRange(i); - for (; indexes.first != indexes.second; ++indexes.first) - { - OsrmMappingTypes::FtSeg seg; - mapping.GetSegmentByIndex(indexes.first, seg); - temporaryBackwardIndex[seg.m_fid].push_back(i); - } - } - - mapping.Unmap(); - LOG(LINFO, ("Temporary index constructed")); - - // Create final index - vector inIndex(m_table->size(), false); - m_nodeIds.reserve(temporaryBackwardIndex.size()); - - size_t removedNodes = 0; - for (size_t i = 0; i < m_table->size(); ++i) - { - uint32_t const fid = m_oldFormat ? m_table->GetFeatureOffset(i) : static_cast(i); - auto it = temporaryBackwardIndex.find(fid); - if (it != temporaryBackwardIndex.end()) - { - inIndex[i] = true; - m_nodeIds.emplace_back(move(it->second)); - - // Remove duplicates nodes emmited by equal choises on a generation route step. - TNodesList & nodesList = m_nodeIds.back(); - size_t const foundNodes = nodesList.size(); - sort(nodesList.begin(), nodesList.end()); - nodesList.erase(unique(nodesList.begin(), nodesList.end()), nodesList.end()); - removedNodes += foundNodes - nodesList.size(); - } - } - - LOG(LDEBUG, ("Backward index constructor removed", removedNodes, "duplicates.")); - - // Pack and save index - succinct::rs_bit_vector(inIndex).swap(m_rankIndex); - - LOG(LINFO, ("Writing additional indexes to data files", bitsFileName, nodesFileName)); - Save(bitsFileName, nodesFileName); -} - -TNodesList const & OsrmFtSegBackwardIndex::GetNodeIdByFid(uint32_t fid) const -{ - ASSERT(m_table, ()); - - size_t const index = m_oldFormat ? m_table->GetFeatureIndexbyOffset(fid) : fid; - ASSERT_LESS(index, m_table->size(), ("Can't find feature index in offsets table")); - if (!m_rankIndex[index]) - return kEmptyList; - - size_t nodeIdx = m_rankIndex.rank(index); - ASSERT_LESS(nodeIdx, m_nodeIds.size(), ()); - return m_nodeIds[nodeIdx]; -} - -void OsrmFtSegBackwardIndex::Clear() -{ - m_nodeIds.clear(); - succinct::rs_bit_vector().swap(m_rankIndex); - m_table.reset(); - m_mappedBits.reset(); -} - -} diff --git a/routing/osrm2feature_map.hpp b/routing/osrm2feature_map.hpp deleted file mode 100644 index 9813743096..0000000000 --- a/routing/osrm2feature_map.hpp +++ /dev/null @@ -1,206 +0,0 @@ -#pragma once - -#include "indexer/features_offsets_table.hpp" - -#include "coding/file_container.hpp" -#include "coding/mmap_reader.hpp" - -#include "platform/platform.hpp" - -#include "base/scope_guard.hpp" - -#include "std/limits.hpp" -#include "std/string.hpp" -#include "std/unordered_map.hpp" -#include "std/utility.hpp" -#include "std/vector.hpp" - -#include "defines.hpp" - -#include "3party/succinct/rs_bit_vector.hpp" -#include "3party/succinct/elias_fano_compressed_list.hpp" - -namespace routing -{ - -typedef uint32_t TOsrmNodeId; -typedef vector TNodesList; -constexpr TOsrmNodeId INVALID_NODE_ID = numeric_limits::max(); -constexpr uint32_t kInvalidFid = numeric_limits::max(); - -namespace OsrmMappingTypes -{ -#pragma pack (push, 1) - struct FtSeg - { - uint32_t m_fid; - uint16_t m_pointStart; - uint16_t m_pointEnd; - - - // No need to initialize something her (for vector). - FtSeg() {} - FtSeg(uint32_t fid, uint32_t ps, uint32_t pe); - - explicit FtSeg(uint64_t x); - uint64_t Store() const; - - bool Merge(FtSeg const & other); - - bool operator == (FtSeg const & other) const - { - return (other.m_fid == m_fid) - && (other.m_pointEnd == m_pointEnd) - && (other.m_pointStart == m_pointStart); - } - - bool IsIntersect(FtSeg const & other) const; - - bool IsForward() const { return m_pointEnd > m_pointStart; } - - bool IsValid() const - { - return m_fid != kInvalidFid; - } - - void Swap(FtSeg & other) - { - swap(m_fid, other.m_fid); - swap(m_pointStart, other.m_pointStart); - swap(m_pointEnd, other.m_pointEnd); - } - - friend string DebugPrint(FtSeg const & seg); - }; - - struct SegOffset - { - TOsrmNodeId m_nodeId; - uint32_t m_offset; - - SegOffset() : m_nodeId(0), m_offset(0) {} - - SegOffset(uint32_t nodeId, uint32_t offset) - : m_nodeId(nodeId), m_offset(offset) - { - } - - friend string DebugPrint(SegOffset const & off); - }; -#pragma pack (pop) - -/// Checks if a smallSeg is inside a bigSeg. Note that the smallSeg must be an ordered segment -/// with 1 point length. -bool IsInside(FtSeg const & bigSeg, FtSeg const & smallSeg); - -/// Splits segment by splitter segment and takes part of it. -/// Warning! This function returns part from the start of the segment to the splitter, including it. -/// Splitter segment points must be ordered. -FtSeg SplitSegment(FtSeg const & segment, FtSeg const & splitter); -} // namespace OsrmMappingTypes - -class OsrmFtSegMapping; - -class OsrmFtSegBackwardIndex -{ - succinct::rs_bit_vector m_rankIndex; - vector m_nodeIds; - unique_ptr m_table; - - unique_ptr m_mappedBits; - - bool m_oldFormat; - - void Save(string const & nodesFileName, string const & bitsFileName); - bool Load(string const & nodesFileName, string const & bitsFileName); - -public: - void Construct(OsrmFtSegMapping & mapping, uint32_t maxNodeId, - FilesMappingContainer & routingFile, - platform::LocalCountryFile const & localFile); - - TNodesList const & GetNodeIdByFid(uint32_t fid) const; - - void Clear(); -}; - -class OsrmFtSegMapping -{ -public: - using TFtSegVec = vector; - - void Clear(); - void Load(FilesMappingContainer & cont, platform::LocalCountryFile const & localFile); - - void Map(FilesMappingContainer & cont); - void Unmap(); - bool IsMapped() const; - - template void ForEachFtSeg(TOsrmNodeId nodeId, ToDo toDo) const - { - pair r = GetSegmentsRange(nodeId); - while (r.first != r.second) - { - OsrmMappingTypes::FtSeg s(m_segments[r.first]); - if (s.IsValid()) - toDo(s); - ++r.first; - } - } - - typedef unordered_map > OsrmNodesT; - void GetOsrmNodes(TFtSegVec const & segments, OsrmNodesT & res) const; - - void GetSegmentByIndex(size_t idx, OsrmMappingTypes::FtSeg & seg) const; - TNodesList const & GetNodeIdByFid(uint32_t fid) const - { - return m_backwardIndex.GetNodeIdByFid(fid); - } - - /// @name For debug purpose only. - //@{ - void DumpSegmentsByFID(uint32_t fID) const; - void DumpSegmentByNode(TOsrmNodeId nodeId) const; - //@} - - /// @name For unit test purpose only. - //@{ - /// @return STL-like range [s, e) of segments indexies for passed node. - /// @note Methods GetSegmentsRange(...) and GetOsrmNodes(...) are not symmetric. - /// For example in Tverskay Oblast for node id 161179 two FtSet can be gotten - /// with GetSegmentsRange() / GetSegmentByIndex(). - /// But having these segments it's impossible to get node id 161179 with the help of - /// GetOsrmNodes(...). - pair GetSegmentsRange(TOsrmNodeId nodeId) const; - /// @return Node id for segment's index. - TOsrmNodeId GetNodeId(uint32_t segInd) const; - - size_t GetSegmentsCount() const { return static_cast(m_segments.size()); } - //@} - -protected: - typedef vector SegOffsetsT; - SegOffsetsT m_offsets; - -private: - succinct::elias_fano_compressed_list m_segments; - FilesMappingContainer::Handle m_handle; - OsrmFtSegBackwardIndex m_backwardIndex; -}; - -class OsrmFtSegMappingBuilder : public OsrmFtSegMapping -{ -public: - OsrmFtSegMappingBuilder(); - - typedef vector FtSegVectorT; - - void Append(TOsrmNodeId nodeId, FtSegVectorT const & data); - void Save(FilesContainerW & cont) const; - -private: - vector m_buffer; - uint64_t m_lastOffset; -}; - -} diff --git a/routing/osrm_data_facade.hpp b/routing/osrm_data_facade.hpp deleted file mode 100644 index 8f4f26a91f..0000000000 --- a/routing/osrm_data_facade.hpp +++ /dev/null @@ -1,312 +0,0 @@ -#pragma once - -#include "routing/cross_routing_context.hpp" - -#include "coding/file_container.hpp" -#include "coding/read_write_utils.hpp" - -#include "base/bits.hpp" - -#include "std/string.hpp" - -#include "defines.hpp" - -#include "3party/succinct/elias_fano.hpp" -#include "3party/succinct/elias_fano_compressed_list.hpp" -#include "3party/succinct/gamma_vector.hpp" -#include "3party/succinct/rs_bit_vector.hpp" -#include "3party/succinct/mapper.hpp" - -#if defined(__clang__) -#pragma clang diagnostic push -#pragma clang diagnostic ignored "-Wunused-local-typedef" -#endif - -// TODO (ldragunov) exclude osrm specific headers from here! They causes "coordinate" problem -#include "3party/osrm/osrm-backend/server/data_structures/datafacade_base.hpp" -#include "3party/osrm/osrm-backend/data_structures/travel_mode.hpp" - -#if defined(__clang__) -#pragma clang diagnostic pop -#endif - -namespace routing -{ - -template class OsrmRawDataFacade : public BaseDataFacade -{ - template void ClearContainer(T & t) - { - T().swap(t); - } - -protected: - succinct::elias_fano_compressed_list m_edgeData; - succinct::rs_bit_vector m_shortcuts; - succinct::elias_fano m_matrix; - succinct::elias_fano_compressed_list m_edgeId; - - uint32_t m_numberOfNodes = 0; - -public: - //OsrmRawDataFacade(): m_numberOfNodes(0) {} - - void LoadRawData(char const * pRawEdgeData, char const * pRawEdgeIds, char const * pRawEdgeShortcuts, char const * pRawFanoMatrix) - { - ClearRawData(); - - ASSERT(pRawEdgeData, ()); - succinct::mapper::map(m_edgeData, pRawEdgeData); - - ASSERT(pRawEdgeIds, ()); - succinct::mapper::map(m_edgeId, pRawEdgeIds); - - ASSERT(pRawEdgeShortcuts, ()); - succinct::mapper::map(m_shortcuts, pRawEdgeShortcuts); - - ASSERT(pRawFanoMatrix, ()); - m_numberOfNodes = *reinterpret_cast(pRawFanoMatrix); - succinct::mapper::map(m_matrix, pRawFanoMatrix + sizeof(m_numberOfNodes)); - } - - void ClearRawData() - { - ClearContainer(m_edgeData); - ClearContainer(m_edgeId); - ClearContainer(m_shortcuts); - ClearContainer(m_matrix); - } - - unsigned GetNumberOfNodes() const override - { - return m_numberOfNodes; - } - - unsigned GetNumberOfEdges() const override - { - return static_cast(m_edgeData.size()); - } - - unsigned GetOutDegree(const NodeID n) const override - { - return EndEdges(n) - BeginEdges(n); - } - - NodeID GetTarget(const EdgeID e) const override - { - return (m_matrix.select(e) / 2) % GetNumberOfNodes(); - } - - EdgeDataT GetEdgeData(const EdgeID e, NodeID node) const override - { - EdgeDataT res; - - res.shortcut = m_shortcuts[e]; - res.id = res.shortcut ? (node - static_cast(bits::ZigZagDecode(m_edgeId[static_cast(m_shortcuts.rank(e))]))) : 0; - res.backward = (m_matrix.select(e) % 2 == 1); - res.forward = !res.backward; - res.distance = static_cast(m_edgeData[e]); - - return res; - } - - EdgeDataT & GetEdgeData(const EdgeID) const override - { - static EdgeDataT res; - ASSERT(false, ("Maps me routing facade does not supports this edge unpacking method")); - return res; - } - - //! TODO: Make proper travelmode getter when we add it to routing file - TravelMode GetTravelModeForEdgeID(const unsigned) const override - { - return TRAVEL_MODE_DEFAULT; - } - - EdgeID BeginEdges(const NodeID n) const override - { - uint64_t idx = 2 * n * (uint64_t)GetNumberOfNodes(); - return n == 0 ? 0 : static_cast(m_matrix.rank(min(idx, m_matrix.size()))); - } - - EdgeID EndEdges(const NodeID n) const override - { - uint64_t const idx = 2 * (n + 1) * (uint64_t)GetNumberOfNodes(); - return static_cast(m_matrix.rank(min(idx, m_matrix.size()))); - } - - EdgeRange GetAdjacentEdgeRange(const NodeID node) const override - { - return osrm::irange(BeginEdges(node), EndEdges(node)); - } - - // searches for a specific edge - EdgeID FindEdge(const NodeID from, const NodeID to) const override - { - EdgeID smallest_edge = SPECIAL_EDGEID; - EdgeWeight smallest_weight = INVALID_EDGE_WEIGHT; - for (auto edge : GetAdjacentEdgeRange(from)) - { - const NodeID target = GetTarget(edge); - const EdgeWeight weight = GetEdgeData(edge).distance; - if (target == to && weight < smallest_weight) - { - smallest_edge = edge; - smallest_weight = weight; - } - } - return smallest_edge; - } - - EdgeID FindEdgeInEitherDirection(const NodeID, const NodeID) const override - { - return (EdgeID)0; - } - - EdgeID FindEdgeIndicateIfReverse(const NodeID, const NodeID, bool &) const override - { - return (EdgeID)0; - } - - // node and edge information access - FixedPointCoordinate GetCoordinateOfNode(const unsigned) const override - { - return FixedPointCoordinate(); - } - - bool EdgeIsCompressed(const unsigned) const override - { - return false; - } - - unsigned GetGeometryIndexForEdgeID(const unsigned) const override - { - return false; - } - - void GetUncompressedGeometry(const unsigned, std::vector &) const override - { - } - - TurnInstruction GetTurnInstructionForEdgeID(const unsigned) const override - { - return TurnInstruction::NoTurn; - } - - bool LocateClosestEndPointForCoordinate(const FixedPointCoordinate &, - FixedPointCoordinate &, - const unsigned = 18) override - { - return false; - } - - /*bool FindPhantomNodeForCoordinate(const FixedPointCoordinate &input_coordinate, - PhantomNode &resulting_phantom_node, - const unsigned zoom_level) override - { - return false; - }*/ - - /*bool IncrementalFindPhantomNodeForCoordinate(const FixedPointCoordinate &input_coordinate, - std::vector &resulting_phantom_node_vector, - const unsigned zoom_level, - const unsigned number_of_results) override - { - return false; - }*/ - - bool IncrementalFindPhantomNodeForCoordinate(const FixedPointCoordinate &, - std::vector &, - const unsigned) override - { - return false; - } - - bool IncrementalFindPhantomNodeForCoordinate(const FixedPointCoordinate &, - PhantomNode &) override - { - return false; - } - - bool IncrementalFindPhantomNodeForCoordinateWithMaxDistance( - const FixedPointCoordinate &, - std::vector> &, - const double, - const unsigned, - const unsigned) override - { - return false; - } - - unsigned GetCheckSum() const override - { - return 0; - } - - unsigned GetNameIndexFromEdgeID(const unsigned) const override - { - return -1; - } - - //void GetName(const unsigned name_id, std::string &result) const override - //{ - //} - - std::string get_name_for_id(const unsigned) const override - { - return std::string(); - } - - std::string GetTimestamp() const override - { - return std::string(); - } -}; - - -template class OsrmDataFacade : public OsrmRawDataFacade -{ - typedef OsrmRawDataFacade super; - - FilesMappingContainer::Handle m_handleEdgeData; - FilesMappingContainer::Handle m_handleEdgeId; - FilesMappingContainer::Handle m_handleEdgeIdFano; - FilesMappingContainer::Handle m_handleShortcuts; - FilesMappingContainer::Handle m_handleFanoMatrix; - - using OsrmRawDataFacade::LoadRawData; - using OsrmRawDataFacade::ClearRawData; - -public: - - void Load(FilesMappingContainer const & container) - { - Clear(); - - // Map huge data first, as we hope it will reduce fragmentation of the program address space. - m_handleFanoMatrix.Assign(container.Map(ROUTING_MATRIX_FILE_TAG)); - ASSERT(m_handleFanoMatrix.IsValid(), ()); - - m_handleEdgeData.Assign(container.Map(ROUTING_EDGEDATA_FILE_TAG)); - ASSERT(m_handleEdgeData.IsValid(), ()); - - m_handleEdgeId.Assign(container.Map(ROUTING_EDGEID_FILE_TAG)); - ASSERT(m_handleEdgeId.IsValid(), ()); - - m_handleShortcuts.Assign(container.Map(ROUTING_SHORTCUTS_FILE_TAG)); - ASSERT(m_handleShortcuts.IsValid(), ()); - - LoadRawData(m_handleEdgeData.GetData(), m_handleEdgeId.GetData(), m_handleShortcuts.GetData(), m_handleFanoMatrix.GetData()); - } - - void Clear() - { - ClearRawData(); - m_handleEdgeData.Unmap(); - m_handleEdgeId.Unmap(); - m_handleShortcuts.Unmap(); - m_handleFanoMatrix.Unmap(); - } -}; - -} diff --git a/routing/osrm_engine.cpp b/routing/osrm_engine.cpp deleted file mode 100644 index c21170297e..0000000000 --- a/routing/osrm_engine.cpp +++ /dev/null @@ -1,94 +0,0 @@ -#include "routing/osrm_engine.hpp" -#include "routing/osrm2feature_map.hpp" - -#include "base/logging.hpp" -#include "base/timer.hpp" - -#include "3party/osrm/osrm-backend/data_structures/internal_route_result.hpp" -#include "3party/osrm/osrm-backend/data_structures/search_engine_data.hpp" -#include "3party/osrm/osrm-backend/routing_algorithms/n_to_m_many_to_many.hpp" -#include "3party/osrm/osrm-backend/routing_algorithms/shortest_path.hpp" - -namespace routing -{ -bool IsRouteExist(InternalRouteResult const & r) -{ - return !(INVALID_EDGE_WEIGHT == r.shortest_path_length || r.segment_end_coordinates.empty() || - r.source_traversed_in_reverse.empty()); -} - -void GenerateRoutingTaskFromNodeId(NodeID const nodeId, bool const isStartNode, - PhantomNode & taskNode) -{ - taskNode.forward_node_id = isStartNode ? nodeId : INVALID_NODE_ID; - taskNode.reverse_node_id = isStartNode ? INVALID_NODE_ID : nodeId; - taskNode.forward_weight = 0; - taskNode.reverse_weight = 0; - taskNode.forward_offset = 0; - taskNode.reverse_offset = 0; - taskNode.name_id = 1; -} - -void FindWeightsMatrix(TRoutingNodes const & sources, TRoutingNodes const & targets, - TRawDataFacade & facade, vector & result) -{ - SearchEngineData engineData; - NMManyToManyRouting pathFinder(&facade, engineData); - PhantomNodeArray sourcesTaskVector(sources.size()); - PhantomNodeArray targetsTaskVector(targets.size()); - for (size_t i = 0; i < sources.size(); ++i) - sourcesTaskVector[i].push_back(sources[i].node); - for (size_t i = 0; i < targets.size(); ++i) - targetsTaskVector[i].push_back(targets[i].node); - - // Calculate time consumption of a NtoM path finding. - my::HighResTimer timer(true); - shared_ptr> resultTable = pathFinder(sourcesTaskVector, targetsTaskVector); - LOG(LINFO, ("Duration of a single one-to-many routing call", timer.ElapsedNano(), "ns")); - timer.Reset(); - ASSERT_EQUAL(resultTable->size(), sources.size() * targets.size(), ()); - result.swap(*resultTable); -} - -FeatureGraphNode::FeatureGraphNode(NodeID const nodeId, bool const isStartNode, - Index::MwmId const & id) - : segmentPoint(m2::PointD::Zero()), mwmId(id) -{ - node.forward_node_id = isStartNode ? nodeId : INVALID_NODE_ID; - node.reverse_node_id = isStartNode ? INVALID_NODE_ID : nodeId; - node.forward_weight = 0; - node.reverse_weight = 0; - node.forward_offset = 0; - node.reverse_offset = 0; - node.name_id = 1; - segment.m_fid = kInvalidFid; -} - -FeatureGraphNode::FeatureGraphNode(NodeID const nodeId, NodeID const reverseNodeId, - bool const isStartNode, Index::MwmId const & id) - : segmentPoint(m2::PointD::Zero()), mwmId(id) -{ - node.forward_node_id = isStartNode ? nodeId : reverseNodeId; - node.reverse_node_id = isStartNode ? reverseNodeId : nodeId; - node.forward_weight = 0; - node.reverse_weight = 0; - node.forward_offset = 0; - node.reverse_offset = 0; - node.name_id = 1; - segment.m_fid = kInvalidFid; -} - -FeatureGraphNode::FeatureGraphNode() -{ - node.forward_node_id = INVALID_NODE_ID; - node.reverse_node_id = INVALID_NODE_ID; - node.forward_weight = 0; - node.reverse_weight = 0; - node.forward_offset = 0; - node.reverse_offset = 0; - node.name_id = 1; - segment.m_fid = kInvalidFid; - segmentPoint = m2::PointD::Zero(); -} - -} // namespace routing diff --git a/routing/osrm_engine.hpp b/routing/osrm_engine.hpp deleted file mode 100644 index 7be0ebae30..0000000000 --- a/routing/osrm_engine.hpp +++ /dev/null @@ -1,74 +0,0 @@ -#pragma once - -#include "routing/osrm2feature_map.hpp" -#include "routing/osrm_data_facade.hpp" - -#include "indexer/index.hpp" - -#include "geometry/point2d.hpp" - -#include "std/vector.hpp" - -#include "3party/osrm/osrm-backend/data_structures/query_edge.hpp" - -namespace routing -{ -/// Single graph node representation for routing task -struct FeatureGraphNode -{ - PhantomNode node; - OsrmMappingTypes::FtSeg segment; - m2::PointD segmentPoint; - Index::MwmId mwmId; - - /*! - * \brief fill FeatureGraphNode with values. - * \param nodeId osrm node idetifier. - * \param isStartNode true if this node will first in the path. - * \param mwmName @nodeId refers node on the graph of this map. - */ - FeatureGraphNode(NodeID const nodeId, bool const isStartNode, Index::MwmId const & id); - - FeatureGraphNode(NodeID const nodeId, NodeID const reverseNodeId, bool const isStartNode, - Index::MwmId const & id); - - /// \brief Invalid graph node constructor - FeatureGraphNode(); -}; - -/*! - * \brief The RawPathData struct is our representation of OSRM PathData struct. - * I use it for excluding dependency from OSRM. Contains OSRM node ID and it's weight. - */ -struct RawPathData -{ - NodeID node; - EdgeWeight segmentWeight; // Time in tenths of a second to pass |node|. - - RawPathData() : node(SPECIAL_NODEID), segmentWeight(INVALID_EDGE_WEIGHT) {} - - RawPathData(NodeID node, EdgeWeight segmentWeight) - : node(node), segmentWeight(segmentWeight) - { - } -}; - -//@todo (dragunov) make proper name -using TRoutingNodes = vector; -using TRawDataFacade = OsrmRawDataFacade; - -/*! - * \brief FindWeightsMatrix Find weights matrix from sources to targets. WARNING it finds only - * weights, not pathes. - * \param sources Sources graph nodes vector. Each source is the representation of a start OSRM - * node. - * \param targets Targets graph nodes vector. Each target is the representation of a finish OSRM - * node. - * \param facade Osrm data facade reference. - * \param packed Result vector with weights. Source nodes are rows. - * cost(source1 -> target1) cost(source1 -> target2) cost(source2 -> target1) cost(source2 -> - * target2) - */ -void FindWeightsMatrix(TRoutingNodes const & sources, TRoutingNodes const & targets, - TRawDataFacade & facade, vector & result); -} // namespace routing diff --git a/routing/router.hpp b/routing/router.hpp index ebd24a5e92..d52f1836d9 100644 --- a/routing/router.hpp +++ b/routing/router.hpp @@ -25,7 +25,7 @@ class Route; enum class RouterType { // @TODO It's necessary to rename Vehicle value to Car. - Vehicle = 0, /// For Car routing (OSRM or AStar). + Vehicle = 0, /// For Car routing. Pedestrian, /// For A star pedestrian routing. Bicycle, /// For A star bicycle routing. Taxi, /// For taxi route calculation Vehicle routing is used. diff --git a/routing/routing_benchmarks/CMakeLists.txt b/routing/routing_benchmarks/CMakeLists.txt index cb9def4e8c..510db77709 100644 --- a/routing/routing_benchmarks/CMakeLists.txt +++ b/routing/routing_benchmarks/CMakeLists.txt @@ -26,7 +26,6 @@ omim_link_libraries( geometry coding base - osrm jansson protobuf stats_client diff --git a/routing/routing_consistency_tests/CMakeLists.txt b/routing/routing_consistency_tests/CMakeLists.txt index 1e3bb63767..6e9236f55a 100644 --- a/routing/routing_consistency_tests/CMakeLists.txt +++ b/routing/routing_consistency_tests/CMakeLists.txt @@ -32,7 +32,6 @@ omim_link_libraries( opening_hours coding base - osrm jansson protobuf bsdiff diff --git a/routing/routing_integration_tests/CMakeLists.txt b/routing/routing_integration_tests/CMakeLists.txt index f01dbc587d..0d0d2505e1 100644 --- a/routing/routing_integration_tests/CMakeLists.txt +++ b/routing/routing_integration_tests/CMakeLists.txt @@ -11,7 +11,6 @@ set( SRC bicycle_route_test.cpp bicycle_turn_test.cpp - cross_section_tests.cpp get_altitude_test.cpp online_cross_tests.cpp pedestrian_route_test.cpp @@ -42,7 +41,6 @@ omim_link_libraries( geometry coding base - osrm jansson protobuf bsdiff diff --git a/routing/routing_integration_tests/cross_section_tests.cpp b/routing/routing_integration_tests/cross_section_tests.cpp deleted file mode 100644 index 03a4064311..0000000000 --- a/routing/routing_integration_tests/cross_section_tests.cpp +++ /dev/null @@ -1,229 +0,0 @@ -#include "testing/testing.hpp" - -#include "routing/cross_mwm_road_graph.hpp" -#include "routing/cross_routing_context.hpp" -#include "routing/osrm2feature_map.hpp" -#include "routing/routing_mapping.hpp" - -#include "storage/country_info_getter.hpp" - -#include "indexer/index.hpp" - -#include "platform/local_country_file.hpp" -#include "platform/local_country_file_utils.hpp" -#include "platform/platform.hpp" - -#include "geometry/point2d.hpp" - -#include "base/buffer_vector.hpp" -#include "base/logging.hpp" -#include "base/random.hpp" - -#include "std/limits.hpp" - -#include - -using namespace routing; - -namespace -{ -UNIT_TEST(CheckCrossSections) -{ - static double constexpr kPointEquality = 0.01; - vector localFiles; - platform::FindAllLocalMapsAndCleanup(numeric_limits::max() /* latestVersion */, - localFiles); - - size_t ingoingErrors = 0; - size_t outgoingErrors = 0; - size_t noRouting = 0; - for (auto & file : localFiles) - { - LOG(LINFO, ("Checking cross table for country:", file.GetCountryName())); - CrossRoutingContextReader crossReader; - - file.SyncWithDisk(); - if (file.GetFiles() != MapOptions::MapWithCarRouting) - { - noRouting++; - LOG(LINFO, ("Warning! Routing file not found for:", file.GetCountryName())); - continue; - } - FilesMappingContainer container(file.GetPath(MapOptions::CarRouting)); - if (!container.IsExist(ROUTING_CROSS_CONTEXT_TAG)) - { - noRouting++; - LOG(LINFO, ("Warning! Mwm file has not cross mwm section:", file.GetCountryName())); - continue; - } - crossReader.Load(container.GetReader(ROUTING_CROSS_CONTEXT_TAG)); - - bool error = false; - crossReader.ForEachIngoingNode([&error](IngoingCrossNode const & node) - { - if (node.m_point.EqualDxDy(ms::LatLon::Zero(), kPointEquality)) - error = true; - }); - if (error) - ingoingErrors++; - - error = false; - crossReader.ForEachOutgoingNode([&error](OutgoingCrossNode const & node) - { - if (node.m_point.EqualDxDy(ms::LatLon::Zero(), kPointEquality)) - error = true; - }); - if (error) - outgoingErrors++; - } - TEST_EQUAL(ingoingErrors, 0, ("Some countries have zero point incomes.")); - TEST_EQUAL(outgoingErrors, 0, ("Some countries have zero point exits.")); - LOG(LINFO, ("Found", localFiles.size(), "countries.", - noRouting, "countries are without routing file.")); -} - -UNIT_TEST(CheckOsrmToFeatureMapping) -{ - vector localFiles; - platform::FindAllLocalMapsAndCleanup(numeric_limits::max() /* latestVersion */, - localFiles); - - size_t errors = 0; - size_t checked = 0; - for (auto & file : localFiles) - { - LOG(LINFO, ("Checking nodes for country:", file.GetCountryName())); - - file.SyncWithDisk(); - if (file.GetFiles() != MapOptions::MapWithCarRouting) - { - LOG(LINFO, ("Warning! Routing file not found for:", file.GetCountryName())); - continue; - } - - FilesMappingContainer container(file.GetPath(MapOptions::CarRouting)); - if (!container.IsExist(ROUTING_FTSEG_FILE_TAG)) - { - LOG(LINFO, ("Warning! Mwm file has not routing ftseg section:", file.GetCountryName())); - continue; - } - ++checked; - - routing::TDataFacade dataFacade; - dataFacade.Load(container); - - OsrmFtSegMapping segMapping; - segMapping.Load(container, file); - segMapping.Map(container); - - for (unsigned i = 0; i < dataFacade.GetNumberOfNodes(); ++i) - { - buffer_vector buffer; - segMapping.ForEachFtSeg(i, MakeBackInsertFunctor(buffer)); - if (buffer.empty()) - { - LOG(LINFO, ("Error found in ", file.GetCountryName())); - errors++; - break; - } - } - } - LOG(LINFO, ("Found", localFiles.size(), "countries. In", checked, "maps with routing have", errors, "with errors.")); - TEST_EQUAL(errors, 0, ("Some countries have osrm and features mismatch.")); -} - -// The idea behind the test is -// 1. to go through all ingoing nodes of all mwms -// 2. to find all cross mwm outgoing edges for each ingoing node -// 3. to find all edges which are ingoing for the outgoing edges -// 4. to check that an edge mentioned in (1) there's between the ingoing edges -// Note. This test may take more than 3 hours. -UNIT_TEST(CrossMwmGraphTest) -{ - vector localFiles; - Index index; - platform::FindAllLocalMapsAndCleanup(numeric_limits::max() /* latestVersion */, - localFiles); - - for (platform::LocalCountryFile & file : localFiles) - { - file.SyncWithDisk(); - index.RegisterMap(file); - } - - for (auto it = localFiles.begin(); it != localFiles.end();) - { - string const & countryName = it->GetCountryName(); - MwmSet::MwmId const mwmId = index.GetMwmIdByCountryFile(it->GetCountryFile()); - if (countryName == "minsk-pass" || mwmId.GetInfo()->GetType() != MwmInfo::COUNTRY) - it = localFiles.erase(it); - else - ++it; - } - - Platform p; - unique_ptr infoGetter = - storage::CountryInfoReader::CreateCountryInfoReader(p); - auto countryFileGetter = [&infoGetter](m2::PointD const & pt) { - return infoGetter->GetRegionCountryId(pt); - }; - - RoutingIndexManager manager(countryFileGetter, index); - CrossMwmRoadGraph crossMwmGraph(manager); - - auto const seed = std::chrono::system_clock::now().time_since_epoch().count(); - LOG(LINFO, ("Seed for RandomSample:", seed)); - std::minstd_rand rng(static_cast(seed)); - std::vector subset = base::RandomSample(localFiles.size(), 10 /* mwm number */, rng); - std::vector subsetCountryFiles; - for (size_t i : subset) - subsetCountryFiles.push_back(localFiles[i]); - - for (platform::LocalCountryFile const & file : subsetCountryFiles) - { - string const & countryName = file.GetCountryName(); - LOG(LINFO, ("Processing", countryName)); - MwmSet::MwmId const mwmId = index.GetMwmIdByCountryFile(file.GetCountryFile()); - - TEST(mwmId.IsAlive(), ("Mwm name:", countryName, "Subset:", subsetCountryFiles)); - TRoutingMappingPtr currentMapping = manager.GetMappingById(mwmId); - if (!currentMapping->IsValid()) - continue; // No routing sections in the mwm. - - currentMapping->LoadCrossContext(); - currentMapping->FreeFileIfPossible(); - CrossRoutingContextReader const & currentContext = currentMapping->m_crossContext; - - size_t ingoingCounter = 0; - currentContext.ForEachIngoingNode([&](IngoingCrossNode const & node) { - ++ingoingCounter; - vector const & targets = - crossMwmGraph.ConstructBorderCross(currentMapping, node); - - for (BorderCross const & t : targets) - { - vector outAdjs; - crossMwmGraph.GetOutgoingEdgesList(t, outAdjs); - for (CrossWeightedEdge const & out : outAdjs) - { - vector inAdjs; - crossMwmGraph.GetIngoingEdgesList(out.GetTarget(), inAdjs); - TEST(find_if(inAdjs.cbegin(), inAdjs.cend(), - [&](CrossWeightedEdge const & e) { - return e.GetTarget() == t && out.GetWeight() == e.GetWeight(); - }) != inAdjs.cend(), - ("ForEachOutgoingNodeNearPoint() and ForEachIngoingNodeNearPoint() arn't " - "correlated. Mwm:", - countryName, "Subset:", subsetCountryFiles)); - } - } - }); - - size_t outgoingCounter = 0; - currentContext.ForEachOutgoingNode( - [&](OutgoingCrossNode const & /* node */) { ++outgoingCounter; }); - - LOG(LINFO, ("Processed:", countryName, "Exits:", outgoingCounter, "Enters:", ingoingCounter)); - } -} -} // namespace diff --git a/routing/routing_mapping.cpp b/routing/routing_mapping.cpp deleted file mode 100644 index cec3570c50..0000000000 --- a/routing/routing_mapping.cpp +++ /dev/null @@ -1,217 +0,0 @@ -#include "routing/routing_mapping.hpp" - -#include "routing/cross_routing_context.hpp" -#include "routing/osrm2feature_map.hpp" -#include "routing/osrm_data_facade.hpp" - -#include "platform/country_file.hpp" -#include "platform/local_country_file.hpp" -#include "platform/mwm_version.hpp" -#include "platform/platform.hpp" - -#include "coding/reader_wrapper.hpp" - -#include "base/logging.hpp" - - -using platform::CountryFile; -using platform::LocalCountryFile; - -namespace -{ -/*! - * \brief CheckMwmConsistency checks versions of mwm and routing files. - * \param localFile reference to country file we need to check. - * \return true if files has same versions. - * \warning Function assumes that the file lock was already taken. - */ -bool CheckMwmConsistency(LocalCountryFile const & localFile) -{ - if (version::IsSingleMwm(localFile.GetVersion())) - return true; - - ModelReaderPtr r1 = FilesContainerR(localFile.GetPath(MapOptions::CarRouting)) - .GetReader(VERSION_FILE_TAG); - ReaderSrc src1(r1.GetPtr()); - ModelReaderPtr r2 = FilesContainerR(localFile.GetPath(MapOptions::Map)) - .GetReader(VERSION_FILE_TAG); - ReaderSrc src2(r2.GetPtr()); - - version::MwmVersion version1; - version::ReadVersion(src1, version1); - - version::MwmVersion version2; - version::ReadVersion(src2, version2); - - return version1.GetVersion() == version2.GetVersion(); -} -} // namespace - -namespace routing -{ - -RoutingMapping::RoutingMapping(string const & countryFile, MwmSet & index) - : m_mapCounter(0), - m_facadeCounter(0), - m_crossContextLoaded(0), - m_countryFile(countryFile), - m_error(IRouter::ResultCode::RouteFileNotExist), - m_pIndex(&index) -{ - m_handle = index.GetMwmHandleByCountryFile(CountryFile(countryFile)); - if (!m_handle.IsAlive()) - return; - - LocalCountryFile const & localFile = m_handle.GetInfo()->GetLocalFile(); - - if (!HasOptions(localFile.GetFiles(), MapOptions::MapWithCarRouting)) - { - m_error = IRouter::ResultCode::RouteFileNotExist; - m_handle = MwmSet::MwmHandle(); - return; - } - - if (!CheckMwmConsistency(localFile)) - { - m_error = IRouter::ResultCode::InconsistentMWMandRoute; - m_handle = MwmSet::MwmHandle(); - return; - } - - m_container.Open(localFile.GetPath(MapOptions::CarRouting)); - if (!m_container.IsExist(ROUTING_MATRIX_FILE_TAG)) - { - m_error = IRouter::ResultCode::RouteFileNotExist; - m_handle = MwmSet::MwmHandle(); - m_container.Close(); - return; - } - - m_mwmId = m_handle.GetId(); - m_error = IRouter::ResultCode::NoError; -} - -void RoutingMapping::LoadFileIfNeeded() -{ - ASSERT(m_pIndex != nullptr, ()); - if (!m_handle.IsAlive() && m_mwmId.IsAlive()) - { - m_handle = m_pIndex->GetMwmHandleById(m_mwmId); - m_container.Open(m_mwmId.GetInfo()->GetLocalFile().GetPath(MapOptions::CarRouting)); - } -} - -void RoutingMapping::FreeFileIfPossible() -{ - if (m_mapCounter == 0 && m_facadeCounter == 0 && m_handle.IsAlive()) - { - m_handle = MwmSet::MwmHandle(); - m_container.Close(); - } -} - -RoutingMapping::~RoutingMapping() -{ - // Clear data while m_container is valid. - m_dataFacade.Clear(); - m_segMapping.Clear(); - m_container.Close(); -} - -void RoutingMapping::Map() -{ - LoadFileIfNeeded(); - if (!m_handle.IsAlive()) - return; - ++m_mapCounter; - if (!m_segMapping.IsMapped()) - { - m_segMapping.Load(m_container, m_handle.GetInfo()->GetLocalFile()); - m_segMapping.Map(m_container); - } -} - -void RoutingMapping::Unmap() -{ - --m_mapCounter; - if (m_mapCounter < 1 && m_segMapping.IsMapped()) - m_segMapping.Unmap(); - FreeFileIfPossible(); -} - -void RoutingMapping::LoadFacade() -{ - if (!m_facadeCounter) - { - LoadFileIfNeeded(); - if (!m_handle.IsAlive()) - return; - m_dataFacade.Load(m_container); - } - ++m_facadeCounter; -} - -void RoutingMapping::FreeFacade() -{ - --m_facadeCounter; - if (!m_facadeCounter) - { - FreeFileIfPossible(); - m_dataFacade.Clear(); - } -} - -void RoutingMapping::LoadCrossContext() -{ - if (m_crossContextLoaded) - return; - - LoadFileIfNeeded(); - - if (!m_handle.IsAlive()) - return; - - if (m_container.IsExist(ROUTING_CROSS_CONTEXT_TAG)) - { - m_crossContext.Load(m_container.GetReader(ROUTING_CROSS_CONTEXT_TAG)); - m_crossContextLoaded = true; - } -} - -void RoutingMapping::FreeCrossContext() -{ - m_crossContextLoaded = false; - m_crossContext = CrossRoutingContextReader(); - FreeFileIfPossible(); -} - -TRoutingMappingPtr RoutingIndexManager::GetMappingByPoint(m2::PointD const & point) -{ - string const name = m_countryFileFn(point); - if (name.empty()) - return TRoutingMappingPtr(new RoutingMapping()); - - return GetMappingByName(name); -} - -TRoutingMappingPtr RoutingIndexManager::GetMappingByName(string const & mapName) -{ - // Check if we have already loaded this file. - auto mapIter = m_mapping.find(mapName); - if (mapIter != m_mapping.end()) - return mapIter->second; - - // Or load and check file. - TRoutingMappingPtr newMapping(new RoutingMapping(mapName, m_index)); - m_mapping[mapName] = newMapping; - return newMapping; -} - -TRoutingMappingPtr RoutingIndexManager::GetMappingById(Index::MwmId const & id) -{ - if (!id.IsAlive()) - return TRoutingMappingPtr(new RoutingMapping()); - return GetMappingByName(id.GetInfo()->GetCountryName()); -} - -} // namespace routing diff --git a/routing/routing_mapping.hpp b/routing/routing_mapping.hpp deleted file mode 100644 index 9e7dc3d2ab..0000000000 --- a/routing/routing_mapping.hpp +++ /dev/null @@ -1,126 +0,0 @@ -#pragma once - -#include "routing/osrm2feature_map.hpp" -#include "routing/osrm_data_facade.hpp" -#include "routing/router.hpp" - -#include "indexer/index.hpp" - -#include "3party/osrm/osrm-backend/data_structures/query_edge.hpp" - -#include "std/algorithm.hpp" -#include "std/unordered_map.hpp" - - -namespace routing -{ -using TDataFacade = OsrmDataFacade; - -/// Datamapping and facade for single MWM and MWM.routing file -struct RoutingMapping -{ - TDataFacade m_dataFacade; - OsrmFtSegMapping m_segMapping; - CrossRoutingContextReader m_crossContext; - - /// Default constructor to create invalid instance for existing client code. - /// @postcondition IsValid() == false. - RoutingMapping() : m_error(IRouter::ResultCode::RouteFileNotExist), m_pIndex(nullptr) {} - /// @param countryFile Country file name without extension. - RoutingMapping(string const & countryFile, MwmSet & index); - ~RoutingMapping(); - - void Map(); - void Unmap(); - - void LoadFacade(); - void FreeFacade(); - - void LoadCrossContext(); - void FreeCrossContext(); - - bool IsValid() const { return m_error == IRouter::ResultCode::NoError && m_mwmId.IsAlive(); } - - IRouter::ResultCode GetError() const { return m_error; } - - /*! - * Returns mentioned country file. Works even the LocalCountryFile does not exist and - * the lock was not taken. - */ - string const & GetCountryName() const { return m_countryFile; } - - Index::MwmId const & GetMwmId() const { return m_mwmId; } - - // Free file handles if it is possible. Works only if there is no mapped sections. Cross section - // will be valid after free. - void FreeFileIfPossible(); - -private: - void LoadFileIfNeeded(); - - size_t m_mapCounter; - size_t m_facadeCounter; - bool m_crossContextLoaded; - string m_countryFile; - FilesMappingContainer m_container; - IRouter::ResultCode m_error; - MwmSet::MwmHandle m_handle; - // We save a mwmId for possibility to unlock a mwm file by rewriting m_handle. - Index::MwmId m_mwmId; - MwmSet * m_pIndex; -}; - -typedef shared_ptr TRoutingMappingPtr; - -//! \brief The MappingGuard struct. Asks mapping to load all data on construction and free it on destruction -class MappingGuard -{ - TRoutingMappingPtr const m_mapping; - -public: - MappingGuard(TRoutingMappingPtr const mapping): m_mapping(mapping) - { - m_mapping->Map(); - m_mapping->LoadFacade(); - } - - ~MappingGuard() - { - m_mapping->Unmap(); - m_mapping->FreeFacade(); - } -}; - -/*! Manager for loading, cashing and building routing indexes. - * Builds and shares special routing contexts. -*/ -class RoutingIndexManager -{ -public: - RoutingIndexManager(TCountryFileFn const & countryFileFn, MwmSet & index) - : m_countryFileFn(countryFileFn), m_index(index) - { - } - - TRoutingMappingPtr GetMappingByPoint(m2::PointD const & point); - - TRoutingMappingPtr GetMappingByName(string const & mapName); - - TRoutingMappingPtr GetMappingById(Index::MwmId const & id); - - template - void ForEachMapping(TFunctor toDo) - { - for_each(m_mapping.begin(), m_mapping.end(), toDo); - } - - void Clear() { m_mapping.clear(); } - -private: - TCountryFileFn m_countryFileFn; - // TODO (ldragunov) Rewrite to mwmId. - unordered_map m_mapping; - MwmSet & m_index; -}; - -} // namespace routing diff --git a/routing/routing_session.hpp b/routing/routing_session.hpp index 1ed72f5a4f..a6f87aa7bd 100644 --- a/routing/routing_session.hpp +++ b/routing/routing_session.hpp @@ -6,6 +6,8 @@ #include "routing/turns.hpp" #include "routing/turns_notification_manager.hpp" +#include "indexer/index.hpp" + #include "traffic/speed_groups.hpp" #include "traffic/traffic_cache.hpp" #include "traffic/traffic_info.hpp" diff --git a/routing/routing_tests/CMakeLists.txt b/routing/routing_tests/CMakeLists.txt index 7a60a3a97e..aa3d69e11e 100644 --- a/routing/routing_tests/CMakeLists.txt +++ b/routing/routing_tests/CMakeLists.txt @@ -1,7 +1,5 @@ project(routing_tests) -include_directories(${OMIM_ROOT}/3party/osrm/osrm-backend/include) - set( SRC applying_traffic_test.cpp @@ -12,7 +10,6 @@ set( checkpoint_predictor_test.cpp coding_test.cpp cross_mwm_connector_test.cpp - cross_routing_tests.cpp cumulative_restriction_test.cpp fake_graph_test.cpp followed_polyline_test.cpp @@ -21,7 +18,6 @@ set( index_graph_tools.hpp nearest_edge_finder_tests.cpp online_cross_fetcher_test.cpp - osrm_router_test.cpp restriction_test.cpp road_access_test.cpp road_graph_builder.cpp @@ -29,7 +25,6 @@ set( road_graph_nearest_edges_test.cpp route_tests.cpp routing_helpers_tests.cpp - routing_mapping_test.cpp routing_session_test.cpp turns_generator_test.cpp turns_sound_test.cpp @@ -52,7 +47,6 @@ omim_link_libraries( geometry coding base - osrm protobuf succinct opening_hours diff --git a/routing/routing_tests/async_router_test.cpp b/routing/routing_tests/async_router_test.cpp index 5ca3e475ae..1a441d9a16 100644 --- a/routing/routing_tests/async_router_test.cpp +++ b/routing/routing_tests/async_router_test.cpp @@ -14,6 +14,7 @@ #include "std/vector.hpp" using namespace routing; +using namespace std::placeholders; using ResultCode = routing::IRouter::ResultCode; diff --git a/routing/routing_tests/cross_routing_tests.cpp b/routing/routing_tests/cross_routing_tests.cpp deleted file mode 100644 index 15df806f59..0000000000 --- a/routing/routing_tests/cross_routing_tests.cpp +++ /dev/null @@ -1,176 +0,0 @@ -#include "testing/testing.hpp" - -#include "routing/cross_mwm_road_graph.hpp" -#include "routing/cross_mwm_router.hpp" -#include "routing/cross_routing_context.hpp" - -#include "coding/reader.hpp" -#include "coding/writer.hpp" - -using namespace routing; - -namespace -{ -// Graph to convertions. -UNIT_TEST(TestCrossRouteConverter) -{ - Index::MwmId aMap, bMap; - vector graphCrosses; - CrossNode const a(1, aMap, {0, 0}), b(2, aMap, {2, 2}); - CrossNode const c(3, bMap, {3, 3}), d(3, bMap, {4, 4}); - graphCrosses.emplace_back(a, b); - graphCrosses.emplace_back(b, c); - graphCrosses.emplace_back(c, d); - FeatureGraphNode startGraphNode; - startGraphNode.node.forward_node_id = 5; - startGraphNode.mwmId = aMap; - FeatureGraphNode finalGraphNode; - finalGraphNode.node.reverse_node_id = 6; - finalGraphNode.mwmId = bMap; - TCheckedPath route; - ConvertToSingleRouterTasks(graphCrosses, startGraphNode, finalGraphNode, route); - TEST_EQUAL(route.size(), 2, ("We have 2 maps aMap and bMap.")); - TEST_EQUAL(route.front().startNode.node, startGraphNode.node, - ("Start node must be replaced by origin.")); - TEST_EQUAL(route.back().finalNode.node, finalGraphNode.node, - ("End node must be replaced by origin.")); -} - -// Cross routing context tests. -UNIT_TEST(TestContextSerialization) -{ - routing::CrossRoutingContextWriter context; - routing::CrossRoutingContextReader newContext; - - context.AddIngoingNode(1, ms::LatLon::Zero()); - context.AddIngoingNode(2, ms::LatLon::Zero()); - context.AddOutgoingNode(3, "foo", ms::LatLon::Zero()); - context.AddOutgoingNode(4, "bar", ms::LatLon::Zero()); - context.ReserveAdjacencyMatrix(); - - vector buffer; - MemWriter > writer(buffer); - context.Save(writer); - TEST_GREATER(buffer.size(), 5, ("Context serializer make some data")); - - MemReader reader(buffer.data(), buffer.size()); - newContext.Load(reader); - vector ingoingNodes; - newContext.ForEachIngoingNode([&ingoingNodes](IngoingCrossNode const & node) - { - ingoingNodes.push_back(node); - }); - TEST_EQUAL(ingoingNodes.size(), 2, ()); - TEST_EQUAL(ingoingNodes[0].m_nodeId, 1, ()); - TEST_EQUAL(ingoingNodes[1].m_nodeId, 2, ()); - - vector outgoingNodes; - newContext.ForEachOutgoingNode([&outgoingNodes](OutgoingCrossNode const & node) - { - outgoingNodes.push_back(node); - }); - TEST_EQUAL(outgoingNodes.size(), 2, ()); - TEST_EQUAL(outgoingNodes[0].m_nodeId, 3, ()); - TEST_EQUAL(newContext.GetOutgoingMwmName(outgoingNodes[0]), string("foo"), ()); - TEST_EQUAL(outgoingNodes[1].m_nodeId, 4, ()); - TEST_EQUAL(newContext.GetOutgoingMwmName(outgoingNodes[1]), string("bar"), ()); -} - -UNIT_TEST(TestAdjacencyMatrix) -{ - routing::CrossRoutingContextWriter context; - routing::CrossRoutingContextReader newContext; - - context.AddIngoingNode(1, ms::LatLon::Zero()); - context.AddIngoingNode(2, ms::LatLon::Zero()); - context.AddIngoingNode(3, ms::LatLon::Zero()); - context.AddOutgoingNode(4, "foo", ms::LatLon::Zero()); - context.ReserveAdjacencyMatrix(); - { - auto ins = context.GetIngoingIterators(); - auto outs = context.GetOutgoingIterators(); - context.SetAdjacencyCost(ins.first, outs.first, 5); - context.SetAdjacencyCost(ins.first + 1, outs.first, 9); - } - - vector buffer; - MemWriter > writer(buffer); - context.Save(writer); - TEST_GREATER(buffer.size(), 5, ("Context serializer make some data")); - - MemReader reader(buffer.data(), buffer.size()); - newContext.Load(reader); - vector ingoingNodes; - newContext.ForEachIngoingNode([&ingoingNodes](IngoingCrossNode const & node) - { - ingoingNodes.push_back(node); - }); - vector outgoingNodes; - newContext.ForEachOutgoingNode([&outgoingNodes](OutgoingCrossNode const & node) - { - outgoingNodes.push_back(node); - }); - TEST_EQUAL(newContext.GetAdjacencyCost(ingoingNodes[0], outgoingNodes[0]), 5, ()); - TEST_EQUAL(newContext.GetAdjacencyCost(ingoingNodes[1], outgoingNodes[0]), 9, ()); - TEST_EQUAL(newContext.GetAdjacencyCost(ingoingNodes[2], outgoingNodes[0]), - routing::kInvalidContextEdgeWeight, ("Default cost")); -} - -UNIT_TEST(TestFindingByPoint) -{ - routing::CrossRoutingContextWriter context; - routing::CrossRoutingContextReader newContext; - - ms::LatLon p1(1., 1.), p2(5., 5.), p3(10., 1.), p4(20., 1.); - - context.AddIngoingNode(1, ms::LatLon::Zero()); - context.AddIngoingNode(2, p1); - context.AddIngoingNode(3, p2); - context.AddOutgoingNode(4, "foo", ms::LatLon::Zero()); - context.AddIngoingNode(5, p3); - context.AddIngoingNode(6, p3); - context.ReserveAdjacencyMatrix(); - - vector buffer; - MemWriter > writer(buffer); - context.Save(writer); - TEST_GREATER(buffer.size(), 5, ("Context serializer make some data")); - - MemReader reader(buffer.data(), buffer.size()); - newContext.Load(reader); - vector node; - auto fn = [&node](IngoingCrossNode const & nd) {node.push_back(nd);}; - TEST(newContext.ForEachIngoingNodeNearPoint(p1, fn), ()); - TEST_EQUAL(node.size(), 1, ()); - TEST_EQUAL(node[0].m_nodeId, 2, ()); - node.clear(); - TEST(newContext.ForEachIngoingNodeNearPoint(p2, fn), ()); - TEST_EQUAL(node.size(), 1, ()); - TEST_EQUAL(node[0].m_nodeId, 3, ()); - node.clear(); - TEST(!newContext.ForEachIngoingNodeNearPoint(p4, fn), ()); - node.clear(); - TEST(newContext.ForEachIngoingNodeNearPoint(p3, fn), ()); - TEST_EQUAL(node.size(), 2, ()); - TEST_EQUAL(node[0].m_nodeId, 5, ()); - TEST_EQUAL(node[1].m_nodeId, 6, ()); - - vector outgoingNode; - auto fnOutgoing = [&outgoingNode](OutgoingCrossNode const & nd) {outgoingNode.push_back(nd);}; - TEST(newContext.ForEachOutgoingNodeNearPoint(ms::LatLon::Zero(), fnOutgoing), ()); - TEST_EQUAL(outgoingNode.size(), 1, ()); - TEST_EQUAL(outgoingNode[0].m_nodeId, 4, ()); - - outgoingNode.clear(); - TEST(!newContext.ForEachOutgoingNodeNearPoint(p3, fnOutgoing), ()); - TEST(outgoingNode.empty(), ()); -} -UNIT_TEST(TestCrossWeightedEdgeComparator) -{ - CrossNode const from(0, Index::MwmId(), ms::LatLon::Zero()); - CrossWeightedEdge const lhs(BorderCross(from, {1, Index::MwmId(), ms::LatLon::Zero()}), 2.0); - CrossWeightedEdge const rhs(BorderCross(from, {2, Index::MwmId(), ms::LatLon::Zero()}), 1.0); - - TEST(lhs < rhs || rhs < lhs || lhs == rhs, ()); -} -} // namespace diff --git a/routing/routing_tests/osrm_router_test.cpp b/routing/routing_tests/osrm_router_test.cpp deleted file mode 100644 index 175e7116b5..0000000000 --- a/routing/routing_tests/osrm_router_test.cpp +++ /dev/null @@ -1,261 +0,0 @@ -#include "testing/testing.hpp" - -#include "routing/index_router.hpp" - -#include "indexer/features_offsets_table.hpp" -#include "geometry/mercator.hpp" - -#include "platform/country_file.hpp" -#include "platform/local_country_file.hpp" -#include "platform/local_country_file_utils.hpp" -#include "platform/platform.hpp" - -#include "platform/platform_tests_support/scoped_mwm.hpp" - -#include "coding/file_writer.hpp" - -#include "defines.hpp" - -#include "base/checked_cast.hpp" -#include "base/scope_guard.hpp" - -#include "std/bind.hpp" -#include "std/unique_ptr.hpp" -#include "std/vector.hpp" - -using namespace routing; -using platform::CountryIndexes; - -namespace -{ - -typedef vector InputDataT; -typedef vector< vector > NodeIdDataT; -typedef vector< pair > RangeDataT; -typedef OsrmMappingTypes::FtSeg SegT; - -void TestNodeId(OsrmFtSegMapping const & mapping, NodeIdDataT const & test) -{ - for (TOsrmNodeId nodeId = 0; nodeId < test.size(); ++nodeId) - { - for (auto idx : test[nodeId]) - TEST_EQUAL(nodeId, mapping.GetNodeId(idx), ()); - } -} - -void TestSegmentRange(OsrmFtSegMapping const & mapping, RangeDataT const & test) -{ - for (TOsrmNodeId nodeId = 0; nodeId < test.size(); ++nodeId) - { - // Input test range is { start, count } but we should pass [start, end). - auto const & r = test[nodeId]; - TEST_EQUAL(mapping.GetSegmentsRange(nodeId), make_pair(r.first, r.first + r.second), ()); - } -} - -void TestMapping(InputDataT const & data, - NodeIdDataT const & nodeIds, - RangeDataT const & ranges) -{ - platform::CountryFile country("TestCountry"); - platform::LocalCountryFile localFile(GetPlatform().WritableDir(), country, 0 /* version */); - localFile.SyncWithDisk(); - platform::tests_support::ScopedMwm mapMwm( - platform::GetFileName(localFile.GetCountryFile().GetName(), MapOptions::Map, - version::FOR_TESTING_TWO_COMPONENT_MWM1)); - static string const ftSegsPath = GetPlatform().WritablePathForFile("test1.tmp"); - - platform::CountryIndexes::PreparePlaceOnDisk(localFile); - string const & featuresOffsetsTablePath = - CountryIndexes::GetPath(localFile, CountryIndexes::Index::Offsets); - MY_SCOPE_GUARD(ftSegsFileDeleter, bind(FileWriter::DeleteFileX, ftSegsPath)); - MY_SCOPE_GUARD(featuresOffsetsTableFileDeleter, - bind(FileWriter::DeleteFileX, featuresOffsetsTablePath)); - MY_SCOPE_GUARD(indexesDeleter, bind(&CountryIndexes::DeleteFromDisk, localFile)); - - { - // Prepare fake features offsets table for input data, because - // OsrmFtSegMapping::Load() loads routing index and creates - // additional helper indexes and some of them require - // FeatureOffsetsTable existence. - // - // As instantiation of FeatureOffsetsTable requires complete MWM - // file with features or at least already searialized - // FeatureOffsetsTable, the purpose of this code is to prepare a - // file with serialized FeatureOffsetsTable and feed it to - // OsrmFtSegMapping. - feature::FeaturesOffsetsTable::Builder tableBuilder; - for (auto const & segVector : data) - { - for (auto const & seg : segVector) - tableBuilder.PushOffset(seg.m_fid); - } - unique_ptr table = - feature::FeaturesOffsetsTable::Build(tableBuilder); - table->Save(featuresOffsetsTablePath); - } - - OsrmFtSegMappingBuilder builder; - for (TOsrmNodeId nodeId = 0; nodeId < data.size(); ++nodeId) - builder.Append(nodeId, data[nodeId]); - - TestNodeId(builder, nodeIds); - TestSegmentRange(builder, ranges); - - { - FilesContainerW w(ftSegsPath); - builder.Save(w); - } - - { - FilesMappingContainer cont(ftSegsPath); - OsrmFtSegMapping mapping; - mapping.Load(cont, localFile); - mapping.Map(cont); - - TestNodeId(mapping, nodeIds); - TestSegmentRange(mapping, ranges); - - for (size_t i = 0; i < mapping.GetSegmentsCount(); ++i) - { - TOsrmNodeId const node = mapping.GetNodeId(base::checked_cast(i)); - size_t count = 0; - mapping.ForEachFtSeg(node, [&] (OsrmMappingTypes::FtSeg const & s) - { - TEST_EQUAL(s, data[node][count++], ()); - }); - TEST_EQUAL(count, data[node].size(), ()); - } - } -} - -bool TestFtSeg(SegT const & s) -{ - return (SegT(s.Store()) == s); -} - -} - -UNIT_TEST(FtSeg_Smoke) -{ - SegT arr[] = { - { 5, 1, 2 }, - { 666, 0, 17 }, - }; - - for (size_t i = 0; i < ARRAY_SIZE(arr); ++i) - TEST(TestFtSeg(arr[i]), (arr[i].Store())); -} - -UNIT_TEST(OsrmFtSegMappingBuilder_Smoke) -{ - { - InputDataT data = - { - { {0, 0, 1} }, - { {1, 0, 1} }, - { {2, 0, 1}, {3, 0, 1} }, - { {4, 0, 1} }, - { {5, 0, 1}, {6, 0, 1}, {7, 0, 1} }, - { {8, 0, 1}, {9, 0, 1}, {10, 0, 1}, {11, 0, 1} }, - { {12, 0, 1} } - }; - - NodeIdDataT nodeIds = - { - { 0 }, - { 1 }, - { 2, 3 }, - { 4 }, - { 5, 6, 7 }, - { 8, 9, 10, 11 }, - { 12 } - }; - - RangeDataT ranges = - { - { 0, 1 }, - { 1, 1 }, - { 2, 2 }, - { 4, 1 }, - { 5, 3 }, - { 8, 4 }, - { 12, 1 }, - }; - - TestMapping(data, nodeIds, ranges); - } - - - { - InputDataT data = - { - { {0, 0, 1} }, - { {1, 0, 1} }, - { {2, 0, 1} }, - { {3, 0, 1} }, - { {4, 0, 1} }, - { {5, 0, 1}, {6, 0, 1} }, - { {7, 0, 1} }, - { {8, 0, 1}, {9, 0, 1}, {10, 0, 1} }, - { {11, 0, 1}, {12, 0, 1}, {13, 0, 1} } - }; - - NodeIdDataT nodeIds = - { - { 0 }, - { 1 }, - { 2 }, - { 3 }, - { 4 }, - { 5, 6 }, - { 7 }, - { 8, 9, 10 }, - { 11, 12, 13 } - }; - - RangeDataT ranges = - { - { 0, 1 }, - { 1, 1 }, - { 2, 1 }, - { 3, 1 }, - { 4, 1 }, - { 5, 2 }, - { 7, 1 }, - { 8, 3 }, - { 11, 3 }, - }; - - TestMapping(data, nodeIds, ranges); - } - - - { - InputDataT data = - { - { {0, 0, 1}, {1, 2, 3} }, - { }, - { {3, 6, 7} }, - { {4, 8, 9}, {5, 10, 11} }, - }; - - NodeIdDataT nodeIds = - { - { 0, 1 }, - { }, - { 3 }, - { 4, 5 }, - }; - - RangeDataT ranges = - { - { 0, 2 }, - { 2, 1 }, - { 3, 1 }, - { 4, 2 }, - }; - - TestMapping(data, nodeIds, ranges); - } -} diff --git a/routing/routing_tests/routing_mapping_test.cpp b/routing/routing_tests/routing_mapping_test.cpp deleted file mode 100644 index 3a24198858..0000000000 --- a/routing/routing_tests/routing_mapping_test.cpp +++ /dev/null @@ -1,126 +0,0 @@ -#include "testing/testing.hpp" -#include "indexer/indexer_tests/test_mwm_set.hpp" - -#include "map/feature_vec_model.hpp" - -#include "routing/routing_mapping.hpp" - -#include "platform/country_file.hpp" -#include "platform/local_country_file.hpp" -#include "platform/platform.hpp" -#include "platform/platform_tests_support/scoped_file.hpp" - -#include "coding/file_name_utils.hpp" -#include "coding/file_writer.hpp" -#include "coding/internal/file_data.hpp" - -using namespace routing; -using namespace tests; -using namespace platform; -using namespace platform::tests_support; - -namespace -{ -class LocalFileGenerator -{ -public: - LocalFileGenerator(string const & fileName) - : m_countryFile(fileName) - , m_testDataFile(platform::GetFileName(m_countryFile.GetName(), MapOptions::MapWithCarRouting, - version::FOR_TESTING_SINGLE_MWM1), - ScopedFile::Mode::Create) - , m_localFile(GetPlatform().WritableDir(), m_countryFile, version::FOR_TESTING_SINGLE_MWM1) - { - m_localFile.SyncWithDisk(); - TEST(m_localFile.OnDisk(MapOptions::MapWithCarRouting), ()); - GenerateNecessarySections(m_localFile); - - m_result = m_testSet.Register(m_localFile); - TEST_EQUAL(m_result.second, MwmSet::RegResult::Success, - ("Can't register temporary localFile map")); - } - - TestMwmSet & GetMwmSet() { return m_testSet; } - - string const & GetCountryName() { return m_countryFile.GetName(); } - - size_t GetNumRefs() { return m_result.first.GetInfo()->GetNumRefs(); } - -private: - - void GenerateNecessarySections(LocalCountryFile const & localFile) - { - FilesContainerW dataCont(localFile.GetPath(MapOptions::CarRouting)); - - FileWriter w1 = dataCont.GetWriter(VERSION_FILE_TAG); - version::WriteVersion(w1, my::SecondsSinceEpoch()); - FileWriter w2 = dataCont.GetWriter(ROUTING_MATRIX_FILE_TAG); - w2.Write("smth", 4); - } - - CountryFile m_countryFile; - ScopedFile m_testDataFile; - LocalCountryFile m_localFile; - TestMwmSet m_testSet; - pair m_result; -}; - -UNIT_TEST(RoutingMappingCountryFileLockTest) -{ - LocalFileGenerator generator("1TestCountry"); - { - RoutingMapping testMapping(generator.GetCountryName(), (generator.GetMwmSet())); - TEST(testMapping.IsValid(), ()); - TEST_EQUAL(generator.GetNumRefs(), 1, ()); - } - // Routing mapping must unlock the file after destruction. - TEST_EQUAL(generator.GetNumRefs(), 0, ()); -} - -UNIT_TEST(IndexManagerLockManagementTest) -{ - string const fileName("1TestCountry"); - LocalFileGenerator generator(fileName); - RoutingIndexManager manager([&fileName](m2::PointD const & q) { return fileName; }, - generator.GetMwmSet()); - { - auto testMapping = manager.GetMappingByName(fileName); - TEST(testMapping->IsValid(), ()); - TEST_EQUAL(generator.GetNumRefs(), 1, ()); - } - // We freed mapping, but it still persists inside the manager cache. - TEST_EQUAL(generator.GetNumRefs(), 1, ()); - - // Test cache clearing. - manager.Clear(); - TEST_EQUAL(generator.GetNumRefs(), 0, ()); -} - -UNIT_TEST(FtSegIsInsideTest) -{ - OsrmMappingTypes::FtSeg seg(123, 1, 5); - OsrmMappingTypes::FtSeg inside(123, 1, 2); - TEST(OsrmMappingTypes::IsInside(seg, inside), ()); - OsrmMappingTypes::FtSeg inside2(123, 3, 4); - TEST(OsrmMappingTypes::IsInside(seg, inside2), ()); - OsrmMappingTypes::FtSeg inside3(123, 4, 5); - TEST(OsrmMappingTypes::IsInside(seg, inside3), ()); - OsrmMappingTypes::FtSeg bseg(123, 5, 1); - TEST(OsrmMappingTypes::IsInside(bseg, inside), ()); - TEST(OsrmMappingTypes::IsInside(bseg, inside2), ()); - TEST(OsrmMappingTypes::IsInside(bseg, inside3), ()); -} - -UNIT_TEST(FtSegSplitSegmentiTest) -{ - OsrmMappingTypes::FtSeg seg(123, 1, 5); - OsrmMappingTypes::FtSeg bseg(123, 5, 1); - OsrmMappingTypes::FtSeg splitter(123, 2, 3); - - OsrmMappingTypes::FtSeg res1(123, 1, 3); - TEST_EQUAL(res1, OsrmMappingTypes::SplitSegment(seg, splitter), ()); - - OsrmMappingTypes::FtSeg res2(123, 5, 2); - TEST_EQUAL(res2, OsrmMappingTypes::SplitSegment(bseg, splitter), ()); -} -} // namespace diff --git a/routing/turns_generator.cpp b/routing/turns_generator.cpp index a24027f978..fd3195dceb 100644 --- a/routing/turns_generator.cpp +++ b/routing/turns_generator.cpp @@ -1,6 +1,7 @@ -#include "routing/routing_mapping.hpp" #include "routing/turns_generator.hpp" +#include "routing/router.hpp" + #include "routing_common/car_model.hpp" #include "indexer/ftypes_matcher.hpp" @@ -14,8 +15,6 @@ #include "std/numeric.hpp" #include "std/string.hpp" -#include "3party/osrm/osrm-backend/data_structures/internal_route_result.hpp" - using namespace routing; using namespace routing::turns; diff --git a/routing/turns_generator.hpp b/routing/turns_generator.hpp index fc284a49ce..b221c18f34 100644 --- a/routing/turns_generator.hpp +++ b/routing/turns_generator.hpp @@ -1,8 +1,6 @@ #pragma once #include "routing/loaded_path_segment.hpp" -#include "routing/osrm2feature_map.hpp" -#include "routing/osrm_engine.hpp" #include "routing/routing_result_graph.hpp" #include "routing/route.hpp" #include "routing/router.hpp" diff --git a/search/search_quality/assessment_tool/CMakeLists.txt b/search/search_quality/assessment_tool/CMakeLists.txt index 2b6b47a101..821a3df09a 100644 --- a/search/search_quality/assessment_tool/CMakeLists.txt +++ b/search/search_quality/assessment_tool/CMakeLists.txt @@ -70,7 +70,6 @@ omim_link_libraries( oauthcpp opening_hours openlr - osrm protobuf pugixml sdf_image diff --git a/storage/storage_integration_tests/CMakeLists.txt b/storage/storage_integration_tests/CMakeLists.txt index 5ce409ea3f..3fe9065274 100644 --- a/storage/storage_integration_tests/CMakeLists.txt +++ b/storage/storage_integration_tests/CMakeLists.txt @@ -48,7 +48,6 @@ omim_link_libraries( agg jansson protobuf - osrm stats_client minizip succinct diff --git a/storage/storage_tests/CMakeLists.txt b/storage/storage_tests/CMakeLists.txt index ca66f18a16..7913f7e5dd 100644 --- a/storage/storage_tests/CMakeLists.txt +++ b/storage/storage_tests/CMakeLists.txt @@ -36,7 +36,6 @@ omim_link_libraries( ugc indexer platform_tests_support - osrm editor oauthcpp platform diff --git a/track_analyzing/track_analyzer/CMakeLists.txt b/track_analyzing/track_analyzer/CMakeLists.txt index a293fbd0c6..f2c57a2f2c 100644 --- a/track_analyzing/track_analyzer/CMakeLists.txt +++ b/track_analyzing/track_analyzer/CMakeLists.txt @@ -35,7 +35,6 @@ omim_link_libraries( icu jansson protobuf - osrm stats_client succinct pugixml