Home | History | Annotate | Download | only in link
      1 /*
      2  * Copyright (C) 2015 The Android Open Source Project
      3  *
      4  * Licensed under the Apache License, Version 2.0 (the "License");
      5  * you may not use this file except in compliance with the License.
      6  * You may obtain a copy of the License at
      7  *
      8  *      http://www.apache.org/licenses/LICENSE-2.0
      9  *
     10  * Unless required by applicable law or agreed to in writing, software
     11  * distributed under the License is distributed on an "AS IS" BASIS,
     12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
     13  * See the License for the specific language governing permissions and
     14  * limitations under the License.
     15  */
     16 
     17 #include "link/TableMerger.h"
     18 
     19 #include "android-base/logging.h"
     20 
     21 #include "ResourceTable.h"
     22 #include "ResourceUtils.h"
     23 #include "ResourceValues.h"
     24 #include "trace/TraceBuffer.h"
     25 #include "ValueVisitor.h"
     26 #include "util/Util.h"
     27 
     28 using ::android::StringPiece;
     29 
     30 namespace aapt {
     31 
     32 TableMerger::TableMerger(IAaptContext* context, ResourceTable* out_table,
     33                          const TableMergerOptions& options)
     34     : context_(context), master_table_(out_table), options_(options) {
     35   // Create the desired package that all tables will be merged into.
     36   master_package_ =
     37       master_table_->CreatePackage(context_->GetCompilationPackage(), context_->GetPackageId());
     38   CHECK(master_package_ != nullptr) << "package name or ID already taken";
     39 }
     40 
     41 bool TableMerger::Merge(const Source& src, ResourceTable* table, bool overlay) {
     42   TRACE_CALL();
     43   // We allow adding new resources if this is not an overlay, or if the options allow overlays
     44   // to add new resources.
     45   return MergeImpl(src, table, overlay, options_.auto_add_overlay || !overlay /*allow_new*/);
     46 }
     47 
     48 // This will merge packages with the same package name (or no package name).
     49 bool TableMerger::MergeImpl(const Source& src, ResourceTable* table, bool overlay, bool allow_new) {
     50   bool error = false;
     51   for (auto& package : table->packages) {
     52     // Only merge an empty package or the package we're building.
     53     // Other packages may exist, which likely contain attribute definitions.
     54     // This is because at compile time it is unknown if the attributes are
     55     // simply uses of the attribute or definitions.
     56     if (package->name.empty() || context_->GetCompilationPackage() == package->name) {
     57       // Merge here. Once the entries are merged and mangled, any references to them are still
     58       // valid. This is because un-mangled references are mangled, then looked up at resolution
     59       // time. Also, when linking, we convert references with no package name to use the compilation
     60       // package name.
     61       error |= !DoMerge(src, package.get(), false /*mangle*/, overlay, allow_new);
     62     }
     63   }
     64   return !error;
     65 }
     66 
     67 // This will merge and mangle resources from a static library. It is assumed that all FileReferences
     68 // have correctly set their io::IFile*.
     69 bool TableMerger::MergeAndMangle(const Source& src, const StringPiece& package_name,
     70                                  ResourceTable* table) {
     71   bool error = false;
     72   for (auto& package : table->packages) {
     73     // Warn of packages with an unrelated ID.
     74     if (package_name != package->name) {
     75       context_->GetDiagnostics()->Warn(DiagMessage(src) << "ignoring package " << package->name);
     76       continue;
     77     }
     78 
     79     bool mangle = package_name != context_->GetCompilationPackage();
     80     merged_packages_.insert(package->name);
     81     error |= !DoMerge(src, package.get(), mangle, false /*overlay*/, true /*allow_new*/);
     82   }
     83   return !error;
     84 }
     85 
     86 static bool MergeType(IAaptContext* context, const Source& src, ResourceTableType* dst_type,
     87                       ResourceTableType* src_type) {
     88   if (src_type->visibility_level > dst_type->visibility_level) {
     89     // The incoming type's visibility is stronger, so we should override the visibility.
     90     if (src_type->visibility_level == Visibility::Level::kPublic) {
     91       // Only copy the ID if the source is public, or else the ID is meaningless.
     92       dst_type->id = src_type->id;
     93     }
     94     dst_type->visibility_level = src_type->visibility_level;
     95   } else if (dst_type->visibility_level == Visibility::Level::kPublic &&
     96              src_type->visibility_level == Visibility::Level::kPublic && dst_type->id &&
     97              src_type->id && dst_type->id.value() != src_type->id.value()) {
     98     // Both types are public and have different IDs.
     99     context->GetDiagnostics()->Error(DiagMessage(src) << "cannot merge type '" << src_type->type
    100                                                       << "': conflicting public IDs");
    101     return false;
    102   }
    103   return true;
    104 }
    105 
    106 static bool MergeEntry(IAaptContext* context, const Source& src,
    107                        ResourceEntry* dst_entry, ResourceEntry* src_entry,
    108                        bool strict_visibility) {
    109   if (strict_visibility
    110       && dst_entry->visibility.level != Visibility::Level::kUndefined
    111       && src_entry->visibility.level != dst_entry->visibility.level) {
    112       context->GetDiagnostics()->Error(
    113           DiagMessage(src) << "cannot merge resource '" << dst_entry->name << "' with conflicting visibilities: "
    114                            << "public and private");
    115     return false;
    116   }
    117 
    118   // Copy over the strongest visibility.
    119   if (src_entry->visibility.level > dst_entry->visibility.level) {
    120     // Only copy the ID if the source is public, or else the ID is meaningless.
    121     if (src_entry->visibility.level == Visibility::Level::kPublic) {
    122       dst_entry->id = src_entry->id;
    123     }
    124     dst_entry->visibility = std::move(src_entry->visibility);
    125   } else if (src_entry->visibility.level == Visibility::Level::kPublic &&
    126              dst_entry->visibility.level == Visibility::Level::kPublic && dst_entry->id &&
    127              src_entry->id && src_entry->id != dst_entry->id) {
    128     // Both entries are public and have different IDs.
    129     context->GetDiagnostics()->Error(DiagMessage(src) << "cannot merge entry '" << src_entry->name
    130                                                       << "': conflicting public IDs");
    131     return false;
    132   }
    133 
    134   // Copy over the rest of the properties, if needed.
    135   if (src_entry->allow_new) {
    136     dst_entry->allow_new = std::move(src_entry->allow_new);
    137   }
    138 
    139   if (src_entry->overlayable_item) {
    140     if (dst_entry->overlayable_item) {
    141       CHECK(src_entry->overlayable_item.value().overlayable != nullptr);
    142       Overlayable* src_overlayable = src_entry->overlayable_item.value().overlayable.get();
    143 
    144       CHECK(dst_entry->overlayable_item.value().overlayable != nullptr);
    145       Overlayable* dst_overlayable = dst_entry->overlayable_item.value().overlayable.get();
    146 
    147       if (src_overlayable->name != dst_overlayable->name
    148           || src_overlayable->actor != dst_overlayable->actor
    149           || src_entry->overlayable_item.value().policies !=
    150              dst_entry->overlayable_item.value().policies) {
    151 
    152         // Do not allow a resource with an overlayable declaration to have that overlayable
    153         // declaration redefined.
    154         context->GetDiagnostics()->Error(DiagMessage(src_entry->overlayable_item.value().source)
    155                                              << "duplicate overlayable declaration for resource '"
    156                                              << src_entry->name << "'");
    157         context->GetDiagnostics()->Error(DiagMessage(dst_entry->overlayable_item.value().source)
    158                                              << "previous declaration here");
    159         return false;
    160       }
    161     }
    162 
    163     dst_entry->overlayable_item = std::move(src_entry->overlayable_item);
    164   }
    165 
    166   return true;
    167 }
    168 
    169 // Modified CollisionResolver which will merge Styleables and Styles. Used with overlays.
    170 //
    171 // Styleables are not actual resources, but they are treated as such during the compilation phase.
    172 //
    173 // Styleables and Styles don't simply overlay each other, their definitions merge and accumulate.
    174 // If both values are Styleables/Styles, we just merge them into the existing value.
    175 static ResourceTable::CollisionResult ResolveMergeCollision(Value* existing, Value* incoming,
    176                                                             StringPool* pool) {
    177   if (Styleable* existing_styleable = ValueCast<Styleable>(existing)) {
    178     if (Styleable* incoming_styleable = ValueCast<Styleable>(incoming)) {
    179       // Styleables get merged.
    180       existing_styleable->MergeWith(incoming_styleable);
    181       return ResourceTable::CollisionResult::kKeepOriginal;
    182     }
    183   } else if (Style* existing_style = ValueCast<Style>(existing)) {
    184     if (Style* incoming_style = ValueCast<Style>(incoming)) {
    185       // Styles get merged.
    186       existing_style->MergeWith(incoming_style, pool);
    187       return ResourceTable::CollisionResult::kKeepOriginal;
    188     }
    189   }
    190   // Delegate to the default handler.
    191   return ResourceTable::ResolveValueCollision(existing, incoming, true /* overlay */);
    192 }
    193 
    194 static ResourceTable::CollisionResult MergeConfigValue(IAaptContext* context,
    195                                                        const ResourceNameRef& res_name,
    196                                                        bool overlay,
    197                                                        ResourceConfigValue* dst_config_value,
    198                                                        ResourceConfigValue* src_config_value,
    199                                                        StringPool* pool) {
    200   using CollisionResult = ResourceTable::CollisionResult;
    201 
    202   Value* dst_value = dst_config_value->value.get();
    203   Value* src_value = src_config_value->value.get();
    204 
    205   CollisionResult collision_result;
    206   if (overlay) {
    207     collision_result = ResolveMergeCollision(dst_value, src_value, pool);
    208   } else {
    209     collision_result = ResourceTable::ResolveValueCollision(dst_value, src_value,
    210                                                             false /* overlay */);
    211   }
    212 
    213   if (collision_result == CollisionResult::kConflict) {
    214     context->GetDiagnostics()->Error(DiagMessage(src_value->GetSource())
    215                                      << "resource '" << res_name << "' has a conflicting value for "
    216                                      << "configuration (" << src_config_value->config << ")");
    217     context->GetDiagnostics()->Note(DiagMessage(dst_value->GetSource())
    218                                     << "originally defined here");
    219     return CollisionResult::kConflict;
    220   }
    221   return collision_result;
    222 }
    223 
    224 bool TableMerger::DoMerge(const Source& src, ResourceTablePackage* src_package, bool mangle_package,
    225                           bool overlay, bool allow_new_resources) {
    226   bool error = false;
    227 
    228   for (auto& src_type : src_package->types) {
    229     ResourceTableType* dst_type = master_package_->FindOrCreateType(src_type->type);
    230     if (!MergeType(context_, src, dst_type, src_type.get())) {
    231       error = true;
    232       continue;
    233     }
    234 
    235     for (auto& src_entry : src_type->entries) {
    236       std::string entry_name = src_entry->name;
    237       if (mangle_package) {
    238         entry_name = NameMangler::MangleEntry(src_package->name, src_entry->name);
    239       }
    240 
    241       ResourceEntry* dst_entry;
    242       if (allow_new_resources || src_entry->allow_new) {
    243         dst_entry = dst_type->FindOrCreateEntry(entry_name);
    244       } else {
    245         dst_entry = dst_type->FindEntry(entry_name);
    246       }
    247 
    248       const ResourceNameRef res_name(src_package->name, src_type->type, src_entry->name);
    249 
    250       if (!dst_entry) {
    251         context_->GetDiagnostics()->Error(DiagMessage(src)
    252                                           << "resource " << res_name
    253                                           << " does not override an existing resource");
    254         context_->GetDiagnostics()->Note(DiagMessage(src) << "define an <add-resource> tag or use "
    255                                                           << "--auto-add-overlay");
    256         error = true;
    257         continue;
    258       }
    259 
    260       if (!MergeEntry(context_, src, dst_entry, src_entry.get(), options_.strict_visibility)) {
    261         error = true;
    262         continue;
    263       }
    264 
    265       for (auto& src_config_value : src_entry->values) {
    266         using CollisionResult = ResourceTable::CollisionResult;
    267 
    268         ResourceConfigValue* dst_config_value = dst_entry->FindValue(
    269             src_config_value->config, src_config_value->product);
    270         if (dst_config_value) {
    271           CollisionResult collision_result =
    272               MergeConfigValue(context_, res_name, overlay, dst_config_value,
    273                                src_config_value.get(), &master_table_->string_pool);
    274           if (collision_result == CollisionResult::kConflict) {
    275             error = true;
    276             continue;
    277           } else if (collision_result == CollisionResult::kKeepOriginal) {
    278             continue;
    279           }
    280         } else {
    281           dst_config_value =
    282               dst_entry->FindOrCreateValue(src_config_value->config, src_config_value->product);
    283         }
    284 
    285         // Continue if we're taking the new resource.
    286 
    287         if (FileReference* f = ValueCast<FileReference>(src_config_value->value.get())) {
    288           std::unique_ptr<FileReference> new_file_ref;
    289           if (mangle_package) {
    290             new_file_ref = CloneAndMangleFile(src_package->name, *f);
    291           } else {
    292             new_file_ref = std::unique_ptr<FileReference>(f->Clone(&master_table_->string_pool));
    293           }
    294           dst_config_value->value = std::move(new_file_ref);
    295 
    296         } else {
    297           Maybe<std::string> original_comment = (dst_config_value->value)
    298               ? dst_config_value->value->GetComment() : Maybe<std::string>();
    299 
    300           dst_config_value->value = std::unique_ptr<Value>(
    301               src_config_value->value->Clone(&master_table_->string_pool));
    302 
    303           // Keep the comment from the original resource and ignore all comments from overlaying
    304           // resources
    305           if (overlay && original_comment) {
    306             dst_config_value->value->SetComment(original_comment.value());
    307           }
    308         }
    309       }
    310     }
    311   }
    312   return !error;
    313 }
    314 
    315 std::unique_ptr<FileReference> TableMerger::CloneAndMangleFile(
    316     const std::string& package, const FileReference& file_ref) {
    317   StringPiece prefix, entry, suffix;
    318   if (util::ExtractResFilePathParts(*file_ref.path, &prefix, &entry, &suffix)) {
    319     std::string mangled_entry = NameMangler::MangleEntry(package, entry.to_string());
    320     std::string newPath = prefix.to_string() + mangled_entry + suffix.to_string();
    321     std::unique_ptr<FileReference> new_file_ref =
    322         util::make_unique<FileReference>(master_table_->string_pool.MakeRef(newPath));
    323     new_file_ref->SetComment(file_ref.GetComment());
    324     new_file_ref->SetSource(file_ref.GetSource());
    325     new_file_ref->type = file_ref.type;
    326     new_file_ref->file = file_ref.file;
    327     return new_file_ref;
    328   }
    329   return std::unique_ptr<FileReference>(file_ref.Clone(&master_table_->string_pool));
    330 }
    331 
    332 bool TableMerger::MergeFile(const ResourceFile& file_desc, bool overlay, io::IFile* file) {
    333   ResourceTable table;
    334   std::string path = ResourceUtils::BuildResourceFileName(file_desc);
    335   std::unique_ptr<FileReference> file_ref =
    336       util::make_unique<FileReference>(table.string_pool.MakeRef(path));
    337   file_ref->SetSource(file_desc.source);
    338   file_ref->type = file_desc.type;
    339   file_ref->file = file;
    340 
    341   ResourceTablePackage* pkg = table.CreatePackage(file_desc.name.package, 0x0);
    342   pkg->FindOrCreateType(file_desc.name.type)
    343       ->FindOrCreateEntry(file_desc.name.entry)
    344       ->FindOrCreateValue(file_desc.config, {})
    345       ->value = std::move(file_ref);
    346 
    347   return DoMerge(file->GetSource(), pkg, false /*mangle*/, overlay /*overlay*/, true /*allow_new*/);
    348 }
    349 
    350 }  // namespace aapt
    351