diff --git a/generator/routing_index_generator.cpp b/generator/routing_index_generator.cpp index ba57a1d1b6..3455ff7806 100644 --- a/generator/routing_index_generator.cpp +++ b/generator/routing_index_generator.cpp @@ -173,21 +173,24 @@ bool RegionsContain(vector const & regions, m2::PointD const & poin return false; } -double CalcClockwiseDistance(vector const & borders, - CrossMwmConnectorSerializer::Transition const & transition) +// Calculate distance from the starting border point to the transition along the border. +// It could be measured clockwise or counterclockwise, direction doesn't matter. +double CalcDistanceAlongTheBorders(vector const & borders, + CrossMwmConnectorSerializer::Transition const & transition) { double distance = 0.0; for (m2::RegionD const & region : borders) { - vector points = region.Data(); - + vector const & points = region.Data(); + CHECK(!points.empty(), ()); m2::PointD const * prev = &points.back(); + for (m2::PointD const & curr : points) { m2::PointD intersection; - if (region.IsIntersect(transition.GetBackPoint(), transition.GetFrontPoint(), *prev, curr, - intersection)) + if (m2::RegionD::IsIntersect(transition.GetBackPoint(), transition.GetFrontPoint(), *prev, + curr, intersection)) { distance += prev->Length(intersection); return distance; @@ -249,17 +252,18 @@ void CalcCrossMwmTransitions(string const & path, string const & mwmFile, string sort(transitions.begin(), transitions.end(), [&](CrossMwmConnectorSerializer::Transition const & lhs, CrossMwmConnectorSerializer::Transition const & rhs) { - return CalcClockwiseDistance(borders, lhs) < CalcClockwiseDistance(borders, rhs); + return CalcDistanceAlongTheBorders(borders, lhs) < + CalcDistanceAlongTheBorders(borders, rhs); }); LOG(LINFO, ("Transition sorted in", timer.ElapsedSeconds(), "seconds")); for (auto const & transition : transitions) { - for (size_t j = 0; j < connectors.size(); ++j) + for (size_t i = 0; i < connectors.size(); ++i) { - VehicleMask const mask = GetVehicleMask(static_cast(j)); - CrossMwmConnectorSerializer::AddTransition(transition, mask, connectors[j]); + VehicleMask const mask = GetVehicleMask(static_cast(i)); + CrossMwmConnectorSerializer::AddTransition(transition, mask, connectors[i]); } } } diff --git a/generator/routing_index_generator.hpp b/generator/routing_index_generator.hpp index 6d70078618..d693a25678 100644 --- a/generator/routing_index_generator.hpp +++ b/generator/routing_index_generator.hpp @@ -1,9 +1,10 @@ #pragma once -#include "std/string.hpp" +#include namespace routing { -bool BuildRoutingIndex(string const & filename, string const & country); -void BuildCrossMwmSection(string const & path, string const & mwmFile, string const & country); +bool BuildRoutingIndex(std::string const & filename, std::string const & country); +void BuildCrossMwmSection(std::string const & path, std::string const & mwmFile, + std::string const & country); } // namespace routing diff --git a/geometry/region2d.hpp b/geometry/region2d.hpp index 934a45c945..b13832651f 100644 --- a/geometry/region2d.hpp +++ b/geometry/region2d.hpp @@ -154,12 +154,12 @@ namespace m2 std::swap(m_rect, rhs.m_rect); } - ContainerT Data() const { return m_points; } + ContainerT const & Data() const { return m_points; } template - inline bool IsIntersect(CoordT const & x11, CoordT const & y11, CoordT const & x12, CoordT const & y12, + static inline bool IsIntersect(CoordT const & x11, CoordT const & y11, CoordT const & x12, CoordT const & y12, CoordT const & x21, CoordT const & y21, CoordT const & x22, CoordT const & y22, - TEqualF equalF, PointT & pt) const + TEqualF equalF, PointT & pt) { double const divider = ((y12 - y11) * (x22 - x21) - (x12 - x11) * (y22-y21)); if (equalF.EqualZeroSquarePrecision(divider)) @@ -180,7 +180,7 @@ namespace m2 return true; } - inline bool IsIntersect(PointT const & p1, PointT const & p2, PointT const & p3, PointT const & p4 , PointT & pt) const + static inline bool IsIntersect(PointT const & p1, PointT const & p2, PointT const & p3, PointT const & p4 , PointT & pt) { return IsIntersect(p1.x, p1.y, p2.x, p2.y, p3.x, p3.y, p4.x, p4.y, typename TraitsT::EqualType(), pt); } diff --git a/routing/CMakeLists.txt b/routing/CMakeLists.txt index 1e07dd43a0..e404c93e93 100644 --- a/routing/CMakeLists.txt +++ b/routing/CMakeLists.txt @@ -18,6 +18,7 @@ set( bicycle_directions.hpp car_router.cpp car_router.hpp + coding.hpp cross_mwm_connector.cpp cross_mwm_connector.hpp cross_mwm_connector_serialization.cpp diff --git a/routing/coding.hpp b/routing/coding.hpp new file mode 100644 index 0000000000..7136333675 --- /dev/null +++ b/routing/coding.hpp @@ -0,0 +1,118 @@ +#pragma once + +#include "routing/routing_exceptions.hpp" + +#include "coding/bit_streams.hpp" +#include "coding/elias_coder.hpp" + +#include "base/assert.hpp" +#include "base/bits.hpp" +#include "base/checked_cast.hpp" +#include "base/macros.hpp" + +#include +#include + +namespace routing +{ +template +T ReadDelta(BitReader & reader) +{ + static_assert(std::is_unsigned::value, "T should be an unsigned type"); + uint64_t const decoded = coding::DeltaCoder::Decode(reader); + if (decoded > std::numeric_limits::max()) + { + MYTHROW(CorruptedDataException, + ("Decoded value", decoded, "out of limit", std::numeric_limits::max())); + } + + return static_cast(decoded); +} + +template +void WriteDelta(BitWriter & writer, T value) +{ + static_assert(std::is_unsigned::value, "T should be an unsigned type"); + ASSERT_GREATER(value, 0, ()); + + bool const success = coding::DeltaCoder::Encode(writer, static_cast(value)); + ASSERT(success, ()); + UNUSED_VALUE(success); +} + +template +T ReadGamma(BitReader & reader) +{ + static_assert(std::is_unsigned::value, "T should be an unsigned type"); + uint64_t const decoded = coding::GammaCoder::Decode(reader); + if (decoded > std::numeric_limits::max()) + { + MYTHROW(CorruptedDataException, + ("Decoded value", decoded, "out of limit", std::numeric_limits::max())); + } + + return static_cast(decoded); +} + +template +void WriteGamma(BitWriter & writer, T value) +{ + static_assert(std::is_unsigned::value, "T should be an unsigned type"); + ASSERT_GREATER(value, 0, ()); + + bool const success = coding::GammaCoder::Encode(writer, static_cast(value)); + ASSERT(success, ()); + UNUSED_VALUE(success); +} + +// C++ standart says: +// if the value can't be represented in the destination unsigned type, +// the result is implementation-defined. +// +// ModularCast makes unambiguous conversion from unsigned value to signed. +// The resulting value is the least signed integer congruent to the source integer +// (modulo 2^n where n is the number of bits used to represent the unsigned type) +template ::type> +Signed ModularCast(Unsigned value) +{ + static_assert(std::is_unsigned::value, "T should be an unsigned type"); + + if (value <= static_cast(std::numeric_limits::max())) + return static_cast(value); + + auto constexpr minSignedT = std::numeric_limits::min(); + return static_cast(value - static_cast(minSignedT)) + minSignedT; +} + +// Encodes current as delta compared with prev. +template ::type> +UnsignedT EncodeZigZagDelta(T prev, T current) +{ + static_assert(std::is_integral::value, "T should be an integral type"); + using SignedT = typename std::make_signed::type; + + auto const unsignedPrev = static_cast(prev); + auto const unsignedCurrent = static_cast(current); + auto originalDelta = ModularCast(static_cast(unsignedCurrent - unsignedPrev)); + static_assert(std::is_same::value, + "It's expected that ModuleCast returns SignedT"); + + auto encodedDelta = bits::ZigZagEncode(originalDelta); + static_assert(std::is_same::value, + "It's expected that bits::ZigZagEncode returns UnsignedT"); + return encodedDelta; +} + +// Reverse function for EncodeZigZagDelta. +template ::type> +T DecodeZigZagDelta(T prev, UnsignedT delta) +{ + static_assert(std::is_integral::value, "T should be an integral type"); + using SignedT = typename std::make_signed::type; + + auto decoded = bits::ZigZagDecode(delta); + static_assert(std::is_same::value, + "It's expected that bits::ZigZagDecode returns SignedT"); + return prev + static_cast(decoded); +} +} // namespace routing diff --git a/routing/cross_mwm_connector.hpp b/routing/cross_mwm_connector.hpp index ab61296d7f..c1b611ef0c 100644 --- a/routing/cross_mwm_connector.hpp +++ b/routing/cross_mwm_connector.hpp @@ -147,7 +147,7 @@ private: std::unordered_map m_transitions; WeightsLoadState m_weightsLoadState = WeightsLoadState::Unknown; uint64_t m_weightsOffset = 0; - Weight m_accuracy = 0; + Weight m_granularity = 0; std::vector m_weights; }; } // namespace routing diff --git a/routing/cross_mwm_connector_serialization.cpp b/routing/cross_mwm_connector_serialization.cpp index 1e951417da..b9a05c517c 100644 --- a/routing/cross_mwm_connector_serialization.cpp +++ b/routing/cross_mwm_connector_serialization.cpp @@ -35,8 +35,8 @@ void CrossMwmConnectorSerializer::WriteWeights(vector const & weights, } writer.Write(kRouteBit, 1); - auto const storedWeight = (weight + kAccuracy - 1) / kAccuracy; - WriteDelta(writer, EncodePositivesDelta(prevWeight, storedWeight)); + auto const storedWeight = (weight + kGranularity - 1) / kGranularity; + WriteDelta(writer, EncodeZigZagDelta(prevWeight, storedWeight) + 1); prevWeight = storedWeight; } } diff --git a/routing/cross_mwm_connector_serialization.hpp b/routing/cross_mwm_connector_serialization.hpp index 1b82dcbafc..cb6a423d9b 100644 --- a/routing/cross_mwm_connector_serialization.hpp +++ b/routing/cross_mwm_connector_serialization.hpp @@ -1,5 +1,6 @@ #pragma once +#include "routing/coding.hpp" #include "routing/cross_mwm_connector.hpp" #include "routing/routing_exceptions.hpp" #include "routing/vehicle_mask.hpp" @@ -8,7 +9,6 @@ #include "indexer/geometry_serialization.hpp" #include "coding/bit_streams.hpp" -#include "coding/elias_coder.hpp" #include "coding/reader.hpp" #include "coding/write_to_sink.hpp" #include "coding/writer.hpp" @@ -17,8 +17,6 @@ #include #include -#include -#include #include namespace routing @@ -181,7 +179,7 @@ public: } connector.m_weightsOffset = weightsOffset; - connector.m_accuracy = header.GetAccuracy(); + connector.m_granularity = header.GetGranularity(); connector.m_weightsLoadState = WeightsLoadState::ReadyToLoad; return; } @@ -194,7 +192,7 @@ public: Source & src) { CHECK(connector.m_weightsLoadState == WeightsLoadState::ReadyToLoad, ()); - CHECK_GREATER(connector.m_accuracy, 0, ()); + CHECK_GREATER(connector.m_granularity, 0, ()); src.Skip(connector.m_weightsOffset); @@ -212,9 +210,9 @@ public: continue; } - Weight const delta = ReadDelta(reader); - Weight const current = DecodePositivesDelta(prev, delta); - connector.m_weights.push_back(current * connector.m_accuracy); + Weight const delta = ReadDelta(reader) - 1; + Weight const current = DecodeZigZagDelta(prev, delta); + connector.m_weights.push_back(current * connector.m_granularity); prev = current; } @@ -240,11 +238,12 @@ private: static uint32_t constexpr kLastVersion = 0; static uint8_t constexpr kNoRouteBit = 0; static uint8_t constexpr kRouteBit = 1; - // Accuracy is integral unit for storing Weight in section. - // Accuracy measured in seconds as well as the Weight. - // Accuracy is the way to tune section size. - static Weight constexpr kAccuracy = 4; - static_assert(kAccuracy > 0, "kAccuracy should be > 0"); + + // Weight is rounded up to the next multiple of kGranularity before being stored in the section. + // kGranularity is measured in seconds as well as Weight. + // Increasing kGranularity will decrease the section's size. + static Weight constexpr kGranularity = 4; + static_assert(kGranularity > 0, "kGranularity should be > 0"); class Section final { @@ -306,7 +305,7 @@ private: WriteToSink(sink, m_version); WriteToSink(sink, m_numTransitions); WriteToSink(sink, m_sizeTransitions); - WriteToSink(sink, m_accuracy); + WriteToSink(sink, m_granularity); m_codingParams.Save(sink); WriteToSink(sink, m_bitsPerMask); @@ -327,7 +326,7 @@ private: m_numTransitions = ReadPrimitiveFromSource(src); m_sizeTransitions = ReadPrimitiveFromSource(src); - m_accuracy = ReadPrimitiveFromSource(src); + m_granularity = ReadPrimitiveFromSource(src); m_codingParams.Load(src); m_bitsPerMask = ReadPrimitiveFromSource(src); @@ -341,7 +340,7 @@ private: uint32_t GetNumTransitions() const { return m_numTransitions; } uint64_t GetSizeTransitions() const { return m_sizeTransitions; } - Weight GetAccuracy() const { return m_accuracy; } + Weight GetGranularity() const { return m_granularity; } serial::CodingParams const & GetCodingParams() const { return m_codingParams; } uint8_t GetBitsPerMask() const { return m_bitsPerMask; } std::vector
const & GetSections() const { return m_sections; } @@ -350,74 +349,12 @@ private: uint32_t m_version = kLastVersion; uint32_t m_numTransitions = 0; uint64_t m_sizeTransitions = 0; - Weight m_accuracy = kAccuracy; + Weight m_granularity = kGranularity; serial::CodingParams m_codingParams; uint8_t m_bitsPerMask = 0; std::vector
m_sections; }; - template - static void WriteDelta(BitWriter & writer, T value) - { - static_assert(std::is_integral::value, "T should be integral type"); - ASSERT_GREATER(value, 0, ()); - - bool const success = coding::DeltaCoder::Encode(writer, static_cast(value)); - ASSERT(success, ()); - UNUSED_VALUE(success); - } - - template - static T ReadDelta(BitReader & reader) - { - static_assert(std::is_integral::value, "T should be integral type"); - uint64_t const decoded = coding::DeltaCoder::Decode(reader); - if (decoded > numeric_limits::max()) - MYTHROW(CorruptedDataException, - ("Decoded value", decoded, "out of limit", numeric_limits::max())); - - return static_cast(decoded); - } - - // Encodes current as delta compared with prev. - // - // Example: - // prev = 3 - // current = 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 - // delta = 5, 3, 1, 2, 4, 6, 7, 8, 9, 10 - template - static T EncodePositivesDelta(T prev, T current) - { - static_assert(std::is_integral::value, "T should be integral type"); - ASSERT_GREATER(prev, 0, ()); - ASSERT_GREATER(current, 0, ()); - - if (current <= prev) - return (prev - current) * 2 + 1; - - if (current < prev * 2) - return (current - prev) * 2; - - return current; - } - - // Reverse function for EncodePositivesDelta. - template - static T DecodePositivesDelta(T prev, T delta) - { - static_assert(std::is_integral::value, "T should be integral type"); - ASSERT_GREATER(prev, 0, ()); - ASSERT_GREATER(delta, 0, ()); - - if (delta >= prev * 2) - return delta; - - if (delta % 2 == 0) - return prev + delta / 2; - - return prev - delta / 2; - } - template static T GetBitsPerMask() { diff --git a/routing/index_graph_serialization.hpp b/routing/index_graph_serialization.hpp index a11b358b25..eb811e84be 100644 --- a/routing/index_graph_serialization.hpp +++ b/routing/index_graph_serialization.hpp @@ -1,12 +1,12 @@ #pragma once +#include "routing/coding.hpp" #include "routing/index_graph.hpp" #include "routing/joint.hpp" #include "routing/routing_exceptions.hpp" #include "routing/vehicle_mask.hpp" #include "coding/bit_streams.hpp" -#include "coding/elias_coder.hpp" #include "coding/reader.hpp" #include "coding/write_to_sink.hpp" @@ -81,17 +81,17 @@ public: uint32_t featureId = -1; for (uint32_t i = 0; i < section.GetNumRoads(); ++i) { - uint32_t const featureDelta = ReadGamma(reader); + uint32_t const featureDelta = ReadGamma(reader); featureId += featureDelta; - uint32_t const jointsNumber = ConvertJointsNumber(ReadGamma(reader)); + uint32_t const jointsNumber = ConvertJointsNumber(ReadGamma(reader)); // See comment above about -1. uint32_t pointId = -1; for (uint32_t j = 0; j < jointsNumber; ++j) { - uint32_t const pointDelta = ReadGamma(reader); + auto const pointDelta = ReadGamma(reader); pointId += pointDelta; Joint::Id const jointId = jointIdDecoder.Read(reader); if (jointId >= section.GetEndJointId()) @@ -283,7 +283,7 @@ private: uint8_t const bit = reader.Read(1); if (bit == kRepeatJointIdBit) { - uint32_t const delta = ReadDelta(reader); + auto const delta = ReadDelta(reader); if (delta > m_count) MYTHROW(CorruptedDataException, ("Joint id delta", delta, "> count =", m_count)); @@ -367,46 +367,6 @@ private: vector m_buffer; }; - template - static void WriteGamma(BitWriter & writer, uint32_t value) - { - ASSERT_NOT_EQUAL(value, 0, ()); - - bool const success = coding::GammaCoder::Encode(writer, static_cast(value)); - ASSERT(success, ()); - UNUSED_VALUE(success); - } - - template - static uint32_t ReadGamma(BitReader & reader) - { - uint64_t const decoded = coding::GammaCoder::Decode(reader); - if (decoded > numeric_limits::max()) - MYTHROW(CorruptedDataException, ("Decoded uint32_t out of limit", decoded)); - - return static_cast(decoded); - } - - template - static void WriteDelta(BitWriter & writer, uint32_t value) - { - ASSERT_NOT_EQUAL(value, 0, ()); - - bool const success = coding::DeltaCoder::Encode(writer, static_cast(value)); - ASSERT(success, ()); - UNUSED_VALUE(success); - } - - template - static uint32_t ReadDelta(BitReader & reader) - { - uint64_t const decoded = coding::DeltaCoder::Decode(reader); - if (decoded > numeric_limits::max()) - MYTHROW(CorruptedDataException, ("Decoded uint32_t out of limit", decoded)); - - return static_cast(decoded); - } - static VehicleMask GetRoadMask(unordered_map const & masks, uint32_t featureId); static uint32_t ConvertJointsNumber(uint32_t jointsNumber); diff --git a/routing/routing.pro b/routing/routing.pro index ed0992c853..f63b74e759 100644 --- a/routing/routing.pro +++ b/routing/routing.pro @@ -71,6 +71,7 @@ HEADERS += \ base/followed_polyline.hpp \ bicycle_directions.hpp \ car_router.hpp \ + coding.hpp \ cross_mwm_connector.hpp \ cross_mwm_connector_serialization.hpp \ cross_mwm_index_graph.hpp \ diff --git a/routing/routing_tests/CMakeLists.txt b/routing/routing_tests/CMakeLists.txt index 349191b673..9844760b4f 100644 --- a/routing/routing_tests/CMakeLists.txt +++ b/routing/routing_tests/CMakeLists.txt @@ -8,6 +8,7 @@ set( astar_progress_test.cpp astar_router_test.cpp async_router_test.cpp + coding_test.cpp cross_mwm_connector_test.cpp cross_routing_tests.cpp cumulative_restriction_test.cpp diff --git a/routing/routing_tests/coding_test.cpp b/routing/routing_tests/coding_test.cpp new file mode 100644 index 0000000000..5ba6b5ca4a --- /dev/null +++ b/routing/routing_tests/coding_test.cpp @@ -0,0 +1,93 @@ +#include "testing/testing.hpp" + +#include "routing/coding.hpp" + +#include +#include + +using namespace routing; + +namespace +{ +template +void ForEachNumber(ToDo && toDo) +{ + for (T number = numeric_limits::min();; ++number) + { + toDo(number); + + if (number == numeric_limits::max()) + break; + } +} + +template +void TestZigZag(T prev, T current) +{ + auto const delta = EncodeZigZagDelta(prev, current); + auto const decodedCurrent = DecodeZigZagDelta(prev, delta); + TEST_EQUAL(decodedCurrent, current, ("prev:", prev, "delta:", delta)); +} + +template +void TestZigZagUnsigned() +{ + static_assert(std::is_unsigned::value, "T should be an unsigned type"); + + constexpr auto max = std::numeric_limits::max(); + constexpr T values[] = {0, 1, 7, max / 2, max - 1, max}; + for (T prev : values) + { + for (T current : values) + TestZigZag(prev, current); + } +} + +template +void TestZigZagSigned() +{ + static_assert(std::is_signed::value, "T should be a signed type"); + + constexpr auto min = std::numeric_limits::min(); + constexpr auto max = std::numeric_limits::max(); + constexpr T values[] = {min, min + 1, min / 2, -7, -1, 0, 1, 7, max / 2, max - 1, max}; + for (T prev : values) + { + for (T current : values) + TestZigZag(prev, current); + } +} +} // namespace + +namespace routing_test +{ +UNIT_TEST(ModuleCastTest) +{ + ForEachNumber([](uint8_t number) { + auto signedNumber = ModularCast(number); + static_assert(std::is_same::value, "int8_t expected"); + TEST_EQUAL(static_cast(signedNumber), number, ("signedNumber:", signedNumber)); + }); +} + +UNIT_TEST(ZigZagUint8) +{ + ForEachNumber([](uint8_t prev) { + ForEachNumber([&](uint8_t current) { TestZigZag(prev, current); }); + }); +} + +UNIT_TEST(ZigZagInt8) +{ + ForEachNumber([](int8_t prev) { + ForEachNumber([&](int8_t current) { TestZigZag(prev, current); }); + }); +} + +UNIT_TEST(ZigZagUint16) { TestZigZagUnsigned(); } +UNIT_TEST(ZigZagInt16) { TestZigZagSigned(); } +UNIT_TEST(ZigZagUint32) { TestZigZagUnsigned(); } +UNIT_TEST(ZigZagInt32) { TestZigZagSigned(); } +UNIT_TEST(ZigZagUint64) { TestZigZagUnsigned(); } +UNIT_TEST(ZigZagInt64) { TestZigZagSigned(); } +} // namespace routing_test diff --git a/routing/routing_tests/cross_mwm_connector_test.cpp b/routing/routing_tests/cross_mwm_connector_test.cpp index c2930ad392..527cbac658 100644 --- a/routing/routing_tests/cross_mwm_connector_test.cpp +++ b/routing/routing_tests/cross_mwm_connector_test.cpp @@ -240,7 +240,7 @@ UNIT_TEST(Serialization) UNIT_TEST(WeightsSerialization) { size_t constexpr kNumTransitions = 3; - std::vector const weights = { + vector const weights = { 4.0, 20.0, CrossMwmConnector::kNoRoute, 12.0, CrossMwmConnector::kNoRoute, 40.0, 48.0, 24.0, 12.0}; TEST_EQUAL(weights.size(), kNumTransitions * kNumTransitions, ()); @@ -301,7 +301,10 @@ UNIT_TEST(WeightsSerialization) { auto const weight = weights[weightIdx]; if (weight != CrossMwmConnector::kNoRoute) - expectedEdges.emplace_back(Segment(mwmId, exitId, 1, false /* forward */), weight); + { + expectedEdges.emplace_back(Segment(mwmId, exitId, 1 /* segmentIdx */, false /* forward */), + weight); + } ++weightIdx; } diff --git a/routing/routing_tests/routing_tests.pro b/routing/routing_tests/routing_tests.pro index 80f07ed450..0ecdd52344 100644 --- a/routing/routing_tests/routing_tests.pro +++ b/routing/routing_tests/routing_tests.pro @@ -26,6 +26,7 @@ SOURCES += \ astar_progress_test.cpp \ astar_router_test.cpp \ async_router_test.cpp \ + coding_test.cpp \ cross_mwm_connector_test.cpp \ cross_routing_tests.cpp \ cumulative_restriction_test.cpp \ diff --git a/xcode/routing/routing.xcodeproj/project.pbxproj b/xcode/routing/routing.xcodeproj/project.pbxproj index e7d736ca87..9e2ba9fe5c 100644 --- a/xcode/routing/routing.xcodeproj/project.pbxproj +++ b/xcode/routing/routing.xcodeproj/project.pbxproj @@ -44,7 +44,9 @@ 0C5FEC6A1DDE193F0017688C /* road_index.hpp in Headers */ = {isa = PBXBuildFile; fileRef = 0C5FEC671DDE193F0017688C /* road_index.hpp */; }; 0C5FEC6B1DDE193F0017688C /* road_point.hpp in Headers */ = {isa = PBXBuildFile; fileRef = 0C5FEC681DDE193F0017688C /* road_point.hpp */; }; 0C5FEC6D1DDE19A40017688C /* index_graph_test.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 0C5FEC6C1DDE19A40017688C /* index_graph_test.cpp */; }; + 0C62BFE61E8ADC3100055A79 /* coding.hpp in Headers */ = {isa = PBXBuildFile; fileRef = 0C62BFE51E8ADC3100055A79 /* coding.hpp */; }; 0C8705051E0182F200BCAF71 /* route_point.hpp in Headers */ = {isa = PBXBuildFile; fileRef = 0C8705041E0182F200BCAF71 /* route_point.hpp */; }; + 0CF5E8AA1E8EA7A1001ED497 /* coding_test.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 0CF5E8A91E8EA7A1001ED497 /* coding_test.cpp */; }; 3462FDAD1DC1E5BF00906FD7 /* libopening_hours.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 3462FDAC1DC1E5BF00906FD7 /* libopening_hours.a */; }; 349D1CE01E3F589900A878FD /* restrictions_serialization.cpp in Sources */ = {isa = PBXBuildFile; fileRef = 349D1CDE1E3F589900A878FD /* restrictions_serialization.cpp */; }; 349D1CE11E3F589900A878FD /* restrictions_serialization.hpp in Headers */ = {isa = PBXBuildFile; fileRef = 349D1CDF1E3F589900A878FD /* restrictions_serialization.hpp */; }; @@ -295,7 +297,9 @@ 0C5FEC671DDE193F0017688C /* road_index.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = road_index.hpp; sourceTree = ""; }; 0C5FEC681DDE193F0017688C /* road_point.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = road_point.hpp; sourceTree = ""; }; 0C5FEC6C1DDE19A40017688C /* index_graph_test.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = index_graph_test.cpp; sourceTree = ""; }; + 0C62BFE51E8ADC3100055A79 /* coding.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = coding.hpp; sourceTree = ""; }; 0C8705041E0182F200BCAF71 /* route_point.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = route_point.hpp; sourceTree = ""; }; + 0CF5E8A91E8EA7A1001ED497 /* coding_test.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = coding_test.cpp; sourceTree = ""; }; 3462FDAC1DC1E5BF00906FD7 /* libopening_hours.a */ = {isa = PBXFileReference; lastKnownFileType = archive.ar; name = libopening_hours.a; path = "../../../omim-build/xcode/Debug/libopening_hours.a"; sourceTree = ""; }; 349D1CDE1E3F589900A878FD /* restrictions_serialization.cpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.cpp; path = restrictions_serialization.cpp; sourceTree = ""; }; 349D1CDF1E3F589900A878FD /* restrictions_serialization.hpp */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.h; path = restrictions_serialization.hpp; sourceTree = ""; }; @@ -589,6 +593,7 @@ 6742ACA71C68A0B1009CB89E /* astar_progress_test.cpp */, 6742ACA81C68A0B1009CB89E /* astar_router_test.cpp */, 6742ACA91C68A0B1009CB89E /* async_router_test.cpp */, + 0CF5E8A91E8EA7A1001ED497 /* coding_test.cpp */, 0C5F5D241E798B3800307B98 /* cross_mwm_connector_test.cpp */, 6742ACAA1C68A0B1009CB89E /* cross_routing_tests.cpp */, 6742ACAB1C68A0B1009CB89E /* followed_polyline_test.cpp */, @@ -715,6 +720,7 @@ 56099E311CC9247E00A7772A /* bicycle_directions.hpp */, 56826BCE1DB51C4E00807C62 /* car_router.cpp */, 56826BCF1DB51C4E00807C62 /* car_router.hpp */, + 0C62BFE51E8ADC3100055A79 /* coding.hpp */, 0C5F5D1E1E798B0400307B98 /* cross_mwm_connector.cpp */, 0C5F5D1F1E798B0400307B98 /* cross_mwm_connector.hpp */, 0C5F5D1C1E798B0400307B98 /* cross_mwm_connector_serialization.cpp */, @@ -898,6 +904,7 @@ 0C08AA351DF83223004195DD /* index_graph_serialization.hpp in Headers */, 670D049F1B0B4A970013A7AC /* nearest_edge_finder.hpp in Headers */, A120B34F1B4A7C0A002F3808 /* online_absent_fetcher.hpp in Headers */, + 0C62BFE61E8ADC3100055A79 /* coding.hpp in Headers */, 674F9BD51B0A580E00704FFA /* road_graph.hpp in Headers */, 56826BD11DB51C4E00807C62 /* car_router.hpp in Headers */, A120B3511B4A7C0A002F3808 /* routing_algorithm.hpp in Headers */, @@ -1140,6 +1147,7 @@ 6753441B1A3F644F00A0A8C3 /* route.cpp in Sources */, 674F9BCA1B0A580E00704FFA /* async_router.cpp in Sources */, 0C5F5D221E798B0400307B98 /* cross_mwm_connector.cpp in Sources */, + 0CF5E8AA1E8EA7A1001ED497 /* coding_test.cpp in Sources */, 675344191A3F644F00A0A8C3 /* osrm2feature_map.cpp in Sources */, 670D049E1B0B4A970013A7AC /* nearest_edge_finder.cpp in Sources */, 0C5F5D251E798B3800307B98 /* cross_mwm_connector_test.cpp in Sources */,