diff --git a/generator/generator_tests/maxspeed_tests.cpp b/generator/generator_tests/maxspeed_tests.cpp index c9e36faa11..f86fec67e8 100644 --- a/generator/generator_tests/maxspeed_tests.cpp +++ b/generator/generator_tests/maxspeed_tests.cpp @@ -3,6 +3,7 @@ #include "generator/generator_tests_support/test_feature.cpp" #include "generator/generator_tests_support/test_mwm_builder.hpp" #include "generator/maxspeed_builder.hpp" +#include "generator/maxspeed_parser.hpp" #include "generator/routing_helpers.hpp" #include "routing/maxspeeds_serialization.hpp" @@ -40,6 +41,7 @@ namespace { +using namespace generator; using namespace measurement_utils; using namespace platform::tests_support; using namespace platform; @@ -127,6 +129,53 @@ bool ParseCsv(string const & maxspeedCsvContent, OsmIdToMaxspeed & mapping) return ParseMaxspeeds(base::JoinPath(testDirFullPath, kCsv), mapping); } +UNIT_TEST(MaxspeedTagValueToSpeedTest) +{ + SpeedInUnits speed; + + TEST(ParseMaxspeedTag("RU:rural", speed), ()); + TEST_EQUAL(speed, SpeedInUnits(90, Units::Metric), ()); + + TEST(ParseMaxspeedTag("90", speed), ()); + TEST_EQUAL(speed, SpeedInUnits(90, Units::Metric), ()); + + TEST(ParseMaxspeedTag("90 ", speed), ()); + TEST_EQUAL(speed, SpeedInUnits(90, Units::Metric), ()); + + TEST(ParseMaxspeedTag("60kmh", speed), ()); + TEST_EQUAL(speed, SpeedInUnits(60, Units::Metric), ()); + + TEST(ParseMaxspeedTag("60 kmh", speed), ()); + TEST_EQUAL(speed, SpeedInUnits(60, Units::Metric), ()); + + TEST(ParseMaxspeedTag("60 kmh", speed), ()); + TEST_EQUAL(speed, SpeedInUnits(60, Units::Metric), ()); + + TEST(ParseMaxspeedTag("60 kmh and some other string", speed), ()); + TEST_EQUAL(speed, SpeedInUnits(60, Units::Metric), ()); + + TEST(ParseMaxspeedTag("75mph", speed), ()); + TEST_EQUAL(speed, SpeedInUnits(75, Units::Imperial), ()); + + TEST(ParseMaxspeedTag("75 mph", speed), ()); + TEST_EQUAL(speed, SpeedInUnits(75, Units::Imperial), ()); + + TEST(ParseMaxspeedTag("75 mph", speed), ()); + TEST_EQUAL(speed, SpeedInUnits(75, Units::Imperial), ()); + + TEST(ParseMaxspeedTag("75 mph and some other string", speed), ()); + TEST_EQUAL(speed, SpeedInUnits(75, Units::Imperial), ()); + + TEST(ParseMaxspeedTag("75mph", speed), ()); + TEST_EQUAL(speed, SpeedInUnits(75, Units::Imperial), ()); + + TEST(!ParseMaxspeedTag("some other string", speed), ()); + TEST(!ParseMaxspeedTag("60 kmph", speed), ()); + TEST(!ParseMaxspeedTag("1234567890 kmh", speed), ()); + TEST(!ParseMaxspeedTag("1234567890 mph", speed), ()); + TEST(!ParseMaxspeedTag("1234567890", speed), ()); +} + UNIT_TEST(ParseMaxspeeds_Smoke) { string const maxspeedCsvContent; diff --git a/generator/generator_tests/speed_cameras_test.cpp b/generator/generator_tests/speed_cameras_test.cpp index 935e734a8a..6118e23a09 100644 --- a/generator/generator_tests/speed_cameras_test.cpp +++ b/generator/generator_tests/speed_cameras_test.cpp @@ -476,60 +476,4 @@ UNIT_TEST(RoadCategoryToSpeedTest) TEST(!RoadCategoryToSpeed("UNKNOWN:unknown", speed), ()); } - -UNIT_TEST(MaxspeedValueToSpeedTest) -{ - SpeedInUnits speed; - - TEST(ParseMaxspeedTag("RU:rural", speed), ()); - TEST_EQUAL(speed, SpeedInUnits(90, Units::Metric), ()); - - TEST(ParseMaxspeedTag("90", speed), ()); - TEST_EQUAL(speed, SpeedInUnits(90, Units::Metric), ()); - - TEST(ParseMaxspeedTag("90 ", speed), ()); - TEST_EQUAL(speed, SpeedInUnits(90, Units::Metric), ()); - - TEST(ParseMaxspeedTag("60kmh", speed), ()); - TEST_EQUAL(speed, SpeedInUnits(60, Units::Metric), ()); - - TEST(ParseMaxspeedTag("60 kmh", speed), ()); - TEST_EQUAL(speed, SpeedInUnits(60, Units::Metric), ()); - - TEST(ParseMaxspeedTag("60 kmh", speed), ()); - TEST_EQUAL(speed, SpeedInUnits(60, Units::Metric), ()); - - TEST(ParseMaxspeedTag("60 kmh and some other string", speed), ()); - TEST_EQUAL(speed, SpeedInUnits(60, Units::Metric), ()); - - TEST(ParseMaxspeedTag("75mph", speed), ()); - TEST_EQUAL(speed, SpeedInUnits(75, Units::Imperial), ()); - - TEST(ParseMaxspeedTag("75 mph", speed), ()); - TEST_EQUAL(speed, SpeedInUnits(75, Units::Imperial), ()); - - TEST(ParseMaxspeedTag("75 mph", speed), ()); - TEST_EQUAL(speed, SpeedInUnits(75, Units::Imperial), ()); - - TEST(ParseMaxspeedTag("75 mph and some other string", speed), ()); - TEST_EQUAL(speed, SpeedInUnits(75, Units::Imperial), ()); - - TEST(ParseMaxspeedTag("75mph", speed), ()); - TEST_EQUAL(speed, SpeedInUnits(75, Units::Imperial), ()); - - TEST(ParseMaxspeedTag("75 mph", speed), ()); - TEST_EQUAL(speed, SpeedInUnits(75, Units::Imperial), ()); - - TEST(ParseMaxspeedTag("75 mph", speed), ()); - TEST_EQUAL(speed, SpeedInUnits(75, Units::Imperial), ()); - - TEST(ParseMaxspeedTag("75 mph and some other string", speed), ()); - TEST_EQUAL(speed, SpeedInUnits(75, Units::Imperial), ()); - - TEST(!ParseMaxspeedTag("some other string", speed), ()); - TEST(!ParseMaxspeedTag("60 kmph", speed), ()); - TEST(!ParseMaxspeedTag("1234567890 kmh", speed), ()); - TEST(!ParseMaxspeedTag("1234567890 mph", speed), ()); - TEST(!ParseMaxspeedTag("1234567890", speed), ()); -} } // namespace diff --git a/generator/maxspeed_builder.hpp b/generator/maxspeed_builder.hpp index b18cc59405..634ab9c6ae 100644 --- a/generator/maxspeed_builder.hpp +++ b/generator/maxspeed_builder.hpp @@ -29,8 +29,9 @@ void BuildMaxspeedSection(std::string const & dataPath, /// \brief Builds maxspeeds section in mwm with |dataPath|. This section contains max speed limits /// if they are available in file |maxspeedFilename|. /// \param maxspeedFilename file name to csv file with maxspeed tag values. -/// \note To start building the section, the following data must be ready: -/// 1. GenerateIntermediateData(). Saves to a file data about maxspeed tags value of road features +/// \note To start building the section, the following steps should be done: +/// 1. Calls GenerateIntermediateData(). It stores data about maxspeed tags value of road features +// to a csv file. /// 2. Calls GenerateFeatures() /// 3. Generates geometry. void BuildMaxspeedSection(std::string const & dataPath, std::string const & osmToFeaturePath, diff --git a/generator/maxspeed_parser.cpp b/generator/maxspeed_parser.cpp index 8ef95eaa99..3cc6d627ae 100644 --- a/generator/maxspeed_parser.cpp +++ b/generator/maxspeed_parser.cpp @@ -219,7 +219,6 @@ string UnitsToString(Units units) Units StringToUnits(string const & units) { - if (units == "Metric") return Units::Metric; if (units == "Imperial") diff --git a/generator/maxspeed_parser.hpp b/generator/maxspeed_parser.hpp index b55a7988de..790f8d4590 100644 --- a/generator/maxspeed_parser.hpp +++ b/generator/maxspeed_parser.hpp @@ -10,14 +10,15 @@ namespace generator { /// \brief Obtains |speed| and |units| by road category based on -/// the table in https://wiki.openstreetmap.org/wiki/Speed_limits -/// This method should be updated regularly. Last update: 23.10.18. +/// the table at https://wiki.openstreetmap.org/wiki/Speed_limits +/// This function should be updated regularly. Last update: 23.10.18. /// \returns true if |speed| and |units| is found for |category| and these fields are filled /// and false otherwise. /// \note For example by passing string "RU:urban", you'll get /// speed = 60 and units = Units::Metric. /// \note If the method returns true the field |speed| may be filled with |kNoneMaxSpeed| value. -/// It means no speed limits for the |category|. It's currently actual for Germany. +/// It means no speed limits for the |category|. It is currently the case for some road categories +/// in Germany. bool RoadCategoryToSpeed(std::string const & category, routing::SpeedInUnits & speed); /// \brief Tries to convert the value of tag maxspeed to speed in appropriate units. diff --git a/platform/platform_tests/measurement_tests.cpp b/platform/platform_tests/measurement_tests.cpp index 0eb20b7997..ea7a1938be 100644 --- a/platform/platform_tests/measurement_tests.cpp +++ b/platform/platform_tests/measurement_tests.cpp @@ -196,7 +196,7 @@ UNIT_TEST(OSMDistanceToMetersString) UNIT_TEST(UnitsConversion) { - double const kEps = 0.00001; + double const kEps = 1e-5; TEST(base::AlmostEqualAbs(MilesToMeters(MetersToMiles(1000.0)), 1000.0, kEps), ()); TEST(base::AlmostEqualAbs(MilesToMeters(1.0), 1609.344, kEps), ()); diff --git a/routing/geometry.cpp b/routing/geometry.cpp index 8be0665c29..c53e3db179 100644 --- a/routing/geometry.cpp +++ b/routing/geometry.cpp @@ -207,7 +207,7 @@ RoadGeometry const & Geometry::GetRoad(uint32_t featureId) unique_ptr GeometryLoader::Create(DataSource const & dataSource, MwmSet::MwmHandle const & handle, shared_ptr vehicleModel, - AttrLoader attrLoader, + AttrLoader && attrLoader, bool loadAltitudes) { CHECK(handle.IsAlive(), ()); diff --git a/routing/geometry.hpp b/routing/geometry.hpp index 4569c519d4..64039a6aeb 100644 --- a/routing/geometry.hpp +++ b/routing/geometry.hpp @@ -96,7 +96,7 @@ public: static std::unique_ptr Create(DataSource const & dataSource, MwmSet::MwmHandle const & handle, std::shared_ptr vehicleModel, - AttrLoader attrLoader, + AttrLoader && attrLoader, bool loadAltitudes); /// This is for stand-alone work. diff --git a/routing/maxspeeds_serialization.hpp b/routing/maxspeeds_serialization.hpp index facca76d7c..62fb680862 100644 --- a/routing/maxspeeds_serialization.hpp +++ b/routing/maxspeeds_serialization.hpp @@ -7,6 +7,7 @@ #include "coding/reader.hpp" #include "coding/simple_dense_coding.hpp" #include "coding/succinct_mapper.hpp" +#include "coding/varint.hpp" #include "coding/write_to_sink.hpp" #include "coding/writer.hpp" @@ -34,6 +35,11 @@ void GetForwardMaxspeedStats(std::vector const & speeds, /// * elias_fano with feature ids which have maxspeed tag in forward direction only /// * SimpleDenseCoding with code of maxspeed /// * table with vector with feature ids and maxspeeds which have maxspeed for both directions. +/// The table is stored in the following format: +/// +/// +/// ... +/// The number of such features is stored in the header. class MaxspeedsSerializer { public: @@ -109,14 +115,14 @@ public: ++header.m_bidirectionalMaxspeedNumber; // Note. |speeds| sorted by feature ids and they are unique. So the first item in |speed| - // has feature id 0 or greater. And the second item in |speed| has feature id grater than 0. + // has feature id 0 or greater. And the second item in |speed| has feature id greater than 0. // This guarantees that for only first saved |s| the if branch below is called. if (prevFeatureId != 0) CHECK_GREATER(s.GetFeatureId(), prevFeatureId, ()); uint32_t delta = (prevFeatureId == 0 ? s.GetFeatureId() : s.GetFeatureId() - prevFeatureId); prevFeatureId = s.GetFeatureId(); - WriteToSink(sink, delta); + WriteVarUint(sink, delta); WriteToSink(sink, static_cast(forwardMacro)); WriteToSink(sink, static_cast(backwardMacro)); } @@ -170,10 +176,10 @@ public: } maxspeeds.m_bidirectionalMaxspeeds.reserve(header.m_bidirectionalMaxspeedNumber); - uint32_t prevFeatureId = 0; + uint32_t featureId = 0; for (size_t i = 0; i < header.m_bidirectionalMaxspeedNumber; ++i) { - auto const delta = ReadPrimitiveFromSource(src); + auto const delta = ReadVarUint(src); auto const forward = ReadPrimitiveFromSource(src); auto const backward = ReadPrimitiveFromSource(src); CHECK(GetMaxspeedConverter().IsValidMacro(forward), (i)); @@ -193,9 +199,8 @@ public: // Note. If neither |forwardSpeed| nor |backwardSpeed| are numeric it means // both of them have value "walk" or "none". So the units are not relevant for this case. - prevFeatureId += delta; - uint32_t fid = (i == 0 ? delta : prevFeatureId); - maxspeeds.m_bidirectionalMaxspeeds.emplace_back(fid, units, forwardSpeed.GetSpeed(), + featureId += delta; + maxspeeds.m_bidirectionalMaxspeeds.emplace_back(featureId, units, forwardSpeed.GetSpeed(), backwardSpeed.GetSpeed()); } }