| /* |
| * 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 "optimize/ResourceDeduper.h" |
| |
| #include <algorithm> |
| |
| #include "DominatorTree.h" |
| #include "ResourceTable.h" |
| #include "trace/TraceBuffer.h" |
| |
| using android::ConfigDescription; |
| |
| namespace aapt { |
| |
| namespace { |
| |
| /** |
| * Remove duplicated key-value entries from dominated resources. |
| * |
| * Based on the dominator tree, we can remove a value of an entry if: |
| * |
| * 1. The configuration for the entry's value is dominated by a configuration |
| * with an equivalent entry value. |
| * 2. All compatible configurations for the entry (those not in conflict and |
| * unrelated by domination with the configuration for the entry's value) have |
| * an equivalent entry value. |
| */ |
| class DominatedKeyValueRemover : public DominatorTree::BottomUpVisitor { |
| public: |
| using Node = DominatorTree::Node; |
| |
| explicit DominatedKeyValueRemover(IAaptContext* context, ResourceEntry* entry) |
| : context_(context), entry_(entry) {} |
| |
| void VisitConfig(Node* node) { |
| Node* parent = node->parent(); |
| if (!parent) { |
| return; |
| } |
| ResourceConfigValue* node_value = node->value(); |
| ResourceConfigValue* parent_value = parent->value(); |
| if (!node_value || !parent_value) { |
| return; |
| } |
| if (!node_value->value->Equals(parent_value->value.get())) { |
| return; |
| } |
| |
| // Compare compatible configs for this entry and ensure the values are |
| // equivalent. |
| const ConfigDescription& node_configuration = node_value->config; |
| for (const auto& sibling : parent->children()) { |
| ResourceConfigValue* sibling_value = sibling->value(); |
| if (!sibling_value->value) { |
| // Sibling was already removed. |
| continue; |
| } |
| if (node_configuration.IsCompatibleWith(sibling_value->config) && |
| !node_value->value->Equals(sibling_value->value.get())) { |
| // The configurations are compatible, but the value is |
| // different, so we can't remove this value. |
| return; |
| } |
| } |
| if (context_->IsVerbose()) { |
| context_->GetDiagnostics()->Note(android::DiagMessage(node_value->value->GetSource()) |
| << "removing dominated duplicate resource with name \"" |
| << entry_->name << "\""); |
| context_->GetDiagnostics()->Note(android::DiagMessage(parent_value->value->GetSource()) |
| << "dominated here"); |
| } |
| node_value->value = {}; |
| } |
| |
| private: |
| DISALLOW_COPY_AND_ASSIGN(DominatedKeyValueRemover); |
| |
| IAaptContext* context_; |
| ResourceEntry* entry_; |
| }; |
| |
| static void DedupeEntry(IAaptContext* context, ResourceEntry* entry) { |
| DominatorTree tree(entry->values); |
| DominatedKeyValueRemover remover(context, entry); |
| tree.Accept(&remover); |
| |
| // Erase the values that were removed. |
| entry->values.erase( |
| std::remove_if( |
| entry->values.begin(), entry->values.end(), |
| [](const std::unique_ptr<ResourceConfigValue>& val) -> bool { |
| return val == nullptr || val->value == nullptr; |
| }), |
| entry->values.end()); |
| } |
| |
| } // namespace |
| |
| bool ResourceDeduper::Consume(IAaptContext* context, ResourceTable* table) { |
| TRACE_CALL(); |
| for (auto& package : table->packages) { |
| for (auto& type : package->types) { |
| for (auto& entry : type->entries) { |
| DedupeEntry(context, entry.get()); |
| } |
| } |
| } |
| return true; |
| } |
| |
| } // namespace aapt |