From 14e8ade9a91ea48cb65b3e14954e13a396e1a94d Mon Sep 17 00:00:00 2001 From: Ryan Mitchell Date: Mon, 11 Jan 2021 16:01:35 -0800 Subject: Optimize FilterApkAssets by caching config ResTable_config of every ResTable_type is read from device every time AssetManager::RebuildFilterList is invoked. For large APKs (like framework-res.apk), this causes a large number of page faults when accessing the config from disk. The configs are also used in the slow path of AssetManager::FindEntryInternal, which makes it even slower. Instead cache the config on the TypeSpec of its ApkAsset. Bug: 177247024 Test: libandroidfw_tests Change-Id: I66d507c4eeb2399f7558f3d9dfc53c157129ada0 --- libs/androidfw/LoadedArsc.cpp | 94 ++++++++++++++----------------------------- 1 file changed, 30 insertions(+), 64 deletions(-) (limited to 'libs/androidfw/LoadedArsc.cpp') diff --git a/libs/androidfw/LoadedArsc.cpp b/libs/androidfw/LoadedArsc.cpp index 2fc3b05011c2..996b42426282 100644 --- a/libs/androidfw/LoadedArsc.cpp +++ b/libs/androidfw/LoadedArsc.cpp @@ -33,7 +33,6 @@ #endif #endif -#include "androidfw/ByteBucketArray.h" #include "androidfw/Chunk.h" #include "androidfw/ResourceUtils.h" #include "androidfw/Util.h" @@ -49,36 +48,24 @@ namespace { // Builder that helps accumulate Type structs and then create a single // contiguous block of memory to store both the TypeSpec struct and // the Type structs. -class TypeSpecPtrBuilder { - public: - explicit TypeSpecPtrBuilder(incfs::verified_map_ptr header) - : header_(header) { - } +struct TypeSpecBuilder { + explicit TypeSpecBuilder(incfs::verified_map_ptr header) : header_(header) {} void AddType(incfs::verified_map_ptr type) { - types_.push_back(type); + TypeSpec::TypeEntry& entry = type_entries.emplace_back(); + entry.config.copyFromDtoH(type->config); + entry.type = type; } - TypeSpecPtr Build() { - // Check for overflow. - using ElementType = incfs::verified_map_ptr; - if ((std::numeric_limits::max() - sizeof(TypeSpec)) / sizeof(ElementType) < - types_.size()) { - return {}; - } - TypeSpec* type_spec = - (TypeSpec*)::malloc(sizeof(TypeSpec) + (types_.size() * sizeof(ElementType))); - type_spec->type_spec = header_; - type_spec->type_count = types_.size(); - memcpy(type_spec + 1, types_.data(), types_.size() * sizeof(ElementType)); - return TypeSpecPtr(type_spec); + TypeSpec Build() { + return {header_, std::move(type_entries)}; } private: - DISALLOW_COPY_AND_ASSIGN(TypeSpecPtrBuilder); + DISALLOW_COPY_AND_ASSIGN(TypeSpecBuilder); incfs::verified_map_ptr header_; - std::vector> types_; + std::vector type_entries; }; } // namespace @@ -322,15 +309,10 @@ base::expected, NullOrIOError> LoadedPackage::Get } base::expected LoadedPackage::CollectConfigurations( - bool exclude_mipmap, std::set* out_configs) const { - const size_t type_count = type_specs_.size(); - for (size_t i = 0; i < type_count; i++) { - const TypeSpecPtr& type_spec = type_specs_[i]; - if (type_spec == nullptr) { - continue; - } + bool exclude_mipmap, std::set* out_configs) const {\ + for (const auto& type_spec : type_specs_) { if (exclude_mipmap) { - const int type_idx = type_spec->type_spec->id - 1; + const int type_idx = type_spec.first - 1; const auto type_name16 = type_string_pool_.stringAt(type_idx); if (UNLIKELY(IsIOError(type_name16))) { return base::unexpected(GetIOError(type_name16.error())); @@ -354,11 +336,8 @@ base::expected LoadedPackage::CollectConfigurations( } } - const auto iter_end = type_spec->types + type_spec->type_count; - for (auto iter = type_spec->types; iter != iter_end; ++iter) { - ResTable_config config; - config.copyFromDtoH((*iter)->config); - out_configs->insert(config); + for (const auto& type_entry : type_spec.second.type_entries) { + out_configs->insert(type_entry.config); } } return {}; @@ -366,19 +345,12 @@ base::expected LoadedPackage::CollectConfigurations( void LoadedPackage::CollectLocales(bool canonicalize, std::set* out_locales) const { char temp_locale[RESTABLE_MAX_LOCALE_LEN]; - const size_t type_count = type_specs_.size(); - for (size_t i = 0; i < type_count; i++) { - const TypeSpecPtr& type_spec = type_specs_[i]; - if (type_spec != nullptr) { - const auto iter_end = type_spec->types + type_spec->type_count; - for (auto iter = type_spec->types; iter != iter_end; ++iter) { - ResTable_config configuration; - configuration.copyFromDtoH((*iter)->config); - if (configuration.locale != 0) { - configuration.getBcp47Locale(temp_locale, canonicalize); - std::string locale(temp_locale); - out_locales->insert(std::move(locale)); - } + for (const auto& type_spec : type_specs_) { + for (const auto& type_entry : type_spec.second.type_entries) { + if (type_entry.config.locale != 0) { + type_entry.config.getBcp47Locale(temp_locale, canonicalize); + std::string locale(temp_locale); + out_locales->insert(std::move(locale)); } } } @@ -398,14 +370,13 @@ base::expected LoadedPackage::FindEntryByName( return base::unexpected(key_idx.error()); } - const TypeSpec* type_spec = type_specs_[*type_idx].get(); + const TypeSpec* type_spec = GetTypeSpecByTypeIndex(*type_idx); if (type_spec == nullptr) { return base::unexpected(std::nullopt); } - const auto iter_end = type_spec->types + type_spec->type_count; - for (auto iter = type_spec->types; iter != iter_end; ++iter) { - const incfs::verified_map_ptr& type = *iter; + for (const auto& type_entry : type_spec->type_entries) { + const incfs::verified_map_ptr& type = type_entry.type; size_t entry_count = dtohl(type->entryCount); for (size_t entry_idx = 0; entry_idx < entry_count; entry_idx++) { @@ -492,7 +463,7 @@ std::unique_ptr LoadedPackage::Load(const Chunk& chunk, // A map of TypeSpec builders, each associated with an type index. // We use these to accumulate the set of Types available for a TypeSpec, and later build a single, // contiguous block of memory that holds all the Types together with the TypeSpec. - std::unordered_map> type_builder_map; + std::unordered_map> type_builder_map; ChunkIterator iter(chunk.data_ptr(), chunk.data_size()); while (iter.HasNext()) { @@ -562,9 +533,9 @@ std::unique_ptr LoadedPackage::Load(const Chunk& chunk, return {}; } - std::unique_ptr& builder_ptr = type_builder_map[type_spec->id - 1]; + std::unique_ptr& builder_ptr = type_builder_map[type_spec->id]; if (builder_ptr == nullptr) { - builder_ptr = util::make_unique(type_spec.verified()); + builder_ptr = util::make_unique(type_spec.verified()); loaded_package->resource_ids_.set(type_spec->id, entry_count); } else { LOG(WARNING) << StringPrintf("RES_TABLE_TYPE_SPEC_TYPE already defined for ID %02x", @@ -584,7 +555,7 @@ std::unique_ptr LoadedPackage::Load(const Chunk& chunk, } // Type chunks must be preceded by their TypeSpec chunks. - std::unique_ptr& builder_ptr = type_builder_map[type->id - 1]; + std::unique_ptr& builder_ptr = type_builder_map[type->id]; if (builder_ptr != nullptr) { builder_ptr->AddType(type.verified()); } else { @@ -722,14 +693,9 @@ std::unique_ptr LoadedPackage::Load(const Chunk& chunk, // Flatten and construct the TypeSpecs. for (auto& entry : type_builder_map) { - uint8_t type_idx = static_cast(entry.first); - TypeSpecPtr type_spec_ptr = entry.second->Build(); - if (type_spec_ptr == nullptr) { - LOG(ERROR) << "Too many type configurations, overflow detected."; - return {}; - } - - loaded_package->type_specs_.editItemAt(type_idx) = std::move(type_spec_ptr); + TypeSpec type_spec = entry.second->Build(); + uint8_t type_id = static_cast(entry.first); + loaded_package->type_specs_[type_id] = std::move(type_spec); } return std::move(loaded_package); -- cgit v1.2.3-59-g8ed1b