diff --git a/coding/file_reader.cpp b/coding/file_reader.cpp index d6bd13b3d0..6092da171f 100644 --- a/coding/file_reader.cpp +++ b/coding/file_reader.cpp @@ -14,9 +14,9 @@ #endif // static -uint32_t const FileReader::kDefaultLogPageSize = 10; +uint32_t const FileReader::kDefaultLogPageSize = 10; // page size is 2^10 = 1024 = 1kb // static -uint32_t const FileReader::kDefaultLogPageCount = 4; +uint32_t const FileReader::kDefaultLogPageCount = 4; // page count is 2^4 = 16, i.e. 16 pages are cached class FileReader::FileReaderData { diff --git a/data/replaced_tags.txt b/data/replaced_tags.txt index 08f84eb583..b4916985c2 100644 --- a/data/replaced_tags.txt +++ b/data/replaced_tags.txt @@ -7,6 +7,8 @@ # means that all shop=ice_cream tags will be converted into amenity=ice_cream during maps generation, # so there will be no shop=ice_cream in MWM map files at all. # +# These tag conversions happen before matching of the mapcss-mapping.csv rules. +# # In the following example # vending=parcel_pickup : amenity=parcel_locker | u # the 'u' flag means update/replace existing tag with the same key (if any), @@ -24,6 +26,8 @@ # ::= string without spaces, '=', ',', '|' symbols # ::= string without spaces, '=', ',', '|' symbols # ::= control flag {'u'} +# +# For implementation details see the TagReplacer class in generator/tag_admixer.hpp. atm=yes : amenity=atm bench=yes : amenity=bench diff --git a/drape_frontend/rule_drawer.cpp b/drape_frontend/rule_drawer.cpp index ad5e91e5ae..cb8a7ebfd6 100644 --- a/drape_frontend/rule_drawer.cpp +++ b/drape_frontend/rule_drawer.cpp @@ -302,12 +302,12 @@ void RuleDrawer::ProcessAreaStyle(FeatureType & f, Stylist const & s, TInsertSha ApplyAreaFeature apply(m_context->GetTileKey(), insertShape, f.GetID(), m_currentScaleGtoP, isBuilding, - m_context->Is3dBuildingsEnabled() && isBuildingOutline, - areaMinHeight, areaHeight, f.GetRank(), + m_context->Is3dBuildingsEnabled() && isBuildingOutline /* skipAreaGeometry */, + areaMinHeight /* minPosZ */, areaHeight /* posZ */, f.GetRank(), s.GetCaptionDescription()); f.ForEachTriangle(apply, zoomLevel); if (applyPointStyle) - apply(featureCenter, true /* hasArea */); + apply(featureCenter, true /* hasArea */); // ApplyPointFeature::operator()() if (CheckCancelled()) return; @@ -336,7 +336,7 @@ void RuleDrawer::ProcessAreaStyle(FeatureType & f, Stylist const & s, TInsertSha /// @todo Can we put this check in the beginning of this function? if (!IsDiscardCustomFeature(f.GetID())) - apply.Finish(m_context->GetTextureManager()); + apply.Finish(m_context->GetTextureManager()); // ApplyPointFeature::Finish() } void RuleDrawer::ProcessLineStyle(FeatureType & f, Stylist const & s, TInsertShapeFn const & insertShape) diff --git a/generator/feature_maker.hpp b/generator/feature_maker.hpp index 42957d6d05..5ad779cbc8 100644 --- a/generator/feature_maker.hpp +++ b/generator/feature_maker.hpp @@ -8,40 +8,33 @@ struct OsmElement; namespace generator { -// Class FeatureMakerSimple is class FeatureMakerBase implementation for simple features building. -// It is only trying to build a feature and does not filter features in any way, -// except for bad geometry. This class is suitable for most cases. +// FeatureMakerSimple is suitable for most cases for simple features. +// It filters features for bad geometry only. class FeatureMakerSimple: public FeatureMakerBase { public: using FeatureMakerBase::FeatureMakerBase; - // FeatureMaker overrides: std::shared_ptr Clone() const override; protected: - // FeatureMaker overrides: void ParseParams(FeatureBuilderParams & params, OsmElement & element) const override; private: - // FeatureMaker overrides: bool BuildFromNode(OsmElement & element, FeatureBuilderParams const & params) override; bool BuildFromWay(OsmElement & element, FeatureBuilderParams const & params) override; bool BuildFromRelation(OsmElement & element, FeatureBuilderParams const & params) override; }; -// The difference between class FeatureMakerSimple and class FeatureMaker is that -// class FeatureMaker processes the types more strictly. +// FeatureMaker additionally filters the types using feature::IsUsefulType. class FeatureMaker : public FeatureMakerSimple { public: using FeatureMakerSimple::FeatureMakerSimple; - // FeatureMaker overrides: std::shared_ptr Clone() const override; private: - // FeatureMaker overrides: void ParseParams(FeatureBuilderParams & params, OsmElement & element) const override; }; } // namespace generator diff --git a/generator/feature_processing_layers.cpp b/generator/feature_processing_layers.cpp index b151e5090b..81d30c5d22 100644 --- a/generator/feature_processing_layers.cpp +++ b/generator/feature_processing_layers.cpp @@ -98,13 +98,21 @@ void RepresentationLayer::Handle(FeatureBuilder & fb) { case feature::GeomType::Area: { + // All closed geometry OsmWays fall through here. + // E.g. a closed way with tags [amenity=restaurant][wifi=yes][cuisine=*] + // should be added as an areal feature with corresponding types + // and no extra linear/point features. + // A closed way with a tag [highway=pedestrian] should be added as a line feature only + // (because there are no areal and/or point drules for the highway-pedestrian type). + // But a closed way tagged [highway=pedestrian][area=yes] should be added as both area and line features + // (because there are areal and line drules for the highway-primary-area type). + // Also an [leisure=playground][barrier=fence] should be added + // as an areal amenity-playground and a linear barrier=fence. + + // Try to create an areal object first (if there are corresponding drules), + // otherwise try to create a point object. HandleArea(fb, params); // CanBeLine ignores exceptional types from TypeAlwaysExists / IsUsefulNondrawableType. - // Areal object with types amenity=restaurant + wifi=yes + cuisine=* should be added as areal object with - // these types and no extra linear/point objects. - // We need extra line object only for case when object has type which is drawable like line: - // amenity=playground + barrier=fence should be added as area object with amenity=playground + linear object - // with barrier=fence. if (CanBeLine(params)) { auto featureLine = MakeLine(fb); @@ -270,7 +278,7 @@ void ComplexFeaturesMixer::Process(std::function m_func; }; @@ -312,6 +315,8 @@ private: buffer_vector(Type::Count)> m_types; }; +// Removes types that are prefixes of another longer type, +// e.g. highway-primary-bridge is left while highway-primary is removed. void LeaveLongestTypes(vector & matchedTypes) { auto const less = [](auto const & lhs, auto const & rhs) { return lhs > rhs; }; @@ -912,6 +917,9 @@ void PostprocessElement(OsmElement * p, FeatureBuilderParams & params) void GetNameAndType(OsmElement * p, FeatureBuilderParams & params, function const & filterType) { + // At this point, some preprocessing could've been done to the tags already + // in TranslatorInterface::Preprocess(), e.g. converting tags according to replaced_tags.txt. + // Stage1: Preprocess tags. PreprocessElement(p); diff --git a/generator/translator.cpp b/generator/translator.cpp index 01c871c3fe..1c39e55e00 100644 --- a/generator/translator.cpp +++ b/generator/translator.cpp @@ -42,13 +42,13 @@ void Translator::SetFilter(std::shared_ptr const & filter) { m_ void Translator::Emit(OsmElement & element) { - Preprocess(element); + Preprocess(element); // Might use replaced_tags.txt via a TagReplacer. if (!m_filter->IsAccepted(element)) return; m_tagsEnricher(element); m_collector->Collect(element); - m_featureMaker->Add(element); + m_featureMaker->Add(element); // A feature is created from OSM tags. FeatureBuilder feature; while (m_featureMaker->GetNextFeature(feature)) { diff --git a/indexer/classificator.cpp b/indexer/classificator.cpp index 67c139f5c9..5a1b5902b7 100644 --- a/indexer/classificator.cpp +++ b/indexer/classificator.cpp @@ -234,11 +234,12 @@ namespace void add_rule(int ft, iter_t i) { + // Define which drule types are applicable to which feature geom types. static const int visible[3][drule::count_of_rules] = { - //{ line, area, symbol, caption, circle, pathtext, waymarker, shield } + //{ line, area, symbol, caption, circle, pathtext, waymarker, shield }, see drule::Key::rule_type_t { 0, 0, 1, 1, 1, 0, 0, 0 }, // fpoint { 1, 0, 0, 0, 0, 1, 0, 1 }, // fline - { 1, 1, 1, 1, 1, 0, 0, 0 } // farea + { 1, 1, 1, 1, 1, 0, 0, 0 } // farea (!!! different from IsDrawableLike(): here area feature can use point and line styles) }; if (visible[ft][i->m_type] == 1) @@ -291,11 +292,12 @@ bool ClassifObject::IsDrawableLike(feature::GeomType gt, bool emptyName) const if (!IsDrawableAny()) return false; + // Define which feature geom types can use which drule types for rendering. static const int visible[3][drule::count_of_rules] = { - //{ line, area, symbol, caption, circle, pathtext, waymarker, shield } + //{ line, area, symbol, caption, circle, pathtext, waymarker, shield }, see drule::Key::rule_type_t {0, 0, 1, 1, 1, 0, 0, 0}, // fpoint {1, 0, 0, 0, 0, 1, 0, 1}, // fline - {0, 1, 0, 0, 0, 0, 0, 0} // farea (!!! key difference with GetSuitable !!!) + {0, 1, 0, 0, 0, 0, 0, 0} // farea (!!! key difference with GetSuitable, see suitable_getter::add_rule()) }; for (auto const & k : m_drawRules) diff --git a/indexer/feature.cpp b/indexer/feature.cpp index 54649723b5..dfc0afe6ca 100644 --- a/indexer/feature.cpp +++ b/indexer/feature.cpp @@ -195,7 +195,7 @@ FeatureType::FeatureType(SharedLoadInfo const * loadInfo, vector && buf { CHECK(m_loadInfo, ()); - m_header = Header(m_data); + m_header = Header(m_data); // Parse the header and optional name/layer/addinfo. } std::unique_ptr FeatureType::CreateFromMapObject(osm::MapObject const & emo) diff --git a/indexer/feature_data.cpp b/indexer/feature_data.cpp index 2294222fe2..97c0b7bb26 100644 --- a/indexer/feature_data.cpp +++ b/indexer/feature_data.cpp @@ -188,6 +188,7 @@ uint8_t CalculateHeader(size_t const typesCount, HeaderGeomType const headerGeom FeatureParamsBase const & params) { ASSERT(typesCount != 0, ("Feature should have at least one type.")); + ASSERT_LESS_OR_EQUAL(typesCount, kMaxTypesCount, ()); uint8_t header = static_cast(typesCount - 1); if (!params.name.IsEmpty()) @@ -427,13 +428,14 @@ bool FeatureParams::FinishAddingTypes() { UselessTypesChecker::Instance().SortUselessToEnd(m_types); - LOG(LWARNING, ("Exceeded max types count:", TypesToString(m_types))); + LOG(LWARNING, ("Exceeded max types count, got total", m_types.size(), ":", TypesToString(m_types))); m_types.resize(kMaxTypesCount); sort(m_types.begin(), m_types.end()); } // Patch fix that removes house number from localities. + /// @todo move this fix elsewhere (osm2type.cpp?) if (!house.IsEmpty() && ftypes::IsLocalityChecker::Instance()(m_types)) { LOG(LWARNING, ("Locality with house number", *this)); diff --git a/indexer/feature_data.hpp b/indexer/feature_data.hpp index b3fbaf7be7..8041192bcb 100644 --- a/indexer/feature_data.hpp +++ b/indexer/feature_data.hpp @@ -41,7 +41,7 @@ namespace feature PointEx = 3U << 5 /// point feature (addinfo = house) }; - static constexpr int kMaxTypesCount = HEADER_MASK_TYPE + 1; + static constexpr int kMaxTypesCount = HEADER_MASK_TYPE + 1; // 8, because there should be no features with 0 types enum Layer : int8_t { diff --git a/platform/constants.hpp b/platform/constants.hpp index 9ce6b7cdbb..f784487701 100644 --- a/platform/constants.hpp +++ b/platform/constants.hpp @@ -2,5 +2,5 @@ #include -constexpr uint32_t READER_CHUNK_LOG_SIZE = 10; -constexpr uint32_t READER_CHUNK_LOG_COUNT = 12; +constexpr uint32_t READER_CHUNK_LOG_SIZE = 10; // 1024 bytes pages +constexpr uint32_t READER_CHUNK_LOG_COUNT = 12; // 2^12 = 4096 pages count, hence 4M size cache overall (for data files only)