From 5643ba275b89239131430281366d1f429cb41456 Mon Sep 17 00:00:00 2001 From: vng Date: Sat, 15 Jan 2011 01:30:04 +0200 Subject: [PATCH] Move processing of inner points into ParseGeometry function. --- indexer/feature.cpp | 78 +++++++++++++++++++++++++++------------- indexer/feature.hpp | 29 +++------------ indexer/feature_impl.hpp | 19 ++++++++-- std/algorithm.hpp | 1 + 4 files changed, 76 insertions(+), 51 deletions(-) diff --git a/indexer/feature.cpp b/indexer/feature.cpp index 3b493b9d8b..2094966e60 100644 --- a/indexer/feature.cpp +++ b/indexer/feature.cpp @@ -10,6 +10,9 @@ #include "../coding/byte_stream.hpp" #include "../base/logging.hpp" +#include "../base/stl_add.hpp" + +#include "../std/algorithm.hpp" #include "../base/start_mem_debug.hpp" @@ -411,8 +414,7 @@ void FeatureBuilder2::Serialize(buffers_holder_t & data) else { // offsets was pushed from high scale index to low - std::reverse(data.m_ptsOffset.begin(), data.m_ptsOffset.end()); - + reverse(data.m_ptsOffset.begin(), data.m_ptsOffset.end()); WriteVarUintArray(data.m_ptsOffset, sink); } } @@ -424,8 +426,7 @@ void FeatureBuilder2::Serialize(buffers_holder_t & data) else { // offsets was pushed from high scale index to low - std::reverse(data.m_trgOffset.begin(), data.m_trgOffset.end()); - + reverse(data.m_trgOffset.begin(), data.m_trgOffset.end()); WriteVarUintArray(data.m_trgOffset, sink); } } @@ -579,7 +580,10 @@ namespace int FeatureType::GetScaleIndex(int scale) { - for (size_t i = 0; i < ARRAY_SIZE(feature::g_arrScales); ++i) + int const count = ARRAY_SIZE(feature::g_arrScales); + if (scale == -1) return count-1; + + for (size_t i = 0; i < count; ++i) if (scale <= feature::g_arrScales[i]) return i; return -1; @@ -755,7 +759,9 @@ void FeatureType::ParseHeader2() const m_ptsSimpMask += (mask << (i << 3)); } - ReadInnerPoints(src, m_Points, ptsCount); + m_InnerPoints.reserve(ptsCount); + src = ArrayByteSource(ReadVarInt64Array( + src.Ptr(), ptsCount, MakeBackInsertFunctor(m_InnerPoints))); } else ReadOffsets(src, ptsMask, m_ptsOffsets); @@ -780,19 +786,49 @@ uint32_t FeatureType::ParseGeometry(int scale) const ParseHeader2(); uint32_t sz = 0; - if (m_Points.empty() && Header() & HEADER_IS_LINE) + if (Header() & HEADER_IS_LINE) { - int const ind = GetScaleIndex(scale, m_ptsOffsets); - if (ind != -1) + if (m_InnerPoints.empty()) { - ReaderSource src( - m_cont->GetReader(feature::GetTagForIndex(GEOMETRY_FILE_TAG, ind))); - src.Skip(m_ptsOffsets[ind]); - feature::LoadPoints(m_Points, src); + // outer geometry + int const ind = GetScaleIndex(scale, m_ptsOffsets); + if (ind != -1) + { + ReaderSource src( + m_cont->GetReader(feature::GetTagForIndex(GEOMETRY_FILE_TAG, ind))); + src.Skip(m_ptsOffsets[ind]); + feature::LoadPoints(m_Points, src); - CalcRect(m_Points, m_LimitRect); - sz = static_cast(src.Pos() - m_ptsOffsets[ind]); + sz = static_cast(src.Pos() - m_ptsOffsets[ind]); + } } + else + { + // inner geometry + size_t const count = m_InnerPoints.size(); + ASSERT_GREATER ( count, 1, () ); + + m_Points.reserve(count); + + uint32_t const scaleIndex = GetScaleIndex(scale); + ASSERT_LESS ( scaleIndex, ARRAY_SIZE(feature::g_arrScales), () ); + + int64_t id = m_InnerPoints.front(); + m_Points.push_back(feature::pts::ToPoint(id)); + + for (size_t i = 1; i < count-1; ++i) + { + id += m_InnerPoints[i]; + // check for point visibility in needed scaleIndex + if (((m_ptsSimpMask >> (2*(i-1))) & 0x3) <= scaleIndex) + m_Points.push_back(feature::pts::ToPoint(id)); + } + + id += m_InnerPoints.back(); + m_Points.push_back(feature::pts::ToPoint(id)); + } + + CalcRect(m_Points, m_LimitRect); } m_bPointsParsed = true; @@ -838,16 +874,10 @@ void FeatureType::ReadOffsets(ArrayByteSource & src, uint8_t mask, offsets_t & o } } -void FeatureType::ReadInnerPoints(ArrayByteSource & src, vector & v, uint8_t count) const +void FeatureType::ReadInnerPoints(ArrayByteSource & src, vector & points, uint8_t count) const { - ASSERT_GREATER ( count, 0, () ); - v.reserve(count); - - int64_t id = 0; - for (uint8_t i = 0; i < count; ++i) - v.push_back(feature::pts::ToPoint(id += ReadVarInt(src))); - - CalcRect(v, m_LimitRect); + src = ArrayByteSource(feature::LoadPointsSimple(src.Ptr(), count, points)); + CalcRect(points, m_LimitRect); } void FeatureType::ParseAll(int scale) const diff --git a/indexer/feature.hpp b/indexer/feature.hpp index d62fdd9857..9c90faec78 100644 --- a/indexer/feature.hpp +++ b/indexer/feature.hpp @@ -357,30 +357,8 @@ public: } else { - if (m_ptsSimpMask != 0 && scale != -1) - { - // inner geometry - uint32_t const scaleIndex = GetScaleIndex(scale); - ASSERT_LESS(scaleIndex, 4, ()); - - f(CoordPointT(m_Points[0].x, m_Points[0].y)); - - size_t const last = m_Points.size()-1; - for (size_t i = 1; i < last; ++i) - { - // check for point visibility in needed scaleIndex - if (((m_ptsSimpMask >> (2*(i-1))) & 0x3) <= scaleIndex) - f(CoordPointT(m_Points[i].x, m_Points[i].y)); - } - - f(CoordPointT(m_Points[last].x, m_Points[last].y)); - } - else - { - // loaded outer geometry - for (size_t i = 0; i < m_Points.size(); ++i) - f(CoordPointT(m_Points[i].x, m_Points[i].y)); - } + for (size_t i = 0; i < m_Points.size(); ++i) + f(CoordPointT(m_Points[i].x, m_Points[i].y)); } } @@ -448,6 +426,7 @@ private: void ParseAll(int scale) const; + mutable vector m_InnerPoints; mutable vector m_Points, m_Triangles; FilesContainerR * m_cont; @@ -461,7 +440,7 @@ private: typedef array offsets_t; // should be synchronized with ARRAY_SIZE(g_arrScales) static void ReadOffsets(ArrayByteSource & src, uint8_t mask, offsets_t & offsets); - void ReadInnerPoints(ArrayByteSource & src, vector & v, uint8_t count) const; + void ReadInnerPoints(ArrayByteSource & src, vector & points, uint8_t count) const; static int GetScaleIndex(int scale); static int GetScaleIndex(int scale, offsets_t const & offset); diff --git a/indexer/feature_impl.hpp b/indexer/feature_impl.hpp index bcd74e761a..9db296639b 100644 --- a/indexer/feature_impl.hpp +++ b/indexer/feature_impl.hpp @@ -60,7 +60,7 @@ namespace feature points_emitter(vector & points, uint32_t count) : m_points(points), m_id(0) { - m_points.reserve(count / 2); + m_points.reserve(count); } void operator() (int64_t id) { @@ -68,6 +68,12 @@ namespace feature } }; + inline void const * ReadPointsSimple( void const * p, size_t count, + vector & points) + { + return ReadVarInt64Array(p, count, points_emitter(points, count)); + } + template void ReadPoints(vector & points, TSource & src) { @@ -76,7 +82,7 @@ namespace feature char * p = &buffer[0]; src.Read(p, count); - ReadVarInt64Array(p, p + count, points_emitter(points, count)); + ReadVarInt64Array(p, p + count, points_emitter(points, count / 2)); } } @@ -102,6 +108,15 @@ namespace feature detail::WriteCells(cells, sink); } + inline void const * LoadPointsSimple( void const * p, size_t count, + vector & points) + { + ASSERT_GREATER ( count, 1, () ); + void const * ret = detail::ReadPointsSimple(p, count, points); + ASSERT_GREATER ( points.size(), 1, () ); + return ret; + } + template void LoadPoints(vector & points, TSource & src) { diff --git a/std/algorithm.hpp b/std/algorithm.hpp index 762650cb8a..448eb2e49f 100644 --- a/std/algorithm.hpp +++ b/std/algorithm.hpp @@ -24,6 +24,7 @@ using std::set_union; using std::set_intersection; using std::set_difference; using std::set_symmetric_difference; +using std::reverse; #ifdef DEBUG_NEW #define new DEBUG_NEW