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/ReferenceLinker.h" 18 19 #include "android-base/logging.h" 20 #include "androidfw/ResourceTypes.h" 21 22 #include "Diagnostics.h" 23 #include "ResourceTable.h" 24 #include "ResourceUtils.h" 25 #include "ResourceValues.h" 26 #include "ValueVisitor.h" 27 #include "link/Linkers.h" 28 #include "process/IResourceTableConsumer.h" 29 #include "process/SymbolTable.h" 30 #include "trace/TraceBuffer.h" 31 #include "util/Util.h" 32 #include "xml/XmlUtil.h" 33 34 using ::aapt::ResourceUtils::StringBuilder; 35 using ::android::StringPiece; 36 37 namespace aapt { 38 39 namespace { 40 41 // The ReferenceLinkerVisitor will follow all references and make sure they point 42 // to resources that actually exist, either in the local resource table, or as external 43 // symbols. Once the target resource has been found, the ID of the resource will be assigned 44 // to the reference object. 45 // 46 // NOTE: All of the entries in the ResourceTable must be assigned IDs. 47 class ReferenceLinkerVisitor : public DescendingValueVisitor { 48 public: 49 using DescendingValueVisitor::Visit; 50 51 ReferenceLinkerVisitor(const CallSite& callsite, IAaptContext* context, SymbolTable* symbols, 52 StringPool* string_pool, xml::IPackageDeclStack* decl) 53 : callsite_(callsite), 54 context_(context), 55 symbols_(symbols), 56 package_decls_(decl), 57 string_pool_(string_pool) {} 58 59 void Visit(Reference* ref) override { 60 if (!ReferenceLinker::LinkReference(callsite_, ref, context_, symbols_, package_decls_)) { 61 error_ = true; 62 } 63 } 64 65 // We visit the Style specially because during this phase, values of attributes are 66 // all RawString values. Now that we are expected to resolve all symbols, we can 67 // lookup the attributes to find out which types are allowed for the attributes' values. 68 void Visit(Style* style) override { 69 if (style->parent) { 70 Visit(&style->parent.value()); 71 } 72 73 for (Style::Entry& entry : style->entries) { 74 std::string err_str; 75 76 // Transform the attribute reference so that it is using the fully qualified package 77 // name. This will also mark the reference as being able to see private resources if 78 // there was a '*' in the reference or if the package came from the private namespace. 79 Reference transformed_reference = entry.key; 80 ResolvePackage(package_decls_, &transformed_reference); 81 82 // Find the attribute in the symbol table and check if it is visible from this callsite. 83 const SymbolTable::Symbol* symbol = ReferenceLinker::ResolveAttributeCheckVisibility( 84 transformed_reference, callsite_, symbols_, &err_str); 85 if (symbol) { 86 // Assign our style key the correct ID. The ID may not exist. 87 entry.key.id = symbol->id; 88 89 // Try to convert the value to a more specific, typed value based on the attribute it is 90 // set to. 91 entry.value = ParseValueWithAttribute(std::move(entry.value), symbol->attribute.get()); 92 93 // Link/resolve the final value (mostly if it's a reference). 94 entry.value->Accept(this); 95 96 // Now verify that the type of this item is compatible with the 97 // attribute it is defined for. We pass `nullptr` as the DiagMessage so that this 98 // check is fast and we avoid creating a DiagMessage when the match is successful. 99 if (!symbol->attribute->Matches(*entry.value, nullptr)) { 100 // The actual type of this item is incompatible with the attribute. 101 DiagMessage msg(entry.key.GetSource()); 102 103 // Call the matches method again, this time with a DiagMessage so we fill in the actual 104 // error message. 105 symbol->attribute->Matches(*entry.value, &msg); 106 context_->GetDiagnostics()->Error(msg); 107 error_ = true; 108 } 109 110 } else { 111 DiagMessage msg(entry.key.GetSource()); 112 msg << "style attribute '"; 113 ReferenceLinker::WriteResourceName(entry.key, callsite_, package_decls_, &msg); 114 msg << "' " << err_str; 115 context_->GetDiagnostics()->Error(msg); 116 error_ = true; 117 } 118 } 119 } 120 121 bool HasError() { 122 return error_; 123 } 124 125 private: 126 DISALLOW_COPY_AND_ASSIGN(ReferenceLinkerVisitor); 127 128 // Transform a RawString value into a more specific, appropriate value, based on the 129 // Attribute. If a non RawString value is passed in, this is an identity transform. 130 std::unique_ptr<Item> ParseValueWithAttribute(std::unique_ptr<Item> value, 131 const Attribute* attr) { 132 if (RawString* raw_string = ValueCast<RawString>(value.get())) { 133 std::unique_ptr<Item> transformed = 134 ResourceUtils::TryParseItemForAttribute(*raw_string->value, attr); 135 136 // If we could not parse as any specific type, try a basic STRING. 137 if (!transformed && (attr->type_mask & android::ResTable_map::TYPE_STRING)) { 138 StringBuilder string_builder; 139 string_builder.AppendText(*raw_string->value); 140 if (string_builder) { 141 transformed = 142 util::make_unique<String>(string_pool_->MakeRef(string_builder.to_string())); 143 } 144 } 145 146 if (transformed) { 147 return transformed; 148 } 149 } 150 return value; 151 } 152 153 const CallSite& callsite_; 154 IAaptContext* context_; 155 SymbolTable* symbols_; 156 xml::IPackageDeclStack* package_decls_; 157 StringPool* string_pool_; 158 bool error_ = false; 159 }; 160 161 class EmptyDeclStack : public xml::IPackageDeclStack { 162 public: 163 EmptyDeclStack() = default; 164 165 Maybe<xml::ExtractedPackage> TransformPackageAlias(const StringPiece& alias) const override { 166 if (alias.empty()) { 167 return xml::ExtractedPackage{{}, true /*private*/}; 168 } 169 return {}; 170 } 171 172 private: 173 DISALLOW_COPY_AND_ASSIGN(EmptyDeclStack); 174 }; 175 176 // The symbol is visible if it is public, or if the reference to it is requesting private access 177 // or if the callsite comes from the same package. 178 bool IsSymbolVisible(const SymbolTable::Symbol& symbol, const Reference& ref, 179 const CallSite& callsite) { 180 if (symbol.is_public || ref.private_reference) { 181 return true; 182 } 183 184 if (ref.name) { 185 const ResourceName& name = ref.name.value(); 186 if (name.package.empty()) { 187 // If the symbol was found, and the package is empty, that means it was found in the local 188 // scope, which is always visible (private local). 189 return true; 190 } 191 192 // The symbol is visible if the reference is local to the same package it is defined in. 193 return callsite.package == name.package; 194 } 195 196 if (ref.id && symbol.id) { 197 return ref.id.value().package_id() == symbol.id.value().package_id(); 198 } 199 return false; 200 } 201 202 } // namespace 203 204 const SymbolTable::Symbol* ReferenceLinker::ResolveSymbol(const Reference& reference, 205 const CallSite& callsite, 206 SymbolTable* symbols) { 207 if (reference.name) { 208 const ResourceName& name = reference.name.value(); 209 if (name.package.empty()) { 210 // Use the callsite's package name if no package name was defined. 211 return symbols->FindByName(ResourceName(callsite.package, name.type, name.entry)); 212 } 213 return symbols->FindByName(name); 214 } else if (reference.id) { 215 return symbols->FindById(reference.id.value()); 216 } else { 217 return nullptr; 218 } 219 } 220 221 const SymbolTable::Symbol* ReferenceLinker::ResolveSymbolCheckVisibility(const Reference& reference, 222 const CallSite& callsite, 223 SymbolTable* symbols, 224 std::string* out_error) { 225 const SymbolTable::Symbol* symbol = ResolveSymbol(reference, callsite, symbols); 226 if (!symbol) { 227 if (out_error) *out_error = "not found"; 228 return nullptr; 229 } 230 231 if (!IsSymbolVisible(*symbol, reference, callsite)) { 232 if (out_error) *out_error = "is private"; 233 return nullptr; 234 } 235 return symbol; 236 } 237 238 const SymbolTable::Symbol* ReferenceLinker::ResolveAttributeCheckVisibility( 239 const Reference& reference, const CallSite& callsite, SymbolTable* symbols, 240 std::string* out_error) { 241 const SymbolTable::Symbol* symbol = 242 ResolveSymbolCheckVisibility(reference, callsite, symbols, out_error); 243 if (!symbol) { 244 return nullptr; 245 } 246 247 if (!symbol->attribute) { 248 if (out_error) *out_error = "is not an attribute"; 249 return nullptr; 250 } 251 return symbol; 252 } 253 254 Maybe<xml::AaptAttribute> ReferenceLinker::CompileXmlAttribute(const Reference& reference, 255 const CallSite& callsite, 256 SymbolTable* symbols, 257 std::string* out_error) { 258 const SymbolTable::Symbol* symbol = 259 ResolveAttributeCheckVisibility(reference, callsite, symbols, out_error); 260 if (!symbol) { 261 return {}; 262 } 263 264 if (!symbol->attribute) { 265 if (out_error) *out_error = "is not an attribute"; 266 return {}; 267 } 268 return xml::AaptAttribute(*symbol->attribute, symbol->id); 269 } 270 271 void ReferenceLinker::WriteResourceName(const Reference& ref, const CallSite& callsite, 272 const xml::IPackageDeclStack* decls, DiagMessage* out_msg) { 273 CHECK(out_msg != nullptr); 274 if (!ref.name) { 275 *out_msg << ref.id.value(); 276 return; 277 } 278 279 *out_msg << ref.name.value(); 280 281 Reference fully_qualified = ref; 282 xml::ResolvePackage(decls, &fully_qualified); 283 284 ResourceName& full_name = fully_qualified.name.value(); 285 if (full_name.package.empty()) { 286 full_name.package = callsite.package; 287 } 288 289 if (full_name != ref.name.value()) { 290 *out_msg << " (aka " << full_name << ")"; 291 } 292 } 293 294 void ReferenceLinker::WriteAttributeName(const Reference& ref, const CallSite& callsite, 295 const xml::IPackageDeclStack* decls, 296 DiagMessage* out_msg) { 297 CHECK(out_msg != nullptr); 298 if (!ref.name) { 299 *out_msg << ref.id.value(); 300 return; 301 } 302 303 const ResourceName& ref_name = ref.name.value(); 304 CHECK_EQ(ref_name.type, ResourceType::kAttr); 305 306 if (!ref_name.package.empty()) { 307 *out_msg << ref_name.package << ":"; 308 } 309 *out_msg << ref_name.entry; 310 311 Reference fully_qualified = ref; 312 xml::ResolvePackage(decls, &fully_qualified); 313 314 ResourceName& full_name = fully_qualified.name.value(); 315 if (full_name.package.empty()) { 316 full_name.package = callsite.package; 317 } 318 319 if (full_name != ref.name.value()) { 320 *out_msg << " (aka " << full_name.package << ":" << full_name.entry << ")"; 321 } 322 } 323 324 bool ReferenceLinker::LinkReference(const CallSite& callsite, Reference* reference, 325 IAaptContext* context, SymbolTable* symbols, 326 const xml::IPackageDeclStack* decls) { 327 CHECK(reference != nullptr); 328 if (!reference->name && !reference->id) { 329 // This is @null. 330 return true; 331 } 332 333 Reference transformed_reference = *reference; 334 xml::ResolvePackage(decls, &transformed_reference); 335 336 std::string err_str; 337 const SymbolTable::Symbol* s = 338 ResolveSymbolCheckVisibility(transformed_reference, callsite, symbols, &err_str); 339 if (s) { 340 // The ID may not exist. This is fine because of the possibility of building 341 // against libraries without assigned IDs. 342 // Ex: Linking against own resources when building a static library. 343 reference->id = s->id; 344 reference->is_dynamic = s->is_dynamic; 345 return true; 346 } 347 348 DiagMessage error_msg(reference->GetSource()); 349 error_msg << "resource "; 350 WriteResourceName(*reference, callsite, decls, &error_msg); 351 error_msg << " " << err_str; 352 context->GetDiagnostics()->Error(error_msg); 353 return false; 354 } 355 356 bool ReferenceLinker::Consume(IAaptContext* context, ResourceTable* table) { 357 TRACE_NAME("ReferenceLinker::Consume"); 358 EmptyDeclStack decl_stack; 359 bool error = false; 360 for (auto& package : table->packages) { 361 // Since we're linking, each package must have a name. 362 CHECK(!package->name.empty()) << "all packages being linked must have a name"; 363 364 for (auto& type : package->types) { 365 for (auto& entry : type->entries) { 366 // First, unmangle the name if necessary. 367 ResourceName name(package->name, type->type, entry->name); 368 NameMangler::Unmangle(&name.entry, &name.package); 369 370 // Symbol state information may be lost if there is no value for the resource. 371 if (entry->visibility.level != Visibility::Level::kUndefined && entry->values.empty()) { 372 context->GetDiagnostics()->Error(DiagMessage(entry->visibility.source) 373 << "no definition for declared symbol '" << name 374 << "'"); 375 error = true; 376 } 377 378 // Ensure that definitions for values declared as overlayable exist 379 if (entry->overlayable_item && entry->values.empty()) { 380 context->GetDiagnostics()->Error(DiagMessage(entry->overlayable_item.value().source) 381 << "no definition for overlayable symbol '" 382 << name << "'"); 383 error = true; 384 } 385 386 // The context of this resource is the package in which it is defined. 387 const CallSite callsite{name.package}; 388 ReferenceLinkerVisitor visitor(callsite, context, context->GetExternalSymbols(), 389 &table->string_pool, &decl_stack); 390 391 for (auto& config_value : entry->values) { 392 config_value->value->Accept(&visitor); 393 } 394 395 if (visitor.HasError()) { 396 error = true; 397 } 398 } 399 } 400 } 401 return !error; 402 } 403 404 } // namespace aapt 405