From ec50effa4b860f2f5b7579f22eb5beaec3337824 Mon Sep 17 00:00:00 2001 From: Vladimir Byko-Ianko Date: Mon, 27 Nov 2017 14:58:25 +0300 Subject: [PATCH] Implementation CalcCrossMwmTransitions(). --- generator/routing_index_generator.cpp | 62 ++++++++++++++++++++++++++- generator/routing_index_generator.hpp | 5 +++ generator/transit_generator.cpp | 10 ----- routing/vehicle_mask.hpp | 1 + routing_common/transit_types.hpp | 15 +++++++ 5 files changed, 81 insertions(+), 12 deletions(-) diff --git a/generator/routing_index_generator.cpp b/generator/routing_index_generator.cpp index 82310d1168..6fa908dd80 100644 --- a/generator/routing_index_generator.cpp +++ b/generator/routing_index_generator.cpp @@ -16,15 +16,19 @@ #include "routing_common/bicycle_model.hpp" #include "routing_common/car_model.hpp" #include "routing_common/pedestrian_model.hpp" +#include "routing_common/transit_serdes.hpp" #include "indexer/coding_params.hpp" #include "indexer/data_header.hpp" #include "indexer/feature.hpp" #include "indexer/feature_processor.hpp" +#include "geometry/point2d.hpp" + #include "coding/file_container.hpp" #include "coding/file_name_utils.hpp" #include "coding/point_to_integer.hpp" +#include "coding/reader.hpp" #include "base/checked_cast.hpp" #include "base/logging.hpp" @@ -262,9 +266,63 @@ void CalcCrossMwmTransitions(string const & mwmFile, string const & mappingFile, CountryParentNameGetterFn const & countryParentNameGetterFn, vector> & transitions) { + LOG(LINFO, ("CalcCrossMwmTransitions(", mwmFile, "...)")); CHECK(mappingFile.empty(), ()); - NOTIMPLEMENTED(); - // @todo(bykoianko) Filling |transitions| based on transit section should be implemented. + try + { + FilesContainerR cont(mwmFile); + CHECK(cont.IsExist(TRANSIT_FILE_TAG), + (TRANSIT_FILE_TAG, "should be generated before", TRANSIT_CROSS_MWM_FILE_TAG)); + auto reader = cont.GetReader(TRANSIT_FILE_TAG); + ReaderSource src(reader); + + transit::FixedSizeDeserializer> numberDeserializer(src); + transit::TransitHeader header; + numberDeserializer(header); + CHECK(header.IsValid(), ("TransitHeader is not valid.", header)); + + transit::Deserializer> deserializer(src); + vector stops; + deserializer(stops); + CHECK(IsValidSortedUnique(stops), ("Transit stops are not valid. Mwm:", mwmFile)); + src.Skip(header.m_edgesOffset - header.m_gatesOffset); // Skipping gates. + vector edges; + deserializer(edges); + CHECK(IsValidSortedUnique(edges), ("Transit edges are not valid. Mwm:", mwmFile)); + + auto const getStopIdPoint = [&stops] (transit::StopId stopId) -> m2::PointD const & { + auto const it = equal_range(stops.cbegin(), stops.cend(), transit::Stop(stopId)); + CHECK_EQUAL(distance(it.first, it.second), 1, + ("An stop with id:", stopId, "is not unique or there's not such item. stops:", stops)); + return it.first->GetPoint(); + }; + + LOG(LINFO, ("CalcCrossMwmTransitions() edges.size():", edges.size())); + + // Index |i| is zero based edge index. This zero based index should be increased with + // |FakeFeatureIds::kTransitGraphFeaturesStart| and then used in Segment class as + // feature id in transit case.Ø + for (size_t i = 0; i < edges.size(); ++i) + { + auto const & e = edges[i]; + m2::PointD const stop1Point = getStopIdPoint(e.GetStop1Id()); + m2::PointD const stop2Point = getStopIdPoint(e.GetStop2Id()); + bool const stop2In = m2::RegionsContain(borders, stop2Point); + if (m2::RegionsContain(borders, stop1Point) == stop2In) + continue; + + LOG(LINFO, ("CalcCrossMwmTransitions() crossing edge:", e)); + // Note. One way mask is set to kTransitMask because all transit edges are one way edges. + transitions.emplace_back(connector::TransitId(e.GetStop1Id(), e.GetStop2Id(), e.GetLineId()), + i /* feature id */, 0 /* segment index */, kTransitMask, + kTransitMask /* one way mask */, stop2In /* forward is enter */, + stop1Point, stop2Point); + } + } + catch (Reader::OpenException const & e) + { + CHECK(false, ("Error while reading", TRANSIT_FILE_TAG, "section.", e.Msg())); + } } /// \brief Fills |transitions| and |connectors| params. diff --git a/generator/routing_index_generator.hpp b/generator/routing_index_generator.hpp index 659708cb5d..1f23caf501 100644 --- a/generator/routing_index_generator.hpp +++ b/generator/routing_index_generator.hpp @@ -10,11 +10,16 @@ using CountryParentNameGetterFn = std::function -bool IsValid(vector const & items) -{ - return all_of(items.cbegin(), items.cend(), [](Item const & item) { return item.IsValid(); }); -} - struct ClearVisitor { template @@ -628,10 +622,6 @@ void ProcessGraph(string const & mwmPath, TCountryId const & countryId, void BuildTransit(string const & mwmDir, TCountryId const & countryId, string const & osmIdToFeatureIdsPath, string const & transitDir) { - LOG(LERROR, ("This method is under construction and should not be used for building production mwm " - "sections.")); - NOTIMPLEMENTED(); - Platform::FilesList graphFiles; Platform::GetFilesByExt(my::AddSlashIfNeeded(transitDir), TRANSIT_FILE_EXTENSION, graphFiles); diff --git a/routing/vehicle_mask.hpp b/routing/vehicle_mask.hpp index 11046157fa..246a153e5e 100644 --- a/routing/vehicle_mask.hpp +++ b/routing/vehicle_mask.hpp @@ -29,6 +29,7 @@ VehicleMask constexpr kAllVehiclesMask = kNumVehicleMasks - 1; VehicleMask constexpr kPedestrianMask = GetVehicleMask(VehicleType::Pedestrian); VehicleMask constexpr kBicycleMask = GetVehicleMask(VehicleType::Bicycle); VehicleMask constexpr kCarMask = GetVehicleMask(VehicleType::Car); +VehicleMask constexpr kTransitMask = GetVehicleMask(VehicleType::Transit); std::string DebugPrint(VehicleType vehicleType); std::string ToString(VehicleType vehicleType); diff --git a/routing_common/transit_types.hpp b/routing_common/transit_types.hpp index 55389cd520..a695c995aa 100644 --- a/routing_common/transit_types.hpp +++ b/routing_common/transit_types.hpp @@ -7,9 +7,11 @@ #include "base/newtype.hpp" #include "base/visitor.hpp" +#include #include #include #include +#include namespace routing { @@ -460,6 +462,19 @@ private: std::string m_title; }; +template +bool IsValid(std::vector const & items) +{ + return std::all_of(items.cbegin(), items.cend(), [](Item const & item) { return item.IsValid(); }); +} + +template +bool IsValidSortedUnique(std::vector const & items) +{ + return IsValid(items) && std::is_sorted(items.cbegin(), items.cend()) && + std::adjacent_find(items.cbegin(), items.cend()) == items.cend(); +} + EdgeFlags GetEdgeFlags(bool transfer, StopId stopId1, StopId stopId2, std::vector const & shapeIds);