| Adam Lesinski | 1ab598f | 2015-08-14 14:26:04 -0700 | [diff] [blame] | 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 |  | 
| Adam Lesinski | cacb28f | 2016-10-19 12:18:14 -0700 | [diff] [blame] | 17 | #include "link/TableMerger.h" | 
| Adam Lesinski | ce5e56e | 2016-10-21 17:56:45 -0700 | [diff] [blame] | 18 |  | 
|  | 19 | #include "android-base/logging.h" | 
|  | 20 |  | 
| Adam Lesinski | 1ab598f | 2015-08-14 14:26:04 -0700 | [diff] [blame] | 21 | #include "ResourceTable.h" | 
| Adam Lesinski | a6fe345 | 2015-12-09 15:20:52 -0800 | [diff] [blame] | 22 | #include "ResourceUtils.h" | 
| Adam Lesinski | 1ab598f | 2015-08-14 14:26:04 -0700 | [diff] [blame] | 23 | #include "ResourceValues.h" | 
| Fabien Sanglard | 2d34e76 | 2019-02-21 15:13:29 -0800 | [diff] [blame] | 24 | #include "trace/TraceBuffer.h" | 
| Adam Lesinski | 1ab598f | 2015-08-14 14:26:04 -0700 | [diff] [blame] | 25 | #include "ValueVisitor.h" | 
| Adam Lesinski | 1ab598f | 2015-08-14 14:26:04 -0700 | [diff] [blame] | 26 | #include "util/Util.h" | 
|  | 27 |  | 
| Adam Lesinski | 1ef0fa9 | 2017-08-15 21:32:49 -0700 | [diff] [blame] | 28 | using ::android::StringPiece; | 
| Adam Lesinski | d5083f6 | 2017-01-16 15:07:21 -0800 | [diff] [blame] | 29 |  | 
| Adam Lesinski | 1ab598f | 2015-08-14 14:26:04 -0700 | [diff] [blame] | 30 | namespace aapt { | 
|  | 31 |  | 
| Adam Lesinski | ce5e56e | 2016-10-21 17:56:45 -0700 | [diff] [blame] | 32 | TableMerger::TableMerger(IAaptContext* context, ResourceTable* out_table, | 
| Adam Lesinski | cacb28f | 2016-10-19 12:18:14 -0700 | [diff] [blame] | 33 | const TableMergerOptions& options) | 
| Ryan Mitchell | 4ea9075 | 2020-07-31 08:21:43 -0700 | [diff] [blame] | 34 | : context_(context), main_table_(out_table), options_(options) { | 
| Adam Lesinski | cacb28f | 2016-10-19 12:18:14 -0700 | [diff] [blame] | 35 | // Create the desired package that all tables will be merged into. | 
| Ryan Mitchell | 9634efb | 2021-03-19 14:53:17 -0700 | [diff] [blame] | 36 | main_package_ = main_table_->FindOrCreatePackage(context_->GetCompilationPackage()); | 
| Ryan Mitchell | 4ea9075 | 2020-07-31 08:21:43 -0700 | [diff] [blame] | 37 | CHECK(main_package_ != nullptr) << "package name or ID already taken"; | 
| Adam Lesinski | 1ab598f | 2015-08-14 14:26:04 -0700 | [diff] [blame] | 38 | } | 
|  | 39 |  | 
| Adam Lesinski | 8780eb6 | 2017-10-31 17:44:39 -0700 | [diff] [blame] | 40 | bool TableMerger::Merge(const Source& src, ResourceTable* table, bool overlay) { | 
| Fabien Sanglard | 2d34e76 | 2019-02-21 15:13:29 -0800 | [diff] [blame] | 41 | TRACE_CALL(); | 
| Adam Lesinski | 0045116 | 2017-10-03 07:44:08 -0700 | [diff] [blame] | 42 | // We allow adding new resources if this is not an overlay, or if the options allow overlays | 
|  | 43 | // to add new resources. | 
| Adam Lesinski | 8780eb6 | 2017-10-31 17:44:39 -0700 | [diff] [blame] | 44 | return MergeImpl(src, table, overlay, options_.auto_add_overlay || !overlay /*allow_new*/); | 
| Adam Lesinski | 64587af | 2016-02-18 18:33:06 -0800 | [diff] [blame] | 45 | } | 
|  | 46 |  | 
| Adam Lesinski | 1ef0fa9 | 2017-08-15 21:32:49 -0700 | [diff] [blame] | 47 | // This will merge packages with the same package name (or no package name). | 
| Adam Lesinski | 8780eb6 | 2017-10-31 17:44:39 -0700 | [diff] [blame] | 48 | bool TableMerger::MergeImpl(const Source& src, ResourceTable* table, bool overlay, bool allow_new) { | 
| Adam Lesinski | cacb28f | 2016-10-19 12:18:14 -0700 | [diff] [blame] | 49 | bool error = false; | 
|  | 50 | for (auto& package : table->packages) { | 
| Adam Lesinski | cacb28f | 2016-10-19 12:18:14 -0700 | [diff] [blame] | 51 | // Only merge an empty package or the package we're building. | 
|  | 52 | // Other packages may exist, which likely contain attribute definitions. | 
|  | 53 | // This is because at compile time it is unknown if the attributes are | 
| Adam Lesinski | b5dc4bd | 2017-02-22 19:29:29 -0800 | [diff] [blame] | 54 | // simply uses of the attribute or definitions. | 
|  | 55 | if (package->name.empty() || context_->GetCompilationPackage() == package->name) { | 
| Adam Lesinski | 1ef0fa9 | 2017-08-15 21:32:49 -0700 | [diff] [blame] | 56 | // Merge here. Once the entries are merged and mangled, any references to them are still | 
|  | 57 | // valid. This is because un-mangled references are mangled, then looked up at resolution | 
|  | 58 | // time. Also, when linking, we convert references with no package name to use the compilation | 
|  | 59 | // package name. | 
| Fabien Sanglard | 2369f54 | 2019-03-19 08:32:46 -0700 | [diff] [blame] | 60 | error |= !DoMerge(src, package.get(), false /*mangle*/, overlay, allow_new); | 
| Adam Lesinski | cacb28f | 2016-10-19 12:18:14 -0700 | [diff] [blame] | 61 | } | 
|  | 62 | } | 
|  | 63 | return !error; | 
| Adam Lesinski | 83f2255 | 2015-11-07 11:51:23 -0800 | [diff] [blame] | 64 | } | 
|  | 65 |  | 
| Adam Lesinski | 8780eb6 | 2017-10-31 17:44:39 -0700 | [diff] [blame] | 66 | // This will merge and mangle resources from a static library. It is assumed that all FileReferences | 
|  | 67 | // have correctly set their io::IFile*. | 
| Adam Lesinski | 1ef0fa9 | 2017-08-15 21:32:49 -0700 | [diff] [blame] | 68 | bool TableMerger::MergeAndMangle(const Source& src, const StringPiece& package_name, | 
| Adam Lesinski | 8780eb6 | 2017-10-31 17:44:39 -0700 | [diff] [blame] | 69 | ResourceTable* table) { | 
| Adam Lesinski | cacb28f | 2016-10-19 12:18:14 -0700 | [diff] [blame] | 70 | bool error = false; | 
|  | 71 | for (auto& package : table->packages) { | 
|  | 72 | // Warn of packages with an unrelated ID. | 
| Adam Lesinski | ce5e56e | 2016-10-21 17:56:45 -0700 | [diff] [blame] | 73 | if (package_name != package->name) { | 
| Adam Lesinski | 1ef0fa9 | 2017-08-15 21:32:49 -0700 | [diff] [blame] | 74 | context_->GetDiagnostics()->Warn(DiagMessage(src) << "ignoring package " << package->name); | 
| Adam Lesinski | cacb28f | 2016-10-19 12:18:14 -0700 | [diff] [blame] | 75 | continue; | 
| Adam Lesinski | 1ab598f | 2015-08-14 14:26:04 -0700 | [diff] [blame] | 76 | } | 
| Adam Lesinski | cacb28f | 2016-10-19 12:18:14 -0700 | [diff] [blame] | 77 |  | 
| Adam Lesinski | ce5e56e | 2016-10-21 17:56:45 -0700 | [diff] [blame] | 78 | bool mangle = package_name != context_->GetCompilationPackage(); | 
|  | 79 | merged_packages_.insert(package->name); | 
| Fabien Sanglard | 2369f54 | 2019-03-19 08:32:46 -0700 | [diff] [blame] | 80 | error |= !DoMerge(src, package.get(), mangle, false /*overlay*/, true /*allow_new*/); | 
| Adam Lesinski | cacb28f | 2016-10-19 12:18:14 -0700 | [diff] [blame] | 81 | } | 
|  | 82 | return !error; | 
| Adam Lesinski | 1ab598f | 2015-08-14 14:26:04 -0700 | [diff] [blame] | 83 | } | 
|  | 84 |  | 
| Adam Lesinski | 1ef0fa9 | 2017-08-15 21:32:49 -0700 | [diff] [blame] | 85 | static bool MergeType(IAaptContext* context, const Source& src, ResourceTableType* dst_type, | 
| Adam Lesinski | ce5e56e | 2016-10-21 17:56:45 -0700 | [diff] [blame] | 86 | ResourceTableType* src_type) { | 
| Ryan Mitchell | 9634efb | 2021-03-19 14:53:17 -0700 | [diff] [blame] | 87 | if (src_type->visibility_level >= dst_type->visibility_level) { | 
| Adam Lesinski | 1ef0fa9 | 2017-08-15 21:32:49 -0700 | [diff] [blame] | 88 | // The incoming type's visibility is stronger, so we should override the visibility. | 
| Adam Lesinski | 71be705 | 2017-12-12 16:48:07 -0800 | [diff] [blame] | 89 | dst_type->visibility_level = src_type->visibility_level; | 
| Adam Lesinski | cacb28f | 2016-10-19 12:18:14 -0700 | [diff] [blame] | 90 | } | 
|  | 91 | return true; | 
| Adam Lesinski | 5c3464c | 2016-08-24 16:03:48 -0700 | [diff] [blame] | 92 | } | 
|  | 93 |  | 
| Ryan Mitchell | e4e989c | 2018-10-29 02:21:50 -0700 | [diff] [blame] | 94 | static bool MergeEntry(IAaptContext* context, const Source& src, | 
| Izabela Orlowska | d51efe8 | 2018-04-24 18:18:29 +0100 | [diff] [blame] | 95 | ResourceEntry* dst_entry, ResourceEntry* src_entry, | 
|  | 96 | bool strict_visibility) { | 
|  | 97 | if (strict_visibility | 
|  | 98 | && dst_entry->visibility.level != Visibility::Level::kUndefined | 
|  | 99 | && src_entry->visibility.level != dst_entry->visibility.level) { | 
|  | 100 | context->GetDiagnostics()->Error( | 
|  | 101 | DiagMessage(src) << "cannot merge resource '" << dst_entry->name << "' with conflicting visibilities: " | 
|  | 102 | << "public and private"); | 
|  | 103 | return false; | 
|  | 104 | } | 
|  | 105 |  | 
| Adam Lesinski | 71be705 | 2017-12-12 16:48:07 -0800 | [diff] [blame] | 106 | // Copy over the strongest visibility. | 
|  | 107 | if (src_entry->visibility.level > dst_entry->visibility.level) { | 
|  | 108 | // Only copy the ID if the source is public, or else the ID is meaningless. | 
|  | 109 | if (src_entry->visibility.level == Visibility::Level::kPublic) { | 
| Adam Lesinski | ce5e56e | 2016-10-21 17:56:45 -0700 | [diff] [blame] | 110 | dst_entry->id = src_entry->id; | 
| Adam Lesinski | 5c3464c | 2016-08-24 16:03:48 -0700 | [diff] [blame] | 111 | } | 
| Adam Lesinski | 71be705 | 2017-12-12 16:48:07 -0800 | [diff] [blame] | 112 | dst_entry->visibility = std::move(src_entry->visibility); | 
|  | 113 | } else if (src_entry->visibility.level == Visibility::Level::kPublic && | 
|  | 114 | dst_entry->visibility.level == Visibility::Level::kPublic && dst_entry->id && | 
|  | 115 | src_entry->id && src_entry->id != dst_entry->id) { | 
| Adam Lesinski | cacb28f | 2016-10-19 12:18:14 -0700 | [diff] [blame] | 116 | // Both entries are public and have different IDs. | 
| Adam Lesinski | 1ef0fa9 | 2017-08-15 21:32:49 -0700 | [diff] [blame] | 117 | context->GetDiagnostics()->Error(DiagMessage(src) << "cannot merge entry '" << src_entry->name | 
|  | 118 | << "': conflicting public IDs"); | 
| Adam Lesinski | cacb28f | 2016-10-19 12:18:14 -0700 | [diff] [blame] | 119 | return false; | 
|  | 120 | } | 
| Adam Lesinski | 71be705 | 2017-12-12 16:48:07 -0800 | [diff] [blame] | 121 |  | 
|  | 122 | // Copy over the rest of the properties, if needed. | 
|  | 123 | if (src_entry->allow_new) { | 
|  | 124 | dst_entry->allow_new = std::move(src_entry->allow_new); | 
|  | 125 | } | 
|  | 126 |  | 
| Ryan Mitchell | 54237ff | 2018-12-13 15:44:29 -0800 | [diff] [blame] | 127 | if (src_entry->overlayable_item) { | 
|  | 128 | if (dst_entry->overlayable_item) { | 
| Ryan Mitchell | ced9a5c | 2019-04-05 10:44:16 -0700 | [diff] [blame] | 129 | CHECK(src_entry->overlayable_item.value().overlayable != nullptr); | 
|  | 130 | Overlayable* src_overlayable = src_entry->overlayable_item.value().overlayable.get(); | 
|  | 131 |  | 
|  | 132 | CHECK(dst_entry->overlayable_item.value().overlayable != nullptr); | 
|  | 133 | Overlayable* dst_overlayable = dst_entry->overlayable_item.value().overlayable.get(); | 
|  | 134 |  | 
|  | 135 | if (src_overlayable->name != dst_overlayable->name | 
|  | 136 | || src_overlayable->actor != dst_overlayable->actor | 
|  | 137 | || src_entry->overlayable_item.value().policies != | 
|  | 138 | dst_entry->overlayable_item.value().policies) { | 
|  | 139 |  | 
|  | 140 | // Do not allow a resource with an overlayable declaration to have that overlayable | 
|  | 141 | // declaration redefined. | 
|  | 142 | context->GetDiagnostics()->Error(DiagMessage(src_entry->overlayable_item.value().source) | 
|  | 143 | << "duplicate overlayable declaration for resource '" | 
|  | 144 | << src_entry->name << "'"); | 
|  | 145 | context->GetDiagnostics()->Error(DiagMessage(dst_entry->overlayable_item.value().source) | 
|  | 146 | << "previous declaration here"); | 
|  | 147 | return false; | 
|  | 148 | } | 
| Adam Lesinski | 71be705 | 2017-12-12 16:48:07 -0800 | [diff] [blame] | 149 | } | 
| Ryan Mitchell | ced9a5c | 2019-04-05 10:44:16 -0700 | [diff] [blame] | 150 |  | 
|  | 151 | dst_entry->overlayable_item = std::move(src_entry->overlayable_item); | 
| Adam Lesinski | 71be705 | 2017-12-12 16:48:07 -0800 | [diff] [blame] | 152 | } | 
| Ryan Mitchell | e4e989c | 2018-10-29 02:21:50 -0700 | [diff] [blame] | 153 |  | 
| Ryan Mitchell | 2fedba9 | 2021-04-23 07:47:38 -0700 | [diff] [blame] | 154 | if (src_entry->staged_id) { | 
|  | 155 | if (dst_entry->staged_id && | 
|  | 156 | dst_entry->staged_id.value().id != src_entry->staged_id.value().id) { | 
|  | 157 | context->GetDiagnostics()->Error(DiagMessage(src_entry->staged_id.value().source) | 
|  | 158 | << "conflicting staged id declaration for resource '" | 
|  | 159 | << src_entry->name << "'"); | 
|  | 160 | context->GetDiagnostics()->Error(DiagMessage(dst_entry->staged_id.value().source) | 
|  | 161 | << "previous declaration here"); | 
|  | 162 | } | 
|  | 163 | dst_entry->staged_id = std::move(src_entry->staged_id); | 
|  | 164 | } | 
|  | 165 |  | 
| Adam Lesinski | cacb28f | 2016-10-19 12:18:14 -0700 | [diff] [blame] | 166 | return true; | 
| Adam Lesinski | 5c3464c | 2016-08-24 16:03:48 -0700 | [diff] [blame] | 167 | } | 
|  | 168 |  | 
| Adam Lesinski | 5924d8c | 2017-05-30 15:15:58 -0700 | [diff] [blame] | 169 | // Modified CollisionResolver which will merge Styleables and Styles. Used with overlays. | 
|  | 170 | // | 
| Adam Lesinski | 1ef0fa9 | 2017-08-15 21:32:49 -0700 | [diff] [blame] | 171 | // Styleables are not actual resources, but they are treated as such during the compilation phase. | 
| Adam Lesinski | 5924d8c | 2017-05-30 15:15:58 -0700 | [diff] [blame] | 172 | // | 
| Adam Lesinski | 1ef0fa9 | 2017-08-15 21:32:49 -0700 | [diff] [blame] | 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. | 
| Donald Chai | 121c6e8 | 2019-06-12 12:51:57 -0700 | [diff] [blame] | 175 | static ResourceTable::CollisionResult ResolveMergeCollision( | 
|  | 176 | bool override_styles_instead_of_overlaying, Value* existing, Value* incoming, | 
|  | 177 | StringPool* pool) { | 
| Adam Lesinski | ce5e56e | 2016-10-21 17:56:45 -0700 | [diff] [blame] | 178 | if (Styleable* existing_styleable = ValueCast<Styleable>(existing)) { | 
|  | 179 | if (Styleable* incoming_styleable = ValueCast<Styleable>(incoming)) { | 
| Adam Lesinski | cacb28f | 2016-10-19 12:18:14 -0700 | [diff] [blame] | 180 | // Styleables get merged. | 
| Adam Lesinski | ce5e56e | 2016-10-21 17:56:45 -0700 | [diff] [blame] | 181 | existing_styleable->MergeWith(incoming_styleable); | 
| Adam Lesinski | cacb28f | 2016-10-19 12:18:14 -0700 | [diff] [blame] | 182 | return ResourceTable::CollisionResult::kKeepOriginal; | 
| Adam Lesinski | 5c3464c | 2016-08-24 16:03:48 -0700 | [diff] [blame] | 183 | } | 
| Donald Chai | 121c6e8 | 2019-06-12 12:51:57 -0700 | [diff] [blame] | 184 | } else if (!override_styles_instead_of_overlaying) { | 
|  | 185 | if (Style* existing_style = ValueCast<Style>(existing)) { | 
|  | 186 | if (Style* incoming_style = ValueCast<Style>(incoming)) { | 
|  | 187 | // Styles get merged. | 
|  | 188 | existing_style->MergeWith(incoming_style, pool); | 
|  | 189 | return ResourceTable::CollisionResult::kKeepOriginal; | 
|  | 190 | } | 
| Adam Lesinski | 5924d8c | 2017-05-30 15:15:58 -0700 | [diff] [blame] | 191 | } | 
| Adam Lesinski | cacb28f | 2016-10-19 12:18:14 -0700 | [diff] [blame] | 192 | } | 
|  | 193 | // Delegate to the default handler. | 
| Adam Lesinski | ce5e56e | 2016-10-21 17:56:45 -0700 | [diff] [blame] | 194 | return ResourceTable::ResolveValueCollision(existing, incoming); | 
| Adam Lesinski | 5c3464c | 2016-08-24 16:03:48 -0700 | [diff] [blame] | 195 | } | 
|  | 196 |  | 
| Adam Lesinski | 5924d8c | 2017-05-30 15:15:58 -0700 | [diff] [blame] | 197 | static ResourceTable::CollisionResult MergeConfigValue(IAaptContext* context, | 
|  | 198 | const ResourceNameRef& res_name, | 
| Adam Lesinski | 8780eb6 | 2017-10-31 17:44:39 -0700 | [diff] [blame] | 199 | bool overlay, | 
| Donald Chai | 121c6e8 | 2019-06-12 12:51:57 -0700 | [diff] [blame] | 200 | bool override_styles_instead_of_overlaying, | 
| Adam Lesinski | 5924d8c | 2017-05-30 15:15:58 -0700 | [diff] [blame] | 201 | ResourceConfigValue* dst_config_value, | 
|  | 202 | ResourceConfigValue* src_config_value, | 
|  | 203 | StringPool* pool) { | 
| Adam Lesinski | cacb28f | 2016-10-19 12:18:14 -0700 | [diff] [blame] | 204 | using CollisionResult = ResourceTable::CollisionResult; | 
| Adam Lesinski | 5c3464c | 2016-08-24 16:03:48 -0700 | [diff] [blame] | 205 |  | 
| Adam Lesinski | ce5e56e | 2016-10-21 17:56:45 -0700 | [diff] [blame] | 206 | Value* dst_value = dst_config_value->value.get(); | 
|  | 207 | Value* src_value = src_config_value->value.get(); | 
| Adam Lesinski | 5c3464c | 2016-08-24 16:03:48 -0700 | [diff] [blame] | 208 |  | 
| Adam Lesinski | ce5e56e | 2016-10-21 17:56:45 -0700 | [diff] [blame] | 209 | CollisionResult collision_result; | 
| Adam Lesinski | cacb28f | 2016-10-19 12:18:14 -0700 | [diff] [blame] | 210 | if (overlay) { | 
| Donald Chai | 121c6e8 | 2019-06-12 12:51:57 -0700 | [diff] [blame] | 211 | collision_result = | 
|  | 212 | ResolveMergeCollision(override_styles_instead_of_overlaying, dst_value, src_value, pool); | 
| Adam Lesinski | cacb28f | 2016-10-19 12:18:14 -0700 | [diff] [blame] | 213 | } else { | 
| Adam Lesinski | 5924d8c | 2017-05-30 15:15:58 -0700 | [diff] [blame] | 214 | collision_result = ResourceTable::ResolveValueCollision(dst_value, src_value); | 
| Adam Lesinski | cacb28f | 2016-10-19 12:18:14 -0700 | [diff] [blame] | 215 | } | 
|  | 216 |  | 
| Adam Lesinski | ce5e56e | 2016-10-21 17:56:45 -0700 | [diff] [blame] | 217 | if (collision_result == CollisionResult::kConflict) { | 
| Adam Lesinski | 5c3464c | 2016-08-24 16:03:48 -0700 | [diff] [blame] | 218 | if (overlay) { | 
| Adam Lesinski | cacb28f | 2016-10-19 12:18:14 -0700 | [diff] [blame] | 219 | return CollisionResult::kTakeNew; | 
| Adam Lesinski | 5c3464c | 2016-08-24 16:03:48 -0700 | [diff] [blame] | 220 | } | 
|  | 221 |  | 
| Adam Lesinski | cacb28f | 2016-10-19 12:18:14 -0700 | [diff] [blame] | 222 | // Error! | 
| Adam Lesinski | 5924d8c | 2017-05-30 15:15:58 -0700 | [diff] [blame] | 223 | context->GetDiagnostics()->Error(DiagMessage(src_value->GetSource()) | 
|  | 224 | << "resource '" << res_name << "' has a conflicting value for " | 
|  | 225 | << "configuration (" << src_config_value->config << ")"); | 
| Adam Lesinski | ce5e56e | 2016-10-21 17:56:45 -0700 | [diff] [blame] | 226 | context->GetDiagnostics()->Note(DiagMessage(dst_value->GetSource()) | 
| Adam Lesinski | cacb28f | 2016-10-19 12:18:14 -0700 | [diff] [blame] | 227 | << "originally defined here"); | 
|  | 228 | return CollisionResult::kConflict; | 
|  | 229 | } | 
| Adam Lesinski | ce5e56e | 2016-10-21 17:56:45 -0700 | [diff] [blame] | 230 | return collision_result; | 
| Adam Lesinski | 5c3464c | 2016-08-24 16:03:48 -0700 | [diff] [blame] | 231 | } | 
|  | 232 |  | 
| Fabien Sanglard | 2369f54 | 2019-03-19 08:32:46 -0700 | [diff] [blame] | 233 | bool TableMerger::DoMerge(const Source& src, ResourceTablePackage* src_package, bool mangle_package, | 
|  | 234 | bool overlay, bool allow_new_resources) { | 
| Adam Lesinski | cacb28f | 2016-10-19 12:18:14 -0700 | [diff] [blame] | 235 | bool error = false; | 
| Adam Lesinski | 1ab598f | 2015-08-14 14:26:04 -0700 | [diff] [blame] | 236 |  | 
| Adam Lesinski | ce5e56e | 2016-10-21 17:56:45 -0700 | [diff] [blame] | 237 | for (auto& src_type : src_package->types) { | 
| Ryan Mitchell | 4ea9075 | 2020-07-31 08:21:43 -0700 | [diff] [blame] | 238 | ResourceTableType* dst_type = main_package_->FindOrCreateType(src_type->type); | 
| Adam Lesinski | ce5e56e | 2016-10-21 17:56:45 -0700 | [diff] [blame] | 239 | if (!MergeType(context_, src, dst_type, src_type.get())) { | 
| Adam Lesinski | cacb28f | 2016-10-19 12:18:14 -0700 | [diff] [blame] | 240 | error = true; | 
|  | 241 | continue; | 
|  | 242 | } | 
|  | 243 |  | 
| Adam Lesinski | ce5e56e | 2016-10-21 17:56:45 -0700 | [diff] [blame] | 244 | for (auto& src_entry : src_type->entries) { | 
|  | 245 | std::string entry_name = src_entry->name; | 
|  | 246 | if (mangle_package) { | 
| Adam Lesinski | 4488f1c | 2017-05-26 17:33:38 -0700 | [diff] [blame] | 247 | entry_name = NameMangler::MangleEntry(src_package->name, src_entry->name); | 
| Adam Lesinski | cacb28f | 2016-10-19 12:18:14 -0700 | [diff] [blame] | 248 | } | 
|  | 249 |  | 
| Adam Lesinski | ce5e56e | 2016-10-21 17:56:45 -0700 | [diff] [blame] | 250 | ResourceEntry* dst_entry; | 
| Adam Lesinski | 71be705 | 2017-12-12 16:48:07 -0800 | [diff] [blame] | 251 | if (allow_new_resources || src_entry->allow_new) { | 
| Adam Lesinski | ce5e56e | 2016-10-21 17:56:45 -0700 | [diff] [blame] | 252 | dst_entry = dst_type->FindOrCreateEntry(entry_name); | 
| Adam Lesinski | cacb28f | 2016-10-19 12:18:14 -0700 | [diff] [blame] | 253 | } else { | 
| Adam Lesinski | ce5e56e | 2016-10-21 17:56:45 -0700 | [diff] [blame] | 254 | dst_entry = dst_type->FindEntry(entry_name); | 
| Adam Lesinski | cacb28f | 2016-10-19 12:18:14 -0700 | [diff] [blame] | 255 | } | 
|  | 256 |  | 
| Adam Lesinski | 4488f1c | 2017-05-26 17:33:38 -0700 | [diff] [blame] | 257 | const ResourceNameRef res_name(src_package->name, src_type->type, src_entry->name); | 
| Adam Lesinski | cacb28f | 2016-10-19 12:18:14 -0700 | [diff] [blame] | 258 |  | 
| Adam Lesinski | ce5e56e | 2016-10-21 17:56:45 -0700 | [diff] [blame] | 259 | if (!dst_entry) { | 
| Adam Lesinski | 4488f1c | 2017-05-26 17:33:38 -0700 | [diff] [blame] | 260 | context_->GetDiagnostics()->Error(DiagMessage(src) | 
|  | 261 | << "resource " << res_name | 
|  | 262 | << " does not override an existing resource"); | 
|  | 263 | context_->GetDiagnostics()->Note(DiagMessage(src) << "define an <add-resource> tag or use " | 
|  | 264 | << "--auto-add-overlay"); | 
| Adam Lesinski | cacb28f | 2016-10-19 12:18:14 -0700 | [diff] [blame] | 265 | error = true; | 
|  | 266 | continue; | 
|  | 267 | } | 
|  | 268 |  | 
| Ryan Mitchell | e4e989c | 2018-10-29 02:21:50 -0700 | [diff] [blame] | 269 | if (!MergeEntry(context_, src, dst_entry, src_entry.get(), options_.strict_visibility)) { | 
| Adam Lesinski | cacb28f | 2016-10-19 12:18:14 -0700 | [diff] [blame] | 270 | error = true; | 
|  | 271 | continue; | 
|  | 272 | } | 
|  | 273 |  | 
| Adam Lesinski | ce5e56e | 2016-10-21 17:56:45 -0700 | [diff] [blame] | 274 | for (auto& src_config_value : src_entry->values) { | 
| Adam Lesinski | cacb28f | 2016-10-19 12:18:14 -0700 | [diff] [blame] | 275 | using CollisionResult = ResourceTable::CollisionResult; | 
|  | 276 |  | 
| Adam Lesinski | ce5e56e | 2016-10-21 17:56:45 -0700 | [diff] [blame] | 277 | ResourceConfigValue* dst_config_value = dst_entry->FindValue( | 
|  | 278 | src_config_value->config, src_config_value->product); | 
|  | 279 | if (dst_config_value) { | 
| Donald Chai | 121c6e8 | 2019-06-12 12:51:57 -0700 | [diff] [blame] | 280 | CollisionResult collision_result = MergeConfigValue( | 
|  | 281 | context_, res_name, overlay, options_.override_styles_instead_of_overlaying, | 
| Ryan Mitchell | 4ea9075 | 2020-07-31 08:21:43 -0700 | [diff] [blame] | 282 | dst_config_value, src_config_value.get(), &main_table_->string_pool); | 
| Adam Lesinski | ce5e56e | 2016-10-21 17:56:45 -0700 | [diff] [blame] | 283 | if (collision_result == CollisionResult::kConflict) { | 
| Adam Lesinski | 5c3464c | 2016-08-24 16:03:48 -0700 | [diff] [blame] | 284 | error = true; | 
|  | 285 | continue; | 
| Adam Lesinski | ce5e56e | 2016-10-21 17:56:45 -0700 | [diff] [blame] | 286 | } else if (collision_result == CollisionResult::kKeepOriginal) { | 
| Adam Lesinski | cacb28f | 2016-10-19 12:18:14 -0700 | [diff] [blame] | 287 | continue; | 
|  | 288 | } | 
|  | 289 | } else { | 
| Adam Lesinski | 5924d8c | 2017-05-30 15:15:58 -0700 | [diff] [blame] | 290 | dst_config_value = | 
|  | 291 | dst_entry->FindOrCreateValue(src_config_value->config, src_config_value->product); | 
| Adam Lesinski | 1ab598f | 2015-08-14 14:26:04 -0700 | [diff] [blame] | 292 | } | 
|  | 293 |  | 
| Adam Lesinski | cacb28f | 2016-10-19 12:18:14 -0700 | [diff] [blame] | 294 | // Continue if we're taking the new resource. | 
| Ryan Mitchell | efcdb95 | 2021-04-14 17:31:37 -0700 | [diff] [blame] | 295 | CloningValueTransformer cloner(&main_table_->string_pool); | 
| Adam Lesinski | 5924d8c | 2017-05-30 15:15:58 -0700 | [diff] [blame] | 296 | if (FileReference* f = ValueCast<FileReference>(src_config_value->value.get())) { | 
| Adam Lesinski | ce5e56e | 2016-10-21 17:56:45 -0700 | [diff] [blame] | 297 | std::unique_ptr<FileReference> new_file_ref; | 
|  | 298 | if (mangle_package) { | 
|  | 299 | new_file_ref = CloneAndMangleFile(src_package->name, *f); | 
| Adam Lesinski | cacb28f | 2016-10-19 12:18:14 -0700 | [diff] [blame] | 300 | } else { | 
| Ryan Mitchell | efcdb95 | 2021-04-14 17:31:37 -0700 | [diff] [blame] | 301 | new_file_ref = std::unique_ptr<FileReference>(f->Transform(cloner)); | 
| Adam Lesinski | cacb28f | 2016-10-19 12:18:14 -0700 | [diff] [blame] | 302 | } | 
| Adam Lesinski | ce5e56e | 2016-10-21 17:56:45 -0700 | [diff] [blame] | 303 | dst_config_value->value = std::move(new_file_ref); | 
| Adam Lesinski | a6fe345 | 2015-12-09 15:20:52 -0800 | [diff] [blame] | 304 |  | 
| Adam Lesinski | cacb28f | 2016-10-19 12:18:14 -0700 | [diff] [blame] | 305 | } else { | 
| Ryan Mitchell | 4382e44 | 2021-07-14 12:53:01 -0700 | [diff] [blame] | 306 | auto original_comment = (dst_config_value->value) | 
|  | 307 | ? dst_config_value->value->GetComment() : std::optional<std::string>(); | 
| Ryan Mitchell | a9e3160 | 2018-06-28 16:41:38 -0700 | [diff] [blame] | 308 |  | 
| Ryan Mitchell | efcdb95 | 2021-04-14 17:31:37 -0700 | [diff] [blame] | 309 | dst_config_value->value = src_config_value->value->Transform(cloner); | 
| Ryan Mitchell | a9e3160 | 2018-06-28 16:41:38 -0700 | [diff] [blame] | 310 |  | 
|  | 311 | // Keep the comment from the original resource and ignore all comments from overlaying | 
|  | 312 | // resources | 
|  | 313 | if (overlay && original_comment) { | 
|  | 314 | dst_config_value->value->SetComment(original_comment.value()); | 
|  | 315 | } | 
| Adam Lesinski | 1ab598f | 2015-08-14 14:26:04 -0700 | [diff] [blame] | 316 | } | 
| Adam Lesinski | cacb28f | 2016-10-19 12:18:14 -0700 | [diff] [blame] | 317 | } | 
| Adam Lesinski | 1ab598f | 2015-08-14 14:26:04 -0700 | [diff] [blame] | 318 | } | 
| Adam Lesinski | cacb28f | 2016-10-19 12:18:14 -0700 | [diff] [blame] | 319 | } | 
|  | 320 | return !error; | 
| Adam Lesinski | 1ab598f | 2015-08-14 14:26:04 -0700 | [diff] [blame] | 321 | } | 
|  | 322 |  | 
| Adam Lesinski | ce5e56e | 2016-10-21 17:56:45 -0700 | [diff] [blame] | 323 | std::unique_ptr<FileReference> TableMerger::CloneAndMangleFile( | 
|  | 324 | const std::string& package, const FileReference& file_ref) { | 
| Adam Lesinski | cacb28f | 2016-10-19 12:18:14 -0700 | [diff] [blame] | 325 | StringPiece prefix, entry, suffix; | 
| Adam Lesinski | ce5e56e | 2016-10-21 17:56:45 -0700 | [diff] [blame] | 326 | if (util::ExtractResFilePathParts(*file_ref.path, &prefix, &entry, &suffix)) { | 
| Adam Lesinski | d5083f6 | 2017-01-16 15:07:21 -0800 | [diff] [blame] | 327 | std::string mangled_entry = NameMangler::MangleEntry(package, entry.to_string()); | 
|  | 328 | std::string newPath = prefix.to_string() + mangled_entry + suffix.to_string(); | 
| Adam Lesinski | ce5e56e | 2016-10-21 17:56:45 -0700 | [diff] [blame] | 329 | std::unique_ptr<FileReference> new_file_ref = | 
| Ryan Mitchell | 4ea9075 | 2020-07-31 08:21:43 -0700 | [diff] [blame] | 330 | util::make_unique<FileReference>(main_table_->string_pool.MakeRef(newPath)); | 
| Adam Lesinski | ce5e56e | 2016-10-21 17:56:45 -0700 | [diff] [blame] | 331 | new_file_ref->SetComment(file_ref.GetComment()); | 
|  | 332 | new_file_ref->SetSource(file_ref.GetSource()); | 
| Adam Lesinski | 0045116 | 2017-10-03 07:44:08 -0700 | [diff] [blame] | 333 | new_file_ref->type = file_ref.type; | 
|  | 334 | new_file_ref->file = file_ref.file; | 
| Adam Lesinski | ce5e56e | 2016-10-21 17:56:45 -0700 | [diff] [blame] | 335 | return new_file_ref; | 
| Adam Lesinski | cacb28f | 2016-10-19 12:18:14 -0700 | [diff] [blame] | 336 | } | 
| Ryan Mitchell | efcdb95 | 2021-04-14 17:31:37 -0700 | [diff] [blame] | 337 |  | 
|  | 338 | CloningValueTransformer cloner(&main_table_->string_pool); | 
|  | 339 | return std::unique_ptr<FileReference>(file_ref.Transform(cloner)); | 
| Adam Lesinski | 1ab598f | 2015-08-14 14:26:04 -0700 | [diff] [blame] | 340 | } | 
|  | 341 |  | 
| Adam Lesinski | 0045116 | 2017-10-03 07:44:08 -0700 | [diff] [blame] | 342 | bool TableMerger::MergeFile(const ResourceFile& file_desc, bool overlay, io::IFile* file) { | 
| Adam Lesinski | cacb28f | 2016-10-19 12:18:14 -0700 | [diff] [blame] | 343 | ResourceTable table; | 
| Adam Lesinski | ce5e56e | 2016-10-21 17:56:45 -0700 | [diff] [blame] | 344 | std::string path = ResourceUtils::BuildResourceFileName(file_desc); | 
|  | 345 | std::unique_ptr<FileReference> file_ref = | 
|  | 346 | util::make_unique<FileReference>(table.string_pool.MakeRef(path)); | 
|  | 347 | file_ref->SetSource(file_desc.source); | 
| Adam Lesinski | 0045116 | 2017-10-03 07:44:08 -0700 | [diff] [blame] | 348 | file_ref->type = file_desc.type; | 
| Adam Lesinski | ce5e56e | 2016-10-21 17:56:45 -0700 | [diff] [blame] | 349 | file_ref->file = file; | 
| Adam Lesinski | a6fe345 | 2015-12-09 15:20:52 -0800 | [diff] [blame] | 350 |  | 
| Ryan Mitchell | 9634efb | 2021-03-19 14:53:17 -0700 | [diff] [blame] | 351 | ResourceTablePackage* pkg = table.FindOrCreatePackage(file_desc.name.package); | 
| Iurii Makhno | cff10ce | 2022-02-15 19:33:50 +0000 | [diff] [blame] | 352 | pkg->FindOrCreateType(file_desc.name.type.type) | 
| Adam Lesinski | ce5e56e | 2016-10-21 17:56:45 -0700 | [diff] [blame] | 353 | ->FindOrCreateEntry(file_desc.name.entry) | 
|  | 354 | ->FindOrCreateValue(file_desc.config, {}) | 
|  | 355 | ->value = std::move(file_ref); | 
| Adam Lesinski | a6fe345 | 2015-12-09 15:20:52 -0800 | [diff] [blame] | 356 |  | 
| Fabien Sanglard | 2369f54 | 2019-03-19 08:32:46 -0700 | [diff] [blame] | 357 | return DoMerge(file->GetSource(), pkg, false /*mangle*/, overlay /*overlay*/, true /*allow_new*/); | 
| Adam Lesinski | 1ab598f | 2015-08-14 14:26:04 -0700 | [diff] [blame] | 358 | } | 
|  | 359 |  | 
| Adam Lesinski | cacb28f | 2016-10-19 12:18:14 -0700 | [diff] [blame] | 360 | }  // namespace aapt |