forked from organicmaps/organicmaps
[search] Implemented bottom-up intersection pass.
This commit is contained in:
parent
80d7ff175f
commit
a63f77e8b6
13 changed files with 343 additions and 93 deletions
|
@ -22,11 +22,7 @@ int const kQueryScale = scales::GetUpperScale();
|
|||
// static
|
||||
double const ReverseGeocoder::kLookupRadiusM = 500.0;
|
||||
|
||||
// static
|
||||
m2::RectD ReverseGeocoder::GetLookupRect(m2::PointD const & center)
|
||||
{
|
||||
return MercatorBounds::RectByCenterXYAndSizeInMeters(center, kLookupRadiusM);
|
||||
}
|
||||
ReverseGeocoder::ReverseGeocoder(Index const & index) : m_index(index) {}
|
||||
|
||||
void ReverseGeocoder::GetNearbyStreets(FeatureType const & addrFt, vector<Street> & streets)
|
||||
{
|
||||
|
@ -35,7 +31,7 @@ void ReverseGeocoder::GetNearbyStreets(FeatureType const & addrFt, vector<Street
|
|||
|
||||
void ReverseGeocoder::GetNearbyStreets(m2::PointD const & center, vector<Street> & streets)
|
||||
{
|
||||
m2::RectD const rect = GetLookupRect(center);
|
||||
m2::RectD const rect = GetLookupRect(center, kLookupRadiusM);
|
||||
|
||||
auto const addStreet = [&](FeatureType const & ft)
|
||||
{
|
||||
|
@ -51,38 +47,18 @@ void ReverseGeocoder::GetNearbyStreets(m2::PointD const & center, vector<Street>
|
|||
return;
|
||||
|
||||
ASSERT(!name.empty(), ());
|
||||
streets.push_back({ft.GetID(), feature::GetMinDistanceMeters(ft, center), name});
|
||||
|
||||
double const distanceM = feature::GetMinDistanceMeters(ft, center);
|
||||
if (distanceM > kLookupRadiusM)
|
||||
return;
|
||||
|
||||
streets.push_back({ft.GetID(), distanceM, name});
|
||||
};
|
||||
|
||||
m_index.ForEachInRect(addStreet, rect, kQueryScale);
|
||||
sort(streets.begin(), streets.end(), my::CompareBy(&Street::m_distanceMeters));
|
||||
}
|
||||
|
||||
void ReverseGeocoder::GetNearbyBuildings(m2::PointD const & center, vector<Building> & buildings)
|
||||
{
|
||||
// Seems like a copy-paste here of the GetNearbyStreets function.
|
||||
// Trying to factor out common logic will cause many variables logic.
|
||||
|
||||
m2::RectD const rect = GetLookupRect(center);
|
||||
|
||||
auto const addBuilding = [&](FeatureType const & ft)
|
||||
{
|
||||
if (!ftypes::IsBuildingChecker::Instance()(ft))
|
||||
return;
|
||||
|
||||
// Skip empty house numbers.
|
||||
string const number = ft.GetHouseNumber();
|
||||
if (number.empty())
|
||||
return;
|
||||
|
||||
buildings.push_back({ft.GetID(), feature::GetMinDistanceMeters(ft, center),
|
||||
number, feature::GetCenter(ft)});
|
||||
};
|
||||
|
||||
m_index.ForEachInRect(addBuilding, rect, kQueryScale);
|
||||
sort(buildings.begin(), buildings.end(), my::CompareBy(&Building::m_distanceMeters));
|
||||
}
|
||||
|
||||
// static
|
||||
size_t ReverseGeocoder::GetMatchedStreetIndex(string const & keyName,
|
||||
vector<Street> const & streets)
|
||||
|
@ -151,4 +127,43 @@ void ReverseGeocoder::GetNearbyAddress(m2::PointD const & center, Address & addr
|
|||
}
|
||||
}
|
||||
|
||||
void ReverseGeocoder::GetNearbyBuildings(m2::PointD const & center, vector<Building> & buildings)
|
||||
{
|
||||
GetNearbyBuildings(center, kLookupRadiusM, buildings);
|
||||
}
|
||||
|
||||
void ReverseGeocoder::GetNearbyBuildings(m2::PointD const & center, double radiusM,
|
||||
vector<Building> & buildings)
|
||||
{
|
||||
// Seems like a copy-paste here of the GetNearbyStreets function.
|
||||
// Trying to factor out common logic will cause many variables logic.
|
||||
|
||||
m2::RectD const rect = GetLookupRect(center, radiusM);
|
||||
|
||||
auto const addBuilding = [&](FeatureType const & ft)
|
||||
{
|
||||
if (!ftypes::IsBuildingChecker::Instance()(ft))
|
||||
return;
|
||||
|
||||
// Skip empty house numbers.
|
||||
string const number = ft.GetHouseNumber();
|
||||
if (number.empty())
|
||||
return;
|
||||
|
||||
double const distanceM = feature::GetMinDistanceMeters(ft, center);
|
||||
if (distanceM > radiusM)
|
||||
return;
|
||||
|
||||
buildings.push_back({ft.GetID(), distanceM, number, feature::GetCenter(ft)});
|
||||
};
|
||||
|
||||
m_index.ForEachInRect(addBuilding, rect, kQueryScale);
|
||||
sort(buildings.begin(), buildings.end(), my::CompareBy(&Building::m_distanceMeters));
|
||||
}
|
||||
|
||||
// static
|
||||
m2::RectD ReverseGeocoder::GetLookupRect(m2::PointD const & center, double radiusM)
|
||||
{
|
||||
return MercatorBounds::RectByCenterXYAndSizeInMeters(center, radiusM);
|
||||
}
|
||||
} // namespace search
|
||||
|
|
|
@ -34,9 +34,8 @@ class ReverseGeocoder
|
|||
|
||||
public:
|
||||
static double const kLookupRadiusM;
|
||||
static m2::RectD GetLookupRect(m2::PointD const & center);
|
||||
|
||||
explicit ReverseGeocoder(Index const & index) : m_index(index) {}
|
||||
explicit ReverseGeocoder(Index const & index);
|
||||
|
||||
using Street = Object;
|
||||
|
||||
|
@ -69,9 +68,14 @@ public:
|
|||
|
||||
void GetNearbyAddress(m2::PointD const & center, Address & addr);
|
||||
|
||||
private:
|
||||
void GetNearbyStreets(m2::PointD const & center, vector<Street> & streets);
|
||||
void GetNearbyBuildings(m2::PointD const & center, vector<Building> & buildings);
|
||||
|
||||
void GetNearbyBuildings(m2::PointD const & center, double radiusM, vector<Building> & buildings);
|
||||
|
||||
private:
|
||||
static m2::RectD GetLookupRect(m2::PointD const & center, double radiusM);
|
||||
|
||||
void GetNearbyStreets(m2::PointD const & center, vector<Street> & streets);
|
||||
};
|
||||
|
||||
} // namespace search
|
||||
|
|
|
@ -83,18 +83,24 @@ UNIT_TEST(SearchQueryV2_Smoke)
|
|||
auto const feynmanStreet = make_shared<TestStreet>(
|
||||
vector<m2::PointD>{m2::PointD(9.999, 9.999), m2::PointD(10, 10), m2::PointD(10.001, 10.001)},
|
||||
"Feynman street", "en");
|
||||
auto const bohrStreet = make_shared<TestStreet>(
|
||||
auto const bohrStreet1 = make_shared<TestStreet>(
|
||||
vector<m2::PointD>{m2::PointD(9.999, 10.001), m2::PointD(10, 10), m2::PointD(10.001, 9.999)},
|
||||
"Bohr street", "en");
|
||||
auto const feynmanHouse = make_shared<TestBuilding>(m2::PointD(10, 10), "Feynman house",
|
||||
auto const bohrStreet2 = make_shared<TestStreet>(
|
||||
vector<m2::PointD>{m2::PointD(10.001, 9.999), m2::PointD(10, 10), m2::PointD(10.002, 9.998)},
|
||||
"Bohr street", "en");
|
||||
auto const bohrStreet3 = make_shared<TestStreet>(
|
||||
vector<m2::PointD>{m2::PointD(10.002, 9.998), m2::PointD(10, 10), m2::PointD(10.003, 9.997)},
|
||||
"Bohr street", "en");
|
||||
auto const feynmanHouse = make_shared<TestBuilding>(m2::PointD(10, 10), "Feynman house 1 unit 1",
|
||||
"1 unit 1", *feynmanStreet, "en");
|
||||
auto const bohrHouse =
|
||||
make_shared<TestBuilding>(m2::PointD(10, 10), "Bohr house", "1 unit 1", *bohrStreet, "en");
|
||||
auto const bohrHouse = make_shared<TestBuilding>(m2::PointD(10, 10), "Bohr house 1 unit 1 ",
|
||||
"1 unit 1", *bohrStreet1, "en");
|
||||
|
||||
auto const hilbertHouse = make_shared<TestBuilding>(
|
||||
vector<m2::PointD>{
|
||||
{10.0005, 10.0005}, {10.0006, 10.0005}, {10.0006, 10.0006}, {10.0005, 10.0006}},
|
||||
"Hilbert house", "1 unit 2", *bohrStreet, "en");
|
||||
"Hilbert house 1 unit 2", "1 unit 2", *bohrStreet1, "en");
|
||||
auto const lantern1 = make_shared<TestPOI>(m2::PointD(10.0005, 10.0005), "lantern 1", "en");
|
||||
auto const lantern2 = make_shared<TestPOI>(m2::PointD(10.0006, 10.0005), "lantern 2", "en");
|
||||
|
||||
|
@ -107,7 +113,9 @@ UNIT_TEST(SearchQueryV2_Smoke)
|
|||
builder.Add(*quantumTeleport2);
|
||||
builder.Add(*quantumCafe);
|
||||
builder.Add(*feynmanStreet);
|
||||
builder.Add(*bohrStreet);
|
||||
builder.Add(*bohrStreet1);
|
||||
builder.Add(*bohrStreet2);
|
||||
builder.Add(*bohrStreet3);
|
||||
builder.Add(*feynmanHouse);
|
||||
builder.Add(*bohrHouse);
|
||||
|
||||
|
@ -196,7 +204,7 @@ UNIT_TEST(SearchQueryV2_Smoke)
|
|||
}
|
||||
|
||||
{
|
||||
TestSearchRequest request(engine, "bohr street 1 unit 2 lantern ", "en",
|
||||
TestSearchRequest request(engine, "bohr street 1 lantern ", "en",
|
||||
search::SearchParams::ALL, viewport);
|
||||
request.Wait();
|
||||
vector<shared_ptr<MatchingRule>> rules = {make_shared<ExactMatch>(wonderlandId, lantern1),
|
||||
|
|
|
@ -17,6 +17,7 @@ void FeaturesLayer::Clear()
|
|||
m_startToken = 0;
|
||||
m_endToken = 0;
|
||||
m_type = SearchModel::SEARCH_TYPE_COUNT;
|
||||
m_hasDelayedFeatures = false;
|
||||
}
|
||||
|
||||
string DebugPrint(FeaturesLayer const & layer)
|
||||
|
|
|
@ -25,6 +25,15 @@ struct FeaturesLayer
|
|||
size_t m_startToken;
|
||||
size_t m_endToken;
|
||||
SearchModel::SearchType m_type;
|
||||
|
||||
// *NOTE* This field is meaningful only when m_type equals to
|
||||
// SEARCH_TYPE_BUILDING.
|
||||
//
|
||||
// When true, m_sortedFeatures contains only features retrieved from
|
||||
// search index by m_subQuery, and it's necessary for Geocoder to
|
||||
// perform additional work to retrieve features matching by house
|
||||
// number.
|
||||
bool m_hasDelayedFeatures;
|
||||
};
|
||||
|
||||
string DebugPrint(FeaturesLayer const & layer);
|
||||
|
|
|
@ -10,6 +10,9 @@ namespace search
|
|||
{
|
||||
namespace v2
|
||||
{
|
||||
// static
|
||||
double const FeaturesLayerMatcher::kBuildingRadiusMeters = 50;
|
||||
|
||||
FeaturesLayerMatcher::FeaturesLayerMatcher(Index & index, MwmContext & context,
|
||||
my::Cancellable const & cancellable)
|
||||
: m_context(context)
|
||||
|
@ -65,6 +68,18 @@ vector<ReverseGeocoder::Street> const & FeaturesLayerMatcher::GetNearbyStreets(
|
|||
return GetNearbyStreetsImpl(featureId, feature);
|
||||
}
|
||||
|
||||
vector<ReverseGeocoder::Building> const & FeaturesLayerMatcher::GetNearbyBuildings(
|
||||
m2::PointD const & center)
|
||||
{
|
||||
auto const it = m_nearbyBuildingsCache.find(center);
|
||||
if (it != m_nearbyBuildingsCache.cend())
|
||||
return it->second;
|
||||
|
||||
auto & buildings = m_nearbyBuildingsCache[center];
|
||||
m_reverseGeocoder.GetNearbyBuildings(center, kBuildingRadiusMeters, buildings);
|
||||
return buildings;
|
||||
}
|
||||
|
||||
uint32_t FeaturesLayerMatcher::GetMatchingStreetImpl(uint32_t houseId, FeatureType & houseFeature)
|
||||
{
|
||||
auto const & streets = GetNearbyStreets(houseId, houseFeature);
|
||||
|
|
|
@ -56,6 +56,7 @@ class FeaturesLayerMatcher
|
|||
{
|
||||
public:
|
||||
static uint32_t const kInvalidId = numeric_limits<uint32_t>::max();
|
||||
static double const kBuildingRadiusMeters;
|
||||
|
||||
FeaturesLayerMatcher(Index & index, MwmContext & context, my::Cancellable const & cancellable);
|
||||
|
||||
|
@ -91,34 +92,39 @@ private:
|
|||
template <typename TFn>
|
||||
void MatchPOIsWithBuildings(FeaturesLayer const & child, FeaturesLayer const & parent, TFn && fn)
|
||||
{
|
||||
static double const kBuildingRadiusMeters = 50;
|
||||
// Following code initially loads centers of POIs, and, then, for
|
||||
// each building, tries to find all POIs located at distance less
|
||||
// than kBuildingRadiusMeters.
|
||||
|
||||
ASSERT_EQUAL(child.m_type, SearchModel::SEARCH_TYPE_POI, ());
|
||||
ASSERT_EQUAL(parent.m_type, SearchModel::SEARCH_TYPE_BUILDING, ());
|
||||
|
||||
auto const & pois = *child.m_sortedFeatures;
|
||||
auto const & buildings = *parent.m_sortedFeatures;
|
||||
|
||||
BailIfCancelled(m_cancellable);
|
||||
|
||||
vector<m2::PointD> poiCenters(child.m_sortedFeatures->size());
|
||||
vector<m2::PointD> poiCenters(pois.size());
|
||||
|
||||
size_t const numPOIs = child.m_sortedFeatures->size();
|
||||
size_t const numPOIs = pois.size();
|
||||
vector<bool> isPOIProcessed(numPOIs);
|
||||
size_t processedPOIs = 0;
|
||||
|
||||
for (size_t i = 0; i < child.m_sortedFeatures->size(); ++i)
|
||||
for (size_t i = 0; i < pois.size(); ++i)
|
||||
{
|
||||
FeatureType poiFt;
|
||||
GetByIndex((*child.m_sortedFeatures)[i], poiFt);
|
||||
GetByIndex(pois[i], poiFt);
|
||||
poiCenters[i] = feature::GetCenter(poiFt, FeatureType::WORST_GEOMETRY);
|
||||
}
|
||||
|
||||
for (size_t i = 0; i < parent.m_sortedFeatures->size() && processedPOIs != numPOIs; ++i)
|
||||
for (size_t i = 0; i < buildings.size() && processedPOIs != numPOIs; ++i)
|
||||
{
|
||||
BailIfCancelled(m_cancellable);
|
||||
|
||||
FeatureType buildingFt;
|
||||
GetByIndex((*parent.m_sortedFeatures)[i], buildingFt);
|
||||
GetByIndex(buildings[i], buildingFt);
|
||||
|
||||
for (size_t j = 0; j < child.m_sortedFeatures->size(); ++j)
|
||||
for (size_t j = 0; j < pois.size(); ++j)
|
||||
{
|
||||
if (isPOIProcessed[j])
|
||||
continue;
|
||||
|
@ -126,12 +132,33 @@ private:
|
|||
double const distMeters = feature::GetMinDistanceMeters(buildingFt, poiCenters[j]);
|
||||
if (distMeters <= kBuildingRadiusMeters)
|
||||
{
|
||||
fn((*child.m_sortedFeatures)[j], (*parent.m_sortedFeatures)[i]);
|
||||
fn(pois[j], buildings[i]);
|
||||
isPOIProcessed[j] = true;
|
||||
++processedPOIs;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (!parent.m_hasDelayedFeatures)
|
||||
return;
|
||||
|
||||
// |buildings| do not contain buildings matching by house number, so
|
||||
// following code reads buildings in POIs vicinities and checks
|
||||
// house numbers.
|
||||
auto const & mwmId = m_context.m_handle.GetId();
|
||||
vector<string> queryTokens;
|
||||
NormalizeHouseNumber(parent.m_subQuery, queryTokens);
|
||||
|
||||
for (size_t i = 0; i < pois.size(); ++i)
|
||||
{
|
||||
for (auto const & building : GetNearbyBuildings(poiCenters[i]))
|
||||
{
|
||||
if (building.m_id.m_mwmId != mwmId)
|
||||
continue;
|
||||
if (HouseNumbersMatch(building.m_name, queryTokens))
|
||||
fn(pois[i], building.m_id.m_index);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
template <typename TFn>
|
||||
|
@ -140,10 +167,33 @@ private:
|
|||
ASSERT_EQUAL(child.m_type, SearchModel::SEARCH_TYPE_POI, ());
|
||||
ASSERT_EQUAL(parent.m_type, SearchModel::SEARCH_TYPE_STREET, ());
|
||||
|
||||
for (uint32_t streetId : *parent.m_sortedFeatures)
|
||||
auto const & pois = *child.m_sortedFeatures;
|
||||
auto const & streets = *parent.m_sortedFeatures;
|
||||
|
||||
// When number of POIs less than number of STREETs, it's faster to
|
||||
// check nearby streets for POIs.
|
||||
if (pois.size() < streets.size())
|
||||
{
|
||||
auto const & mwmId = m_context.m_handle.GetId();
|
||||
for (uint32_t poiId : pois)
|
||||
{
|
||||
for (auto const & street : GetNearbyStreets(poiId))
|
||||
{
|
||||
if (street.m_id.m_mwmId != mwmId)
|
||||
continue;
|
||||
|
||||
uint32_t const streetId = street.m_id.m_index;
|
||||
if (binary_search(streets.begin(), streets.end(), streetId))
|
||||
fn(poiId, streetId);
|
||||
}
|
||||
}
|
||||
return;
|
||||
}
|
||||
|
||||
for (uint32_t streetId : streets)
|
||||
{
|
||||
BailIfCancelled(m_cancellable);
|
||||
m_loader.ForEachInVicinity(streetId, *child.m_sortedFeatures, bind(fn, _1, streetId));
|
||||
m_loader.ForEachInVicinity(streetId, pois, bind(fn, _1, streetId));
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -151,37 +201,31 @@ private:
|
|||
void MatchBuildingsWithStreets(FeaturesLayer const & child, FeaturesLayer const & parent,
|
||||
TFn && fn)
|
||||
{
|
||||
// child.m_sortedFeatures contains only buildings matched by name,
|
||||
// not by house number. So, we need to add to
|
||||
// child.m_sortedFeatures all buildings match by house number
|
||||
// here.
|
||||
|
||||
ASSERT_EQUAL(child.m_type, SearchModel::SEARCH_TYPE_BUILDING, ());
|
||||
ASSERT_EQUAL(parent.m_type, SearchModel::SEARCH_TYPE_STREET, ());
|
||||
|
||||
vector<string> queryTokens;
|
||||
NormalizeHouseNumber(child.m_subQuery, queryTokens);
|
||||
bool const queryLooksLikeHouseNumber =
|
||||
!queryTokens.empty() && feature::IsHouseNumber(child.m_subQuery);
|
||||
auto const & buildings = *child.m_sortedFeatures;
|
||||
auto const & streets = *parent.m_sortedFeatures;
|
||||
|
||||
// When building name does not look like a house number it's
|
||||
// faster to check nearby streets for each building instead of
|
||||
// street vicinities loading.
|
||||
if (!queryLooksLikeHouseNumber &&
|
||||
child.m_sortedFeatures->size() < parent.m_sortedFeatures->size())
|
||||
// When all buildings are in |buildings| and number of buildins
|
||||
// less than number of streets, it's probably faster to check
|
||||
// nearby streets for each building instead of street vicinities
|
||||
// loading.
|
||||
if (!child.m_hasDelayedFeatures && buildings.size() < streets.size())
|
||||
{
|
||||
auto const & streets = *parent.m_sortedFeatures;
|
||||
for (uint32_t houseId : *child.m_sortedFeatures)
|
||||
{
|
||||
uint32_t streetId = GetMatchingStreet(houseId);
|
||||
if (binary_search(parent.m_sortedFeatures->begin(), parent.m_sortedFeatures->end(),
|
||||
streetId))
|
||||
{
|
||||
if (binary_search(streets.begin(), streets.end(), streetId))
|
||||
fn(houseId, streetId);
|
||||
}
|
||||
}
|
||||
return;
|
||||
}
|
||||
|
||||
vector<string> queryTokens;
|
||||
NormalizeHouseNumber(child.m_subQuery, queryTokens);
|
||||
|
||||
auto const & checker = ftypes::IsBuildingChecker::Instance();
|
||||
uint32_t numFilterInvocations = 0;
|
||||
auto houseNumberFilter = [&](uint32_t id, FeatureType & feature, bool & loaded) -> bool
|
||||
|
@ -190,7 +234,7 @@ private:
|
|||
if ((numFilterInvocations & 0xFF) == 0)
|
||||
BailIfCancelled(m_cancellable);
|
||||
|
||||
if (binary_search(child.m_sortedFeatures->begin(), child.m_sortedFeatures->end(), id))
|
||||
if (binary_search(buildings.begin(), buildings.end(), id))
|
||||
return true;
|
||||
|
||||
// HouseNumbersMatch() calls are expensive, so following code
|
||||
|
@ -204,13 +248,15 @@ private:
|
|||
GetByIndex(id, feature);
|
||||
loaded = true;
|
||||
}
|
||||
|
||||
if (!checker(feature))
|
||||
return false;
|
||||
|
||||
string const houseNumber = feature.GetHouseNumber();
|
||||
if (!queryLooksLikeHouseNumber || !feature::IsHouseNumber(houseNumber))
|
||||
if (!child.m_hasDelayedFeatures)
|
||||
return false;
|
||||
if (queryTokens[0][0] != houseNumber[0])
|
||||
|
||||
string const houseNumber = feature.GetHouseNumber();
|
||||
if (!feature::IsHouseNumber(houseNumber))
|
||||
return false;
|
||||
return HouseNumbersMatch(houseNumber, queryTokens);
|
||||
};
|
||||
|
@ -227,7 +273,7 @@ private:
|
|||
};
|
||||
|
||||
ProjectionOnStreet proj;
|
||||
for (uint32_t streetId : *parent.m_sortedFeatures)
|
||||
for (uint32_t streetId : streets)
|
||||
{
|
||||
BailIfCancelled(m_cancellable);
|
||||
StreetVicinityLoader::Street const & street = m_loader.GetStreet(streetId);
|
||||
|
@ -267,6 +313,8 @@ private:
|
|||
vector<ReverseGeocoder::Street> const & GetNearbyStreets(uint32_t featureId,
|
||||
FeatureType & feature);
|
||||
|
||||
vector<ReverseGeocoder::Building> const & GetNearbyBuildings(m2::PointD const & center);
|
||||
|
||||
uint32_t GetMatchingStreetImpl(uint32_t houseId, FeatureType & houseFeature);
|
||||
|
||||
vector<ReverseGeocoder::Street> const & GetNearbyStreetsImpl(uint32_t featureId,
|
||||
|
@ -282,9 +330,14 @@ private:
|
|||
ReverseGeocoder m_reverseGeocoder;
|
||||
|
||||
// Cache of streets in a feature's vicinity. All lists in the cache
|
||||
// are ordered by distance.
|
||||
// are ordered by distance from feature.
|
||||
unordered_map<uint32_t, vector<ReverseGeocoder::Street>> m_nearbyStreetsCache;
|
||||
|
||||
// Cache of buildings near in a POI's vicinity. All lists in the
|
||||
// cache are ordered by distance from POI.
|
||||
unordered_map<m2::PointD, vector<ReverseGeocoder::Building>, m2::PointD::Hash>
|
||||
m_nearbyBuildingsCache;
|
||||
|
||||
// Cache of correct streets for buildings. Current search algorithm
|
||||
// supports only one street for a building, whereas buildings can be
|
||||
// located on multiple streets.
|
||||
|
|
|
@ -11,6 +11,39 @@ namespace search
|
|||
{
|
||||
namespace v2
|
||||
{
|
||||
namespace
|
||||
{
|
||||
// This function tries to estimate amount of work needed to perform an
|
||||
// intersection pass on a sequence of layers.
|
||||
template<typename TIt>
|
||||
uint64_t CalcPassCost(TIt begin, TIt end)
|
||||
{
|
||||
uint64_t cost = 0;
|
||||
|
||||
if (begin == end)
|
||||
return cost;
|
||||
|
||||
uint64_t reachable = max((*begin)->m_sortedFeatures->size(), static_cast<uint64_t>(1));
|
||||
for (++begin; begin != end; ++begin)
|
||||
{
|
||||
uint64_t const layer = max((*begin)->m_sortedFeatures->size(), static_cast<uint64_t>(1));
|
||||
cost += layer * reachable;
|
||||
reachable = min(reachable, layer);
|
||||
}
|
||||
return cost;
|
||||
}
|
||||
|
||||
uint64_t CalcTopDownPassCost(vector<FeaturesLayer const *> const & layers)
|
||||
{
|
||||
return CalcPassCost(layers.rbegin(), layers.rend());
|
||||
}
|
||||
|
||||
uint64_t CalcBottomUpPassCost(vector<FeaturesLayer const *> const & layers)
|
||||
{
|
||||
return CalcPassCost(layers.begin(), layers.end());
|
||||
}
|
||||
} // namespace
|
||||
|
||||
FeaturesLayerPathFinder::FeaturesLayerPathFinder(my::Cancellable const & cancellable)
|
||||
: m_cancellable(cancellable)
|
||||
{
|
||||
|
@ -23,10 +56,28 @@ void FeaturesLayerPathFinder::FindReachableVertices(FeaturesLayerMatcher & match
|
|||
if (layers.empty())
|
||||
return;
|
||||
|
||||
uint64_t const topDownCost = CalcTopDownPassCost(layers);
|
||||
uint64_t const bottomUpCost = CalcBottomUpPassCost(layers);
|
||||
|
||||
if (bottomUpCost < topDownCost)
|
||||
FindReachableVerticesBottomUp(matcher, layers, reachable);
|
||||
else
|
||||
FindReachableVerticesTopDown(matcher, layers, reachable);
|
||||
}
|
||||
|
||||
void FeaturesLayerPathFinder::FindReachableVerticesTopDown(
|
||||
FeaturesLayerMatcher & matcher, vector<FeaturesLayer const *> const & layers,
|
||||
vector<uint32_t> & reachable)
|
||||
{
|
||||
reachable = *(layers.back()->m_sortedFeatures);
|
||||
|
||||
vector<uint32_t> buffer;
|
||||
|
||||
auto addEdge = [&](uint32_t childFeature, uint32_t /* parentFeature */)
|
||||
{
|
||||
buffer.push_back(childFeature);
|
||||
};
|
||||
|
||||
// The order matters here, as we need to intersect BUILDINGs with
|
||||
// STREETs first, and then POIs with BUILDINGs.
|
||||
for (size_t i = layers.size() - 1; i != 0; --i)
|
||||
|
@ -36,19 +87,92 @@ void FeaturesLayerPathFinder::FindReachableVertices(FeaturesLayerMatcher & match
|
|||
if (reachable.empty())
|
||||
break;
|
||||
|
||||
buffer.clear();
|
||||
auto addEdge = [&](uint32_t childFeature, uint32_t /* parentFeature */)
|
||||
{
|
||||
buffer.push_back(childFeature);
|
||||
};
|
||||
|
||||
FeaturesLayer parent(*layers[i]);
|
||||
parent.m_sortedFeatures = &reachable;
|
||||
matcher.Match(*layers[i - 1], parent, addEdge);
|
||||
parent.m_hasDelayedFeatures = false;
|
||||
|
||||
FeaturesLayer child(*layers[i - 1]);
|
||||
child.m_hasDelayedFeatures = false;
|
||||
if (child.m_type == SearchModel::SEARCH_TYPE_BUILDING)
|
||||
{
|
||||
vector<string> tokens;
|
||||
NormalizeHouseNumber(child.m_subQuery, tokens);
|
||||
bool const looksLikeHouseNumber = !tokens.empty() && feature::IsHouseNumber(tokens.front());
|
||||
child.m_hasDelayedFeatures = looksLikeHouseNumber;
|
||||
}
|
||||
|
||||
buffer.clear();
|
||||
matcher.Match(child, parent, addEdge);
|
||||
reachable.swap(buffer);
|
||||
my::SortUnique(reachable);
|
||||
}
|
||||
}
|
||||
|
||||
void FeaturesLayerPathFinder::FindReachableVerticesBottomUp(
|
||||
FeaturesLayerMatcher & matcher, vector<FeaturesLayer const *> const & layers,
|
||||
vector<uint32_t> & reachable)
|
||||
{
|
||||
reachable = *(layers.front()->m_sortedFeatures);
|
||||
|
||||
unordered_map<uint32_t, uint32_t> parentGraph;
|
||||
for (uint32_t id : reachable)
|
||||
parentGraph[id] = id;
|
||||
|
||||
unordered_map<uint32_t, uint32_t> nparentGraph;
|
||||
vector<uint32_t> buffer;
|
||||
|
||||
// We're going from low levels to high levels and we need to report
|
||||
// features from the lowest layer that are reachable from the higher
|
||||
// layer by some path. Therefore, following code checks and updates
|
||||
// edges from features in |reachable| to features on the lowest
|
||||
// layer.
|
||||
auto addEdge = [&](uint32_t childFeature, uint32_t parentFeature)
|
||||
{
|
||||
auto const it = parentGraph.find(childFeature);
|
||||
if (it == parentGraph.cend())
|
||||
return;
|
||||
nparentGraph[parentFeature] = it->second;
|
||||
buffer.push_back(parentFeature);
|
||||
};
|
||||
|
||||
|
||||
for (size_t i = 0; i + 1 != layers.size(); ++i)
|
||||
{
|
||||
BailIfCancelled(m_cancellable);
|
||||
|
||||
if (reachable.empty())
|
||||
break;
|
||||
|
||||
FeaturesLayer child(*layers[i]);
|
||||
child.m_sortedFeatures = &reachable;
|
||||
child.m_hasDelayedFeatures = false;
|
||||
|
||||
FeaturesLayer parent(*layers[i + 1]);
|
||||
parent.m_hasDelayedFeatures = false;
|
||||
if (parent.m_type == SearchModel::SEARCH_TYPE_BUILDING)
|
||||
{
|
||||
vector<string> tokens;
|
||||
NormalizeHouseNumber(parent.m_subQuery, tokens);
|
||||
bool const looksLikeHouseNumber = !tokens.empty() && feature::IsHouseNumber(tokens.front());
|
||||
parent.m_hasDelayedFeatures = looksLikeHouseNumber;
|
||||
}
|
||||
|
||||
buffer.clear();
|
||||
nparentGraph.clear();
|
||||
matcher.Match(child, parent, addEdge);
|
||||
parentGraph.swap(nparentGraph);
|
||||
reachable.swap(buffer);
|
||||
my::SortUnique(reachable);
|
||||
}
|
||||
|
||||
buffer.clear();
|
||||
for (uint32_t id : reachable)
|
||||
{
|
||||
ASSERT_NOT_EQUAL(parentGraph.count(id), 0, ());
|
||||
buffer.push_back(parentGraph[id]);
|
||||
}
|
||||
reachable.swap(buffer);
|
||||
my::SortUnique(reachable);
|
||||
}
|
||||
} // namespace v2
|
||||
} // namespace search
|
||||
|
|
|
@ -5,8 +5,8 @@
|
|||
#include "std/vector.hpp"
|
||||
|
||||
#if defined(DEBUG)
|
||||
#include "base/logging.hpp"
|
||||
#include "base/timer.hpp"
|
||||
#include "std/cstdio.hpp"
|
||||
#endif // defined(DEBUG)
|
||||
|
||||
class FeaturesVector;
|
||||
|
@ -48,9 +48,8 @@ public:
|
|||
// FindReachableVertices() will work fast for most cases
|
||||
// (significantly less than 1 second).
|
||||
#if defined(DEBUG)
|
||||
fprintf(stderr, "FeaturesLayerPathFinder()\n");
|
||||
for (auto const * layer : layers)
|
||||
fprintf(stderr, "Layer: %s\n", DebugPrint(*layer).c_str());
|
||||
LOG(LINFO, (DebugPrint(*layer)));
|
||||
my::Timer timer;
|
||||
#endif // defined(DEBUG)
|
||||
|
||||
|
@ -58,7 +57,7 @@ public:
|
|||
FindReachableVertices(matcher, layers, reachable);
|
||||
|
||||
#if defined(DEBUG)
|
||||
fprintf(stderr, "Found: %zu, elapsed: %lf seconds\n", reachable.size(), timer.ElapsedSeconds());
|
||||
LOG(LINFO, ("Found:", reachable.size(), "elapsed:", timer.ElapsedSeconds(), "seconds"));
|
||||
#endif // defined(DEBUG)
|
||||
|
||||
for (uint32_t featureId : reachable)
|
||||
|
@ -70,6 +69,18 @@ private:
|
|||
vector<FeaturesLayer const *> const & layers,
|
||||
vector<uint32_t> & reachable);
|
||||
|
||||
// Tries to find all |reachable| features from the lowest layer in a
|
||||
// high level -> low level pass.
|
||||
void FindReachableVerticesTopDown(FeaturesLayerMatcher & matcher,
|
||||
vector<FeaturesLayer const *> const & layers,
|
||||
vector<uint32_t> & reachable);
|
||||
|
||||
// Tries to find all |reachable| features from the lowest layer in a
|
||||
// low level -> high level pass.
|
||||
void FindReachableVerticesBottomUp(FeaturesLayerMatcher & matcher,
|
||||
vector<FeaturesLayer const *> const & layers,
|
||||
vector<uint32_t> & reachable);
|
||||
|
||||
my::Cancellable const & m_cancellable;
|
||||
};
|
||||
} // namespace v2
|
||||
|
|
|
@ -32,7 +32,6 @@
|
|||
|
||||
#if defined(DEBUG)
|
||||
#include "base/timer.hpp"
|
||||
#include "std/cstdio.hpp"
|
||||
#endif
|
||||
|
||||
#if defined(USE_GOOGLE_PROFILER) && defined(OMIM_OS_LINUX)
|
||||
|
@ -117,7 +116,7 @@ void Geocoder::Go(vector<FeatureID> & results)
|
|||
my::Timer timer;
|
||||
MY_SCOPE_GUARD(printDuration, [&timer]()
|
||||
{
|
||||
fprintf(stderr, "Total geocoding time: %lf seconds.\n", timer.ElapsedSeconds());
|
||||
LOG(LINFO, ("Total geocoding time:", timer.ElapsedSeconds(), "seconds"));
|
||||
});
|
||||
#endif
|
||||
#if defined(USE_GOOGLE_PROFILER) && defined(OMIM_OS_LINUX)
|
||||
|
@ -410,7 +409,7 @@ void Geocoder::DoGeocoding(size_t curToken)
|
|||
FeatureType feature;
|
||||
m_context->m_vector.GetByIndex(featureId, feature);
|
||||
feature.ParseTypes();
|
||||
SearchModel::SearchType searchType = m_model.GetSearchType(feature);
|
||||
SearchModel::SearchType const searchType = m_model.GetSearchType(feature);
|
||||
|
||||
// All SEARCH_TYPE_CITY features were filtered in DoGeocodingWithLocalities().
|
||||
if (searchType < SearchModel::SEARCH_TYPE_CITY)
|
||||
|
|
|
@ -180,10 +180,15 @@ bool HouseNumbersMatch(string const & houseNumber, string const & query)
|
|||
|
||||
bool HouseNumbersMatch(string const & houseNumber, vector<string> const & queryTokens)
|
||||
{
|
||||
if (houseNumber.empty() || queryTokens.empty())
|
||||
return false;
|
||||
if (queryTokens[0][0] != houseNumber[0])
|
||||
return false;
|
||||
|
||||
vector<string> houseNumberTokens;
|
||||
NormalizeHouseNumber(houseNumber, houseNumberTokens);
|
||||
|
||||
if (houseNumberTokens.empty() || queryTokens.empty())
|
||||
if (houseNumberTokens.empty())
|
||||
return false;
|
||||
|
||||
// Check first tokens (hope, house numbers).
|
||||
|
|
|
@ -13,6 +13,8 @@
|
|||
|
||||
namespace search
|
||||
{
|
||||
namespace v2
|
||||
{
|
||||
StreetVicinityLoader::StreetVicinityLoader(MwmValue & value, FeaturesVector const & featuresVector,
|
||||
int scale, double offsetMeters)
|
||||
: m_index(value.m_cont.GetReader(INDEX_FILE_TAG), value.m_factory)
|
||||
|
@ -58,4 +60,5 @@ void StreetVicinityLoader::LoadStreet(uint32_t featureId, Street & street)
|
|||
if (!points.empty())
|
||||
street.m_calculator = make_unique<ProjectionOnStreetCalculator>(move(points), m_offsetMeters);
|
||||
}
|
||||
} // namespace v2
|
||||
} // namespace search
|
||||
|
|
|
@ -19,6 +19,8 @@ class MwmValue;
|
|||
|
||||
namespace search
|
||||
{
|
||||
namespace v2
|
||||
{
|
||||
// This class is able to load features in a street's vicinity.
|
||||
//
|
||||
// NOTE: this class *IS NOT* thread-safe.
|
||||
|
@ -84,4 +86,5 @@ private:
|
|||
|
||||
DISALLOW_COPY_AND_MOVE(StreetVicinityLoader);
|
||||
};
|
||||
} // namespace v2
|
||||
} // namespace search
|
||||
|
|
Loading…
Add table
Reference in a new issue