| /* | 
 |  * Copyright (C) 2016 The Android Open Source Project | 
 |  * | 
 |  * Licensed under the Apache License, Version 2.0 (the "License"); | 
 |  * you may not use this file except in compliance with the License. | 
 |  * You may obtain a copy of the License at | 
 |  * | 
 |  *      http://www.apache.org/licenses/LICENSE-2.0 | 
 |  * | 
 |  * Unless required by applicable law or agreed to in writing, software | 
 |  * distributed under the License is distributed on an "AS IS" BASIS, | 
 |  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
 |  * See the License for the specific language governing permissions and | 
 |  * limitations under the License. | 
 |  */ | 
 |  | 
 | #include "split/TableSplitter.h" | 
 |  | 
 | #include <algorithm> | 
 | #include <map> | 
 | #include <set> | 
 | #include <unordered_set> | 
 | #include <unordered_map> | 
 | #include <vector> | 
 |  | 
 | #include "android-base/logging.h" | 
 |  | 
 | #include "ConfigDescription.h" | 
 | #include "ResourceTable.h" | 
 | #include "util/Util.h" | 
 |  | 
 | namespace aapt { | 
 |  | 
 | using ConfigClaimedMap = std::unordered_map<ResourceConfigValue*, bool>; | 
 | using ConfigDensityGroups = | 
 |     std::map<ConfigDescription, std::vector<ResourceConfigValue*>>; | 
 |  | 
 | static ConfigDescription CopyWithoutDensity(const ConfigDescription& config) { | 
 |   ConfigDescription without_density = config; | 
 |   without_density.density = 0; | 
 |   return without_density; | 
 | } | 
 |  | 
 | /** | 
 |  * Selects values that match exactly the constraints given. | 
 |  */ | 
 | class SplitValueSelector { | 
 |  public: | 
 |   explicit SplitValueSelector(const SplitConstraints& constraints) { | 
 |     for (const ConfigDescription& config : constraints.configs) { | 
 |       if (config.density == 0) { | 
 |         density_independent_configs_.insert(config); | 
 |       } else { | 
 |         density_dependent_config_to_density_map_[CopyWithoutDensity(config)] = | 
 |             config.density; | 
 |       } | 
 |     } | 
 |   } | 
 |  | 
 |   std::vector<ResourceConfigValue*> SelectValues( | 
 |       const ConfigDensityGroups& density_groups, | 
 |       ConfigClaimedMap* claimed_values) { | 
 |     std::vector<ResourceConfigValue*> selected; | 
 |  | 
 |     // Select the regular values. | 
 |     for (auto& entry : *claimed_values) { | 
 |       // Check if the entry has a density. | 
 |       ResourceConfigValue* config_value = entry.first; | 
 |       if (config_value->config.density == 0 && !entry.second) { | 
 |         // This is still available. | 
 |         if (density_independent_configs_.find(config_value->config) != | 
 |             density_independent_configs_.end()) { | 
 |           selected.push_back(config_value); | 
 |  | 
 |           // Mark the entry as taken. | 
 |           entry.second = true; | 
 |         } | 
 |       } | 
 |     } | 
 |  | 
 |     // Now examine the densities | 
 |     for (auto& entry : density_groups) { | 
 |       // We do not care if the value is claimed, since density values can be | 
 |       // in multiple splits. | 
 |       const ConfigDescription& config = entry.first; | 
 |       const std::vector<ResourceConfigValue*>& related_values = entry.second; | 
 |       auto density_value_iter = | 
 |           density_dependent_config_to_density_map_.find(config); | 
 |       if (density_value_iter != | 
 |           density_dependent_config_to_density_map_.end()) { | 
 |         // Select the best one! | 
 |         ConfigDescription target_density = config; | 
 |         target_density.density = density_value_iter->second; | 
 |  | 
 |         ResourceConfigValue* best_value = nullptr; | 
 |         for (ResourceConfigValue* this_value : related_values) { | 
 |           if (!best_value || | 
 |               this_value->config.isBetterThan(best_value->config, | 
 |                                               &target_density)) { | 
 |             best_value = this_value; | 
 |           } | 
 |         } | 
 |         CHECK(best_value != nullptr); | 
 |  | 
 |         // When we select one of these, they are all claimed such that the base | 
 |         // doesn't include any anymore. | 
 |         (*claimed_values)[best_value] = true; | 
 |         selected.push_back(best_value); | 
 |       } | 
 |     } | 
 |     return selected; | 
 |   } | 
 |  | 
 |  private: | 
 |   DISALLOW_COPY_AND_ASSIGN(SplitValueSelector); | 
 |  | 
 |   std::set<ConfigDescription> density_independent_configs_; | 
 |   std::map<ConfigDescription, uint16_t> | 
 |       density_dependent_config_to_density_map_; | 
 | }; | 
 |  | 
 | /** | 
 |  * Marking non-preferred densities as claimed will make sure the base doesn't | 
 |  * include them, | 
 |  * leaving only the preferred density behind. | 
 |  */ | 
 | static void MarkNonPreferredDensitiesAsClaimed( | 
 |     const std::vector<uint16_t>& preferred_densities, const ConfigDensityGroups& density_groups, | 
 |     ConfigClaimedMap* config_claimed_map) { | 
 |   for (auto& entry : density_groups) { | 
 |     const ConfigDescription& config = entry.first; | 
 |     const std::vector<ResourceConfigValue*>& related_values = entry.second; | 
 |  | 
 |     // There can be multiple best values if there are multiple preferred densities. | 
 |     std::unordered_set<ResourceConfigValue*> best_values; | 
 |  | 
 |     // For each preferred density, find the value that is the best. | 
 |     for (uint16_t preferred_density : preferred_densities) { | 
 |       ConfigDescription target_density = config; | 
 |       target_density.density = preferred_density; | 
 |       ResourceConfigValue* best_value = nullptr; | 
 |       for (ResourceConfigValue* this_value : related_values) { | 
 |         if (!best_value || this_value->config.isBetterThan(best_value->config, &target_density)) { | 
 |           best_value = this_value; | 
 |         } | 
 |       } | 
 |       CHECK(best_value != nullptr); | 
 |       best_values.insert(best_value); | 
 |     } | 
 |  | 
 |     // Claim all the values that aren't the best so that they will be removed from the base. | 
 |     for (ResourceConfigValue* this_value : related_values) { | 
 |       if (best_values.find(this_value) == best_values.end()) { | 
 |         (*config_claimed_map)[this_value] = true; | 
 |       } | 
 |     } | 
 |   } | 
 | } | 
 | bool TableSplitter::VerifySplitConstraints(IAaptContext* context) { | 
 |   bool error = false; | 
 |   for (size_t i = 0; i < split_constraints_.size(); i++) { | 
 |     for (size_t j = i + 1; j < split_constraints_.size(); j++) { | 
 |       for (const ConfigDescription& config : split_constraints_[i].configs) { | 
 |         if (split_constraints_[j].configs.find(config) != | 
 |             split_constraints_[j].configs.end()) { | 
 |           context->GetDiagnostics()->Error(DiagMessage() | 
 |                                            << "config '" << config | 
 |                                            << "' appears in multiple splits, " | 
 |                                            << "target split ambiguous"); | 
 |           error = true; | 
 |         } | 
 |       } | 
 |     } | 
 |   } | 
 |   return !error; | 
 | } | 
 |  | 
 | void TableSplitter::SplitTable(ResourceTable* original_table) { | 
 |   const size_t split_count = split_constraints_.size(); | 
 |   for (auto& pkg : original_table->packages) { | 
 |     // Initialize all packages for splits. | 
 |     for (size_t idx = 0; idx < split_count; idx++) { | 
 |       ResourceTable* split_table = splits_[idx].get(); | 
 |       split_table->CreatePackage(pkg->name, pkg->id); | 
 |     } | 
 |  | 
 |     for (auto& type : pkg->types) { | 
 |       if (type->type == ResourceType::kMipmap) { | 
 |         // Always keep mipmaps. | 
 |         continue; | 
 |       } | 
 |  | 
 |       for (auto& entry : type->entries) { | 
 |         if (options_.config_filter) { | 
 |           // First eliminate any resource that we definitely don't want. | 
 |           for (std::unique_ptr<ResourceConfigValue>& config_value : | 
 |                entry->values) { | 
 |             if (!options_.config_filter->Match(config_value->config)) { | 
 |               // null out the entry. We will clean up and remove nulls at the | 
 |               // end for performance reasons. | 
 |               config_value.reset(); | 
 |             } | 
 |           } | 
 |         } | 
 |  | 
 |         // Organize the values into two separate buckets. Those that are | 
 |         // density-dependent | 
 |         // and those that are density-independent. | 
 |         // One density technically matches all density, it's just that some | 
 |         // densities | 
 |         // match better. So we need to be aware of the full set of densities to | 
 |         // make this | 
 |         // decision. | 
 |         ConfigDensityGroups density_groups; | 
 |         ConfigClaimedMap config_claimed_map; | 
 |         for (const std::unique_ptr<ResourceConfigValue>& config_value : | 
 |              entry->values) { | 
 |           if (config_value) { | 
 |             config_claimed_map[config_value.get()] = false; | 
 |  | 
 |             if (config_value->config.density != 0) { | 
 |               // Create a bucket for this density-dependent config. | 
 |               density_groups[CopyWithoutDensity(config_value->config)] | 
 |                   .push_back(config_value.get()); | 
 |             } | 
 |           } | 
 |         } | 
 |  | 
 |         // First we check all the splits. If it doesn't match one of the splits, | 
 |         // we | 
 |         // leave it in the base. | 
 |         for (size_t idx = 0; idx < split_count; idx++) { | 
 |           const SplitConstraints& split_constraint = split_constraints_[idx]; | 
 |           ResourceTable* split_table = splits_[idx].get(); | 
 |  | 
 |           // Select the values we want from this entry for this split. | 
 |           SplitValueSelector selector(split_constraint); | 
 |           std::vector<ResourceConfigValue*> selected_values = | 
 |               selector.SelectValues(density_groups, &config_claimed_map); | 
 |  | 
 |           // No need to do any work if we selected nothing. | 
 |           if (!selected_values.empty()) { | 
 |             // Create the same resource structure in the split. We do this | 
 |             // lazily because we might not have actual values for each | 
 |             // type/entry. | 
 |             ResourceTablePackage* split_pkg = | 
 |                 split_table->FindPackage(pkg->name); | 
 |             ResourceTableType* split_type = | 
 |                 split_pkg->FindOrCreateType(type->type); | 
 |             if (!split_type->id) { | 
 |               split_type->id = type->id; | 
 |               split_type->symbol_status = type->symbol_status; | 
 |             } | 
 |  | 
 |             ResourceEntry* split_entry = | 
 |                 split_type->FindOrCreateEntry(entry->name); | 
 |             if (!split_entry->id) { | 
 |               split_entry->id = entry->id; | 
 |               split_entry->symbol_status = entry->symbol_status; | 
 |             } | 
 |  | 
 |             // Copy the selected values into the new Split Entry. | 
 |             for (ResourceConfigValue* config_value : selected_values) { | 
 |               ResourceConfigValue* new_config_value = | 
 |                   split_entry->FindOrCreateValue(config_value->config, | 
 |                                                  config_value->product); | 
 |               new_config_value->value = std::unique_ptr<Value>( | 
 |                   config_value->value->Clone(&split_table->string_pool)); | 
 |             } | 
 |           } | 
 |         } | 
 |  | 
 |         if (!options_.preferred_densities.empty()) { | 
 |           MarkNonPreferredDensitiesAsClaimed(options_.preferred_densities, | 
 |                                              density_groups, | 
 |                                              &config_claimed_map); | 
 |         } | 
 |  | 
 |         // All splits are handled, now check to see what wasn't claimed and | 
 |         // remove | 
 |         // whatever exists in other splits. | 
 |         for (std::unique_ptr<ResourceConfigValue>& config_value : | 
 |              entry->values) { | 
 |           if (config_value && config_claimed_map[config_value.get()]) { | 
 |             // Claimed, remove from base. | 
 |             config_value.reset(); | 
 |           } | 
 |         } | 
 |  | 
 |         // Now erase all nullptrs. | 
 |         entry->values.erase( | 
 |             std::remove(entry->values.begin(), entry->values.end(), nullptr), | 
 |             entry->values.end()); | 
 |       } | 
 |     } | 
 |   } | 
 | } | 
 |  | 
 | }  // namespace aapt |