diff --git a/search/search.pro b/search/search.pro index 17f1f40fef..b24dcac6c4 100644 --- a/search/search.pro +++ b/search/search.pro @@ -47,6 +47,7 @@ HEADERS += \ v2/features_layer_matcher.hpp \ v2/features_layer_path_finder.hpp \ v2/geocoder.hpp \ + v2/geometry_cache.hpp \ v2/house_numbers_matcher.hpp \ v2/house_to_street_table.hpp \ v2/intersection_result.hpp \ @@ -90,6 +91,7 @@ SOURCES += \ v2/features_layer_matcher.cpp \ v2/features_layer_path_finder.cpp \ v2/geocoder.cpp \ + v2/geometry_cache.cpp \ v2/house_numbers_matcher.cpp \ v2/house_to_street_table.cpp \ v2/intersection_result.cpp \ diff --git a/search/v2/geocoder.cpp b/search/v2/geocoder.cpp index b66e8101e5..9b9c18cf44 100644 --- a/search/v2/geocoder.cpp +++ b/search/v2/geocoder.cpp @@ -64,10 +64,12 @@ size_t constexpr kMaxNumCountries = 5; // included into World map. size_t constexpr kMaxNumLocalities = kMaxNumCities + kMaxNumStates + kMaxNumCountries; -// List of countries we're supporting search by state. Elements of the +size_t constexpr kPivotRectsCacheSize = 10; +size_t constexpr kLocalityRectsCacheSize = 10; + +// list of countries we're supporting search by state. Elements of the // list should be valid prefixes of corresponding mwms names. string const kCountriesWithStates[] = {"US_", "Canada_"}; -double constexpr kComparePoints = MercatorBounds::GetCellID2PointAbsEpsilon(); strings::UniString const kUniSpace(strings::MakeUniString(" ")); @@ -418,6 +420,8 @@ Geocoder::Geocoder(Index & index, storage::CountryInfoGetter const & infoGetter) , m_infoGetter(infoGetter) , m_numTokens(0) , m_model(SearchModel::Instance()) + , m_pivotRectsCache(kPivotRectsCacheSize, static_cast(*this)) + , m_localityRectsCache(kLocalityRectsCacheSize, static_cast(*this)) , m_pivotFeatures(index) , m_streets(nullptr) , m_villages(nullptr) @@ -586,15 +590,12 @@ void Geocoder::GoImpl(vector> & infos, bool inViewport) m_matcher = it->second.get(); m_matcher->SetContext(m_context.get()); - unique_ptr viewportCBV; - if (inViewport) - { - viewportCBV = v2::RetrieveGeometryFeatures(*m_context, cancellable, m_params.m_pivot, - m_params.m_scale); - } - PrepareAddressFeatures(); + coding::CompressedBitVector const * viewportCBV = nullptr; + if (inViewport) + viewportCBV = RetrieveGeometryFeatures(*m_context, m_params.m_pivot, RECT_ID_PIVOT); + if (viewportCBV) { for (size_t i = 0; i < m_numTokens; ++i) @@ -634,12 +635,14 @@ void Geocoder::GoImpl(vector> & infos, bool inViewport) void Geocoder::ClearCaches() { - m_geometryFeatures.clear(); + m_pivotRectsCache.Clear(); + m_localityRectsCache.Clear(); + m_pivotFeatures.Clear(); + m_addressFeatures.clear(); m_matchersCache.clear(); m_streetsCache.clear(); m_villages.reset(); - m_pivotFeatures.ClearCaches(); } void Geocoder::PrepareRetrievalParams(size_t curToken, size_t endToken) @@ -746,9 +749,8 @@ void Geocoder::FillLocalitiesTable() feature::GetCenter(ft), ftypes::GetRadiusByPopulation(ft.GetPopulation())); #if defined(DEBUG) - string name; - ft.GetName(StringUtf8Multilang::kDefaultCode, name); - LOG(LDEBUG, ("City =", name)); + ft.GetName(StringUtf8Multilang::kDefaultCode, city.m_defaultName); + LOG(LDEBUG, ("City =", city.m_defaultName, ftypes::GetRadiusByPopulation(ft.GetPopulation()))); #endif m_cities[{l.m_startToken, l.m_endToken}].push_back(city); @@ -829,9 +831,8 @@ void Geocoder::FillVillageLocalities() feature::GetCenter(ft), ftypes::GetRadiusByPopulation(ft.GetPopulation())); #if defined(DEBUG) - string name; - ft.GetName(StringUtf8Multilang::kDefaultCode, name); - LOG(LDEBUG, ("Village =", name)); + ft.GetName(StringUtf8Multilang::kDefaultCode, village.m_defaultName); + LOG(LDEBUG, ("Village =", village.m_defaultName)); #endif m_cities[{l.m_startToken, l.m_endToken}].push_back(village); @@ -972,7 +973,8 @@ void Geocoder::MatchCities() if (m_context->GetInfo()->GetType() == MwmInfo::WORLD) continue; - auto const * cityFeatures = RetrieveGeometryFeatures(*m_context, city.m_rect, city.m_featureId); + auto const * cityFeatures = + RetrieveGeometryFeatures(*m_context, city.m_rect, RECT_ID_LOCALITY); if (coding::CompressedBitVector::IsEmpty(cityFeatures)) continue; @@ -985,7 +987,7 @@ void Geocoder::MatchCities() void Geocoder::MatchAroundPivot() { - auto const * features = RetrieveGeometryFeatures(*m_context, m_params.m_pivot, PIVOT_ID); + auto const * features = RetrieveGeometryFeatures(*m_context, m_params.m_pivot, RECT_ID_PIVOT); if (!features) return; @@ -1427,24 +1429,14 @@ unique_ptr Geocoder::LoadVillages(MwmContext & cont coding::CompressedBitVector const * Geocoder::RetrieveGeometryFeatures(MwmContext const & context, m2::RectD const & rect, - int id) + RectId id) { - /// @todo - /// - Implement more smart strategy according to id. - /// - Move all rect limits here - - auto & features = m_geometryFeatures[context.GetId()]; - for (auto const & v : features) + switch (id) { - if (v.m_rect.IsRectInside(rect)) - return v.m_cbv.get(); + case RECT_ID_PIVOT: return m_pivotRectsCache.Get(context, rect, m_params.m_scale); + case RECT_ID_LOCALITY: return m_localityRectsCache.Get(context, rect, m_params.m_scale); + case RECT_ID_COUNT: ASSERT(false, ("Invalid RectId.")); return nullptr; } - - auto cbv = v2::RetrieveGeometryFeatures(context, *this, rect, m_params.m_scale); - - auto const * result = cbv.get(); - features.push_back({m2::Inflate(rect, kComparePoints, kComparePoints), move(cbv), id}); - return result; } SearchModel::SearchType Geocoder::GetSearchTypeInGeocoding(uint32_t featureId) diff --git a/search/v2/geocoder.hpp b/search/v2/geocoder.hpp index 309fbde96c..0c14d6acc1 100644 --- a/search/v2/geocoder.hpp +++ b/search/v2/geocoder.hpp @@ -5,6 +5,7 @@ #include "search/search_query_params.hpp" #include "search/v2/features_layer.hpp" #include "search/v2/features_layer_path_finder.hpp" +#include "search/v2/geometry_cache.hpp" #include "search/v2/mwm_context.hpp" #include "search/v2/nested_rects_cache.hpp" #include "search/v2/pre_ranking_info.hpp" @@ -134,6 +135,9 @@ public: m2::RectD m_rect; SearchModel::SearchType m_type; +#if defined(DEBUG) + string m_defaultName; +#endif }; using TResult = pair; @@ -154,17 +158,18 @@ public: void ClearCaches(); private: + enum RectId + { + RECT_ID_PIVOT, + RECT_ID_LOCALITY, + RECT_ID_COUNT + }; + void GoImpl(vector> & infos, bool inViewport); template using TLocalitiesCache = map, vector>; - enum - { - VIEWPORT_ID = -1, - PIVOT_ID = -2, - }; - SearchQueryParams::TSynonymsVector const & GetTokens(size_t i) const; // Fills |m_retrievalParams| with [curToken, endToken) subsequence @@ -253,9 +258,8 @@ private: unique_ptr LoadVillages(MwmContext & context); /// A caching wrapper around Retrieval::RetrieveGeometryFeatures. - /// param[in] Optional query id. Use VIEWPORT_ID, POSITION_ID or feature index for locality. - coding::CompressedBitVector const * RetrieveGeometryFeatures( - MwmContext const & context, m2::RectD const & rect, int id); + coding::CompressedBitVector const * RetrieveGeometryFeatures(MwmContext const & context, + m2::RectD const & rect, RectId id); // This is a faster wrapper around SearchModel::GetSearchType(), as // it uses pre-loaded lists of streets and villages. @@ -302,15 +306,13 @@ private: TLocalitiesCache m_cities; TLocalitiesCache m_regions[REGION_TYPE_COUNT]; - // Cache of geometry features. - struct FeaturesInRect - { - m2::RectD m_rect; - unique_ptr m_cbv; - int m_id; - }; - map> m_geometryFeatures; + // Caches of features in rects. These caches are separated from + // TLocalitiesCache because the latter are quite lightweight and not + // all of them are needed. + GeometryCache m_pivotRectsCache; + GeometryCache m_localityRectsCache; + // Cache of nested rects used to estimate distance from a feature to the pivot. NestedRectsCache m_pivotFeatures; // Cache of posting lists for each token in the query. TODO (@y, diff --git a/search/v2/geometry_cache.cpp b/search/v2/geometry_cache.cpp new file mode 100644 index 0000000000..0779771c0e --- /dev/null +++ b/search/v2/geometry_cache.cpp @@ -0,0 +1,66 @@ +#include "search/v2/geometry_cache.hpp" + +#include "search/retrieval.hpp" +#include "search/v2/mwm_context.hpp" + +#include "geometry/mercator.hpp" + +#include "base/assert.hpp" + +#include "std/algorithm.hpp" +#include "std/utility.hpp" + +namespace search +{ +namespace v2 +{ +namespace +{ +double constexpr kComparePoints = MercatorBounds::GetCellID2PointAbsEpsilon(); +} // namespace + +GeometryCache::Entry::Entry() : m_scale(0) {} + +GeometryCache::Entry::Entry(m2::RectD const & lowerBound, m2::RectD const & upperBound, + unique_ptr cbv, int scale) + : m_lowerBound(lowerBound), m_upperBound(upperBound), m_cbv(move(cbv)), m_scale(scale) +{ +} + +bool GeometryCache::Entry::Matches(m2::RectD const & rect, int scale) const +{ + return m_scale == scale && rect.IsRectInside(m_lowerBound) && m_upperBound.IsRectInside(rect); +} + +GeometryCache::GeometryCache(size_t maxNumEntries, my::Cancellable const & cancellable) + : m_maxNumEntries(maxNumEntries), m_cancellable(cancellable) +{ + CHECK_GREATER(m_maxNumEntries, 0, ()); +} + +coding::CompressedBitVector const * GeometryCache::Get(MwmContext const & context, + m2::RectD const & rect, int scale) +{ + auto & entries = m_entries[context.GetId()]; + auto it = entries.begin(); + for (; it != entries.end() && !it->Matches(rect, scale); ++it) + ; + if (it != entries.end()) + { + if (it != entries.begin()) + iter_swap(entries.begin(), it); + return entries.front().m_cbv.get(); + } + + auto cbv = v2::RetrieveGeometryFeatures(context, m_cancellable, rect, scale); + entries.emplace_front(rect, m2::Inflate(rect, kComparePoints, kComparePoints), move(cbv), scale); + if (entries.size() == m_maxNumEntries + 1) + entries.pop_back(); + + ASSERT_LESS_OR_EQUAL(entries.size(), m_maxNumEntries, ()); + ASSERT(!entries.empty(), ()); + + return entries.front().m_cbv.get(); +} +} // namespace v2 +} // namespace search diff --git a/search/v2/geometry_cache.hpp b/search/v2/geometry_cache.hpp new file mode 100644 index 0000000000..9b9b399bf5 --- /dev/null +++ b/search/v2/geometry_cache.hpp @@ -0,0 +1,64 @@ +#pragma once + +#include "indexer/mwm_set.hpp" + +#include "coding/compressed_bit_vector.hpp" + +#include "geometry/rect2d.hpp" + +#include "std/cstdint.hpp" +#include "std/deque.hpp" +#include "std/map.hpp" +#include "std/unique_ptr.hpp" + +namespace my +{ +class Cancellable; +}; + +namespace search +{ +namespace v2 +{ +class MwmContext; + +// This class represents a simple cache of features in rects for all mwms. +// +// *NOTE* This class is not thread-safe. +class GeometryCache +{ +public: + // |maxNumEntries| denotes the maximum number of rectangles that + // will be cached for each mwm individually. + GeometryCache(size_t maxNumEntries, my::Cancellable const & cancellable); + + // Returns (hopefully, cached) list of features in a given + // rect. Note that return value may be invalidated on next calls to + // this method. + coding::CompressedBitVector const * Get(MwmContext const & context, m2::RectD const & rect, + int scale); + + inline void Clear() { m_entries.clear(); } + +private: + struct Entry + { + Entry(); + + Entry(m2::RectD const & lowerBound, m2::RectD const & upperBound, + unique_ptr cbv, int scale); + + bool Matches(m2::RectD const & rect, int scale) const; + + m2::RectD m_lowerBound; + m2::RectD m_upperBound; + unique_ptr m_cbv; + int m_scale; + }; + + map> m_entries; + size_t const m_maxNumEntries; + my::Cancellable const & m_cancellable; +}; +} // namespace v2 +} // namespace search diff --git a/search/v2/nested_rects_cache.cpp b/search/v2/nested_rects_cache.cpp index cb4dfa56ef..8708e4b00a 100644 --- a/search/v2/nested_rects_cache.cpp +++ b/search/v2/nested_rects_cache.cpp @@ -33,7 +33,7 @@ void NestedRectsCache::SetPosition(m2::PointD const & position, int scale) return; m_position = position; m_scale = scale; - UpdateCaches(); + Update(); } double NestedRectsCache::GetDistanceToFeatureMeters(FeatureID const & id) const @@ -61,7 +61,7 @@ double NestedRectsCache::GetDistanceToFeatureMeters(FeatureID const & id) const return RankingInfo::kMaxDistMeters; } -void NestedRectsCache::ClearCaches() +void NestedRectsCache::Clear() { for (int scale = 0; scale != RECT_SCALE_COUNT; ++scale) { @@ -84,7 +84,7 @@ double NestedRectsCache::GetRadiusMeters(RectScale scale) } } -void NestedRectsCache::UpdateCaches() +void NestedRectsCache::Update() { for (int scale = 0; scale != RECT_SCALE_COUNT; ++scale) { diff --git a/search/v2/nested_rects_cache.hpp b/search/v2/nested_rects_cache.hpp index 9e0a4daece..a4f7387a04 100644 --- a/search/v2/nested_rects_cache.hpp +++ b/search/v2/nested_rects_cache.hpp @@ -19,7 +19,7 @@ public: double GetDistanceToFeatureMeters(FeatureID const & id) const; - void ClearCaches(); + void Clear(); private: enum RectScale @@ -34,7 +34,7 @@ private: static double GetRadiusMeters(RectScale scale); - void UpdateCaches(); + void Update(); Index & m_index; int m_scale; diff --git a/std/algorithm.hpp b/std/algorithm.hpp index 8c49244ac3..46d9c64fef 100644 --- a/std/algorithm.hpp +++ b/std/algorithm.hpp @@ -17,6 +17,7 @@ using std::find_first_of; using std::find_if; using std::for_each; using std::is_sorted; +using std::iter_swap; using std::lexicographical_compare; using std::lower_bound; using std::max;