From c171f12836fd3909030f8ea50dc90383541655be Mon Sep 17 00:00:00 2001 From: Maksim Andrianov Date: Wed, 27 Nov 2019 18:53:02 +0300 Subject: [PATCH] Review fixes --- generator/CMakeLists.txt | 4 +- generator/collector_building_parts.cpp | 36 ++++----- generator/collector_building_parts.hpp | 4 +- generator/collector_interface.hpp | 1 + .../complex_generator/complex_generator.cpp | 4 +- generator/filter_complex.hpp | 2 +- .../final_processor_intermediate_mwm.cpp | 73 +++++++++++-------- .../final_processor_intermediate_mwm.hpp | 4 +- generator/hierarchy.cpp | 45 +++++++----- generator/hierarchy.hpp | 12 +-- generator/intermediate_data.hpp | 2 +- indexer/complex/tree_node.hpp | 8 +- 12 files changed, 109 insertions(+), 86 deletions(-) diff --git a/generator/CMakeLists.txt b/generator/CMakeLists.txt index 20e180befb..2d93810c90 100644 --- a/generator/CMakeLists.txt +++ b/generator/CMakeLists.txt @@ -38,10 +38,10 @@ set( coastlines_generator.cpp coastlines_generator.hpp collection_base.hpp - collector_building_parts.cpp - collector_building_parts.hpp collector_boundary_postcode.cpp collector_boundary_postcode.hpp + collector_building_parts.cpp + collector_building_parts.hpp collector_camera.cpp collector_camera.hpp collector_city_area.cpp diff --git a/generator/collector_building_parts.cpp b/generator/collector_building_parts.cpp index 92abb7fea0..79c1fce28b 100644 --- a/generator/collector_building_parts.cpp +++ b/generator/collector_building_parts.cpp @@ -18,7 +18,6 @@ #include "base/control_flow.hpp" #include "base/stl_helpers.hpp" -#include #include #include #include @@ -32,15 +31,13 @@ class RelationFetcher public: RelationFetcher(IdRelationVec & elements) : m_elements(elements) {} - template - base::ControlFlow operator()(uint64_t id, Reader & reader) + void operator()(uint64_t id, generator::cache::OSMElementCacheReader & reader) { RelationElement element; if (reader.Read(id, element)) m_elements.emplace_back(id, std::move(element)); else LOG(LWARNING, ("Cannot read relation with id", id)); - return base::ControlFlow::Continue; } private: @@ -77,49 +74,50 @@ void BuildingPartsCollector::CollectFeature(feature::FeatureBuilder const & fb, auto const parts = FindAllBuildingParts(topId); if (!parts.empty()) - WritePair(MakeCompositeId(fb), parts); + WriteBuildingParts(MakeCompositeId(fb), parts); } std::vector BuildingPartsCollector::FindAllBuildingParts( base::GeoObjectId const & id) { - std::vector buildingPatrs; + std::vector buildingParts; RelationElement relation; - if (!m_cache->GetRalation(id.GetSerialId(), relation)) + if (!m_cache->GetRelation(id.GetSerialId(), relation)) { LOG(LWARNING, ("Cannot read relation with id", id)); - return buildingPatrs; + return buildingParts; } for (auto const & v : relation.m_ways) { if (v.second == "part") - buildingPatrs.emplace_back(base::MakeOsmWay(v.first)); + buildingParts.emplace_back(base::MakeOsmWay(v.first)); } for (auto const & v : relation.m_relations) { if (v.second == "part") - buildingPatrs.emplace_back(base::MakeOsmRelation(v.first)); + buildingParts.emplace_back(base::MakeOsmRelation(v.first)); } - return buildingPatrs; + return buildingParts; } base::GeoObjectId BuildingPartsCollector::FindTopRelation(OsmElement const & element) { IdRelationVec elements; RelationFetcher fetcher(elements); + base::ControlFlowWrapper wrapper(fetcher); IdRelationVec::const_iterator it; if (element.IsWay()) { - m_cache->ForEachRelationByNodeCached(element.m_id, fetcher); + m_cache->ForEachRelationByWayCached(element.m_id, wrapper); it = base::FindIf(elements, [&](auto const & idRelation) { return idRelation.second.GetWayRole(element.m_id) == "outline"; }); } else if (element.IsRelation()) { - m_cache->ForEachRelationByRelationCached(element.m_id, fetcher); + m_cache->ForEachRelationByRelationCached(element.m_id, wrapper); it = base::FindIf(elements, [&](auto const & idRelation) { return idRelation.second.GetRelationRole(element.m_id) == "outline"; }); @@ -128,8 +126,8 @@ base::GeoObjectId BuildingPartsCollector::FindTopRelation(OsmElement const & ele return it != std::end(elements) ? base::MakeOsmRelation(it->first) : base::GeoObjectId(); } -void BuildingPartsCollector::WritePair(CompositeId const & id, - std::vector const & buildingParts) +void BuildingPartsCollector::WriteBuildingParts(CompositeId const & id, + std::vector const & buildingParts) { WriteToSink(m_writer, id.m_mainId.GetEncodedId()); WriteToSink(m_writer, id.m_additionalId.GetEncodedId()); @@ -166,16 +164,18 @@ BuildingToBuildingPartsMap::BuildingToBuildingPartsMap(std::string const & filen while (src.Size() > 0) { base::GeoObjectId const mainId(ReadPrimitiveFromSource(src)); - CompositeId const outlineId(mainId, base::GeoObjectId(ReadPrimitiveFromSource(src))); + base::GeoObjectId const additionalId(ReadPrimitiveFromSource(src)); + CompositeId const outlineId(mainId, additionalId); std::vector buildingParts; auto contSize = ReadVarUint(src); buildingParts.reserve(contSize); while (contSize--) { base::GeoObjectId const id(ReadPrimitiveFromSource(src)); - m_buildingParts.emplace_back(id); buildingParts.emplace_back(id); } + m_buildingParts.insert(std::end(m_buildingParts), + std::begin(buildingParts), std::end(buildingParts)); m_outlineToBuildingPart.emplace_back(outlineId, std::move(buildingParts)); } @@ -190,7 +190,7 @@ bool BuildingToBuildingPartsMap::HasBuildingPart(base::GeoObjectId const & id) return std::binary_search(std::cbegin(m_buildingParts), std::cend(m_buildingParts), id); } -std::vector const & BuildingToBuildingPartsMap::GetBuildingPartByOutlineId( +std::vector const & BuildingToBuildingPartsMap::GetBuildingPartsByOutlineId( CompositeId const & id) { auto const it = diff --git a/generator/collector_building_parts.hpp b/generator/collector_building_parts.hpp index 35fba0f5fb..614a174690 100644 --- a/generator/collector_building_parts.hpp +++ b/generator/collector_building_parts.hpp @@ -48,7 +48,7 @@ public: private: base::GeoObjectId FindTopRelation(OsmElement const & element); std::vector FindAllBuildingParts(base::GeoObjectId const & id); - void WritePair(CompositeId const & id, std::vector const & buildingParts); + void WriteBuildingParts(CompositeId const & id, std::vector const & buildingParts); std::shared_ptr m_cache; std::unique_ptr m_writer; @@ -60,7 +60,7 @@ public: explicit BuildingToBuildingPartsMap(std::string const & filename); bool HasBuildingPart(base::GeoObjectId const & id); - std::vector const & GetBuildingPartByOutlineId(CompositeId const & id); + std::vector const & GetBuildingPartsByOutlineId(CompositeId const & id); private: std::vector m_buildingParts; diff --git a/generator/collector_interface.hpp b/generator/collector_interface.hpp index fc35c20c5e..5434687300 100644 --- a/generator/collector_interface.hpp +++ b/generator/collector_interface.hpp @@ -44,6 +44,7 @@ class CityAreaCollector; class CrossMwmOsmWaysCollector; class RoutingCityBoundariesCollector; class BuildingPartsCollector; + namespace cache { class IntermediateDataReaderInterface; diff --git a/generator/complex_generator/complex_generator.cpp b/generator/complex_generator/complex_generator.cpp index f95a4f4b93..630f2566bd 100644 --- a/generator/complex_generator/complex_generator.cpp +++ b/generator/complex_generator/complex_generator.cpp @@ -54,7 +54,7 @@ DEFINE_string(maps_build_path, "", "Directory of any of the previous map generations. It is assumed that it will " "contain a directory with mwm(for example 190423) and a directory with mappings from " "osm is to a feature id."); -DEFINE_bool(popularity, false, "Build complexes for caclulation of popularity of objects."); +DEFINE_bool(popularity, false, "Build complexes for calculation of popularity of objects."); DEFINE_string(output, "", "Output filename"); DEFINE_bool(debug, false, "Debug mode."); @@ -119,7 +119,7 @@ MAIN_WITH_ERROR_HANDLING([](int argc, char ** argv) { // Directory FLAGS_maps_build_path must contain 'osm2ft' directory with *.mwm.osm2ft auto const osm2FtPath = base::JoinPath(FLAGS_maps_build_path, "osm2ft"); // Find directory with *.mwm. Directory FLAGS_maps_build_path must contain directory with *.mwm, - // which name must contain six digits. + // whose name must consist of six digits. Platform::FilesList files; pl.GetFilesByRegExp(FLAGS_maps_build_path, "[0-9]{6}", files); CHECK_EQUAL(files.size(), 1, ()); diff --git a/generator/filter_complex.hpp b/generator/filter_complex.hpp index 3b47d2f733..bc34ca0f79 100644 --- a/generator/filter_complex.hpp +++ b/generator/filter_complex.hpp @@ -6,7 +6,7 @@ namespace generator { -// The filter will leave only elements for complex. +// The filter will leave only elements for complexes. class FilterComplex : public FilterInterface { public: diff --git a/generator/final_processor_intermediate_mwm.cpp b/generator/final_processor_intermediate_mwm.cpp index bd7a904407..7c0683f27d 100644 --- a/generator/final_processor_intermediate_mwm.cpp +++ b/generator/final_processor_intermediate_mwm.cpp @@ -603,10 +603,10 @@ void ComplexFinalProcessor::UseCentersEnricher(std::string const & mwmPath, m_osm2ftPath = osm2ftPath; } -std::unique_ptr ComplexFinalProcessor::CreateEnricher( +std::unique_ptr ComplexFinalProcessor::CreateEnricher( std::string const & countryName) const { - return std::make_unique( + return std::make_unique( base::JoinPath(m_osm2ftPath, countryName + DATA_FILE_EXTENSION OSM2FEATURE_FILE_EXTENSION), base::JoinPath(m_mwmPath, countryName + DATA_FILE_EXTENSION)); } @@ -628,10 +628,10 @@ void ComplexFinalProcessor::Process() auto countryName = filename; strings::ReplaceLast(countryName, DATA_FILE_EXTENSION_TMP, ""); // https://wiki.openstreetmap.org/wiki/Simple_3D_buildings - // An object with tag 'building:part' is a part of a releation with tag 'building' or - // contained in a object with tag 'building'. This case is first. We need to remove objects - // with tag 'building:part', which were saved by the collector. Then we will add them - // separately. + // An object with tag 'building:part' is a part of a relation with outline 'building' or + // is contained in an object with tag 'building'. We will split data and work with + // these cases separately. First of all let's remove objects with tag building:part is contained + // in relations. We will add them back after data processing. std::unordered_map relationBuildingParts; if (m_buildingToParts) relationBuildingParts = RemoveRelationBuildingParts(filename); @@ -640,7 +640,14 @@ void ComplexFinalProcessor::Process() hierarchy::HierarchyBuilder builder(base::JoinPath(m_mwmTmpPath, filename)); builder.SetGetMainTypeFunction(m_getMainType); builder.SetFilter(m_filter); - auto nodes = builder.Build(); + auto trees = builder.Build(); + + // We remove tree roots with tag 'building:part'. + base::EraseIf(trees, [](auto const & node) { + static auto const & buildingPartChecker = ftypes::IsBuildingPartChecker::Instance(); + return buildingPartChecker(node->GetData().GetTypes()); + }); + // We want to transform // building // |_building-part @@ -649,19 +656,20 @@ void ComplexFinalProcessor::Process() // building // |_building-part // |_building-part - FlattenBuildingParts(nodes); + FlattenBuildingParts(trees); // In the end we add objects, which were saved by the collector. if (m_buildingToParts) - AddRelationBuildingParts(nodes, relationBuildingParts); + AddRelationBuildingParts(trees, relationBuildingParts); - hierarchy::HierarchyLinesBuilder linesBuilder(std::move(nodes)); + // We create and save hierarchy lines. + hierarchy::HierarchyLinesBuilder hierarchyBuilder(std::move(trees)); if (m_useCentersEnricher) - linesBuilder.SetHierarchyLineEnricher(CreateEnricher(countryName)); + hierarchyBuilder.SetHierarchyEntryEnricher(CreateEnricher(countryName)); - linesBuilder.SetCountry(countryName); - linesBuilder.SetGetMainTypeFunction(m_getMainType); - linesBuilder.SetGetNameFunction(m_getName); - return linesBuilder.GetHierarchyLines(); + hierarchyBuilder.SetCountry(countryName); + hierarchyBuilder.SetGetMainTypeFunction(m_getMainType); + hierarchyBuilder.SetGetNameFunction(m_getName); + return hierarchyBuilder.GetHierarchyLines(); }); futures.emplace_back(std::move(future)); }); @@ -708,7 +716,7 @@ void ComplexFinalProcessor::AddRelationBuildingParts( tree_node::PostOrderVisit(node, [&](auto const & n) { auto const buildingId = n->GetData().GetCompositeId(); - auto const & ids = m_buildingToParts->GetBuildingPartByOutlineId(buildingId); + auto const & ids = m_buildingToParts->GetBuildingPartsByOutlineId(buildingId); for (auto const & id : ids) { if (m.count(id) == 0) @@ -731,30 +739,33 @@ void ComplexFinalProcessor::FlattenBuildingParts(hierarchy::HierarchyBuilder::No if (node->HasParent()) continue; - static auto const & buildingPartChecker = ftypes::IsBuildingPartChecker::Instance(); - if (buildingPartChecker(node->GetData().GetTypes())) - continue; - std::vector< std::pair> buildingPartsTrees; - while (true) - { - auto buildingPartsTree = tree_node::FindIf( - node, [](auto const & data) { return buildingPartChecker(data.GetTypes()); }); - if (!buildingPartsTree) - break; + static auto const & buildingPartChecker = ftypes::IsBuildingPartChecker::Instance(); + std::function visit; + visit = [&](auto const & nd) { + if (buildingPartChecker(nd->GetData().GetTypes())) + { + CHECK(nd->HasParent(), ()); + auto building = nd->GetParent(); + buildingPartsTrees.emplace_back(building, nd); + return; + } - CHECK(buildingPartsTree->HasParent(), ()); - auto building = buildingPartsTree->GetParent(); - Unlink(buildingPartsTree, building); - buildingPartsTrees.emplace_back(building, buildingPartsTree); - } + CHECK(!buildingPartChecker(node->GetData().GetTypes()), ()); + for (auto const & ch : nd->GetChildren()) + visit(ch); + }; + + visit(node); for (auto const & buildingAndParts : buildingPartsTrees) { + Unlink(buildingAndParts.second, buildingAndParts.first); tree_node::PostOrderVisit(buildingAndParts.second, [&](auto const & buildingPartNode) { + CHECK(buildingPartChecker(buildingPartNode->GetData().GetTypes()), ()); buildingPartNode->RemoveChildren(); tree_node::Link(buildingPartNode, buildingAndParts.first); }); diff --git a/generator/final_processor_intermediate_mwm.hpp b/generator/final_processor_intermediate_mwm.hpp index 365b5de6e8..0d6cffca70 100644 --- a/generator/final_processor_intermediate_mwm.hpp +++ b/generator/final_processor_intermediate_mwm.hpp @@ -160,7 +160,7 @@ public: private: static void FlattenBuildingParts(hierarchy::HierarchyBuilder::Node::Ptrs & nodes); - std::unique_ptr CreateEnricher( + std::unique_ptr CreateEnricher( std::string const & countryName) const; void WriteLines(std::vector const & lines); std::unordered_map RemoveRelationBuildingParts( @@ -173,7 +173,7 @@ private: hierarchy::PrintFn m_printFunction; hierarchy::GetNameFn m_getName; std::shared_ptr m_filter; - std::unique_ptr m_buildingToBuildingParts; + std::unique_ptr m_buildingToParts; bool m_useCentersEnricher = false; std::string m_mwmTmpPath; std::string m_outFilename; diff --git a/generator/hierarchy.cpp b/generator/hierarchy.cpp index f9842a9d8e..20d36dc787 100644 --- a/generator/hierarchy.cpp +++ b/generator/hierarchy.cpp @@ -5,6 +5,8 @@ #include "geometry/mercator.hpp" #include "geometry/rect2d.hpp" +#include "base/assert.hpp" + #include #include #include @@ -13,7 +15,6 @@ #include #include #include -#include "base/assert.hpp" #include #include @@ -107,9 +108,9 @@ HierarchyLinker::Node::Ptr HierarchyLinker::FindPlaceParent(HierarchyPlace const auto const point = place.GetCenter(); m_tree.ForEachInRect({point, point}, [&](auto const & candidateNode) { // https://wiki.openstreetmap.org/wiki/Simple_3D_buildings - // An object with tag 'building:part' is a part of a releation with tag 'building' or contained - // in a object with tag 'building'. This case is second. We suppose a building part is only - // inside a building. + // An object with tag 'building:part' is a part of a relation with outline 'building' or + // is contained in a object with tag 'building'. This case is second. We suppose a building part is + // only inside a building. static auto const & buildingChecker = ftypes::IsBuildingChecker::Instance(); static auto const & buildingPartChecker = ftypes::IsBuildingPartChecker::Instance(); auto const & candidate = candidateNode->GetData(); @@ -195,17 +196,22 @@ HierarchyBuilder::Node::Ptrs HierarchyBuilder::Build() places.reserve(fbs.size()); std::transform(std::cbegin(fbs), std::cend(fbs), std::back_inserter(places), [](auto const & fb) { return std::make_shared(HierarchyPlace(fb)); }); - return HierarchyLinker(std::move(places)).Link(); + auto nodes = HierarchyLinker(std::move(places)).Link(); + // We leave only the trees. + base::EraseIf(nodes, [](auto const & node) { + return node->HasParent(); + }); + return nodes; } -HierarchyLineEnricher::HierarchyLineEnricher(std::string const & osm2FtIdsPath, +HierarchyEntryEnricher::HierarchyEntryEnricher(std::string const & osm2FtIdsPath, std::string const & countryFullPath) : m_featureGetter(countryFullPath) { CHECK(m_osm2FtIds.ReadFromFile(osm2FtIdsPath), (osm2FtIdsPath)); } -boost::optional HierarchyLineEnricher::GetFeatureCenter(CompositeId const & id) const +boost::optional HierarchyEntryEnricher::GetFeatureCenter(CompositeId const & id) const { auto const optIds = m_osm2FtIds.GetFeatureIds(id); if (optIds.empty()) @@ -227,9 +233,10 @@ boost::optional HierarchyLineEnricher::GetFeatureCenter(CompositeId (id, optIds)); } - for (auto type : - {base::Underlying(feature::GeomType::Point), base::Underlying(feature::GeomType::Area), - base::Underlying(feature::GeomType::Line)}) + for (auto type : { + base::Underlying(feature::GeomType::Point), + base::Underlying(feature::GeomType::Area), + base::Underlying(feature::GeomType::Line)}) { if (m.count(type) != 0) return m[type]; @@ -238,8 +245,8 @@ boost::optional HierarchyLineEnricher::GetFeatureCenter(CompositeId return {}; } -HierarchyLinesBuilder::HierarchyLinesBuilder(HierarchyBuilder::Node::Ptrs && nodes) - : m_nodes(std::move(nodes)) +HierarchyLinesBuilder::HierarchyLinesBuilder(HierarchyBuilder::Node::Ptrs && trees) + : m_trees(std::move(trees)) { } @@ -255,8 +262,8 @@ void HierarchyLinesBuilder::SetCountry(storage::CountryId const & country) m_countryName = country; } -void HierarchyLinesBuilder::SetHierarchyLineEnricher( - std::unique_ptr && enricher) +void HierarchyLinesBuilder::SetHierarchyEntryEnricher( + std::unique_ptr && enricher) { m_enricher = std::move(enricher); } @@ -267,9 +274,13 @@ std::vector HierarchyLinesBuilder::GetHierarchyLines() CHECK(m_getMainType, ()); std::vector lines; - lines.reserve(m_nodes.size()); - std::transform(std::cbegin(m_nodes), std::cend(m_nodes), std::back_inserter(lines), - [&](auto const & n) { return Transform(n); }); + for (auto const & tree : m_trees) + { + tree_node::PreOrderVisit(tree, [&](auto const & node) { + lines.emplace_back(Transform(node)); + }); + } + return lines; } diff --git a/generator/hierarchy.hpp b/generator/hierarchy.hpp index edf40a17de..93a816402b 100644 --- a/generator/hierarchy.hpp +++ b/generator/hierarchy.hpp @@ -117,10 +117,10 @@ protected: std::shared_ptr m_filter; }; -class HierarchyLineEnricher +class HierarchyEntryEnricher { public: - HierarchyLineEnricher(std::string const & osm2FtIdsPath, std::string const & countryFullPath); + HierarchyEntryEnricher(std::string const & osm2FtIdsPath, std::string const & countryFullPath); boost::optional GetFeatureCenter(CompositeId const & id) const; @@ -132,12 +132,12 @@ private: class HierarchyLinesBuilder { public: - HierarchyLinesBuilder(HierarchyBuilder::Node::Ptrs && nodes); + HierarchyLinesBuilder(HierarchyBuilder::Node::Ptrs && trees); void SetGetMainTypeFunction(GetMainTypeFn const & getMainType); void SetGetNameFunction(GetNameFn const & getName); void SetCountry(storage::CountryId const & country); - void SetHierarchyLineEnricher(std::unique_ptr && enricher); + void SetHierarchyEntryEnricher(std::unique_ptr && enricher); std::vector GetHierarchyLines(); @@ -145,11 +145,11 @@ private: m2::PointD GetCenter(HierarchyBuilder::Node::Ptr const & node); HierarchyEntry Transform(HierarchyBuilder::Node::Ptr const & node); - HierarchyBuilder::Node::Ptrs m_nodes; + HierarchyBuilder::Node::Ptrs m_trees; GetMainTypeFn m_getMainType; GetNameFn m_getName; storage::CountryId m_countryName; - std::unique_ptr m_enricher; + std::unique_ptr m_enricher; }; } // namespace hierarchy } // namespace generator diff --git a/generator/intermediate_data.hpp b/generator/intermediate_data.hpp index 9e8c2f6618..ce85e84aba 100644 --- a/generator/intermediate_data.hpp +++ b/generator/intermediate_data.hpp @@ -216,7 +216,7 @@ public: // TODO |GetNode()|, |lat|, |lon| are used as y, x in real. bool GetNode(Key id, double & lat, double & lon) const override { - return m_nodes.GetPoint(id, lat, lon); + bool GetRalation(Key id, RelationElement & e) { return m_relations.Read(id, e); } } bool GetWay(Key id, WayElement & e) override { return m_ways.Read(id, e); } diff --git a/indexer/complex/tree_node.hpp b/indexer/complex/tree_node.hpp index e3cb145bf7..18a18ec223 100644 --- a/indexer/complex/tree_node.hpp +++ b/indexer/complex/tree_node.hpp @@ -121,17 +121,17 @@ template void PostOrderVisit(types::Ptr const & node, Fn && fn) { base::ControlFlowWrapper wrapper(std::forward(fn)); - std::function const &)> preOrderVisitDetail; - preOrderVisitDetail = [&](auto const & node) { + std::function const &)> postOrderVisitDetail; + postOrderVisitDetail = [&](auto const & node) { for (auto const & ch : node->GetChildren()) { - if (preOrderVisitDetail(ch) == base::ControlFlow::Break) + if (postOrderVisitDetail(ch) == base::ControlFlow::Break) return base::ControlFlow::Break; } return wrapper(node); }; - preOrderVisitDetail(node); + postOrderVisitDetail(node); } template