/* * Copyright (C) 2015 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 "ConfigDescription.h" #include "NameMangler.h" #include "ResourceTable.h" #include "ResourceValues.h" #include "ValueVisitor.h" #include "util/Util.h" #include <algorithm> #include <androidfw/ResourceTypes.h> #include <memory> #include <string> #include <tuple> namespace aapt { static bool lessThanType(const std::unique_ptr<ResourceTableType>& lhs, ResourceType rhs) { return lhs->type < rhs; } template <typename T> static bool lessThanStructWithName(const std::unique_ptr<T>& lhs, const StringPiece16& rhs) { return lhs->name.compare(0, lhs->name.size(), rhs.data(), rhs.size()) < 0; } ResourceTablePackage* ResourceTable::findPackage(const StringPiece16& name) { const auto last = packages.end(); auto iter = std::lower_bound(packages.begin(), last, name, lessThanStructWithName<ResourceTablePackage>); if (iter != last && name == (*iter)->name) { return iter->get(); } return nullptr; } ResourceTablePackage* ResourceTable::findPackageById(uint8_t id) { for (auto& package : packages) { if (package->id && package->id.value() == id) { return package.get(); } } return nullptr; } ResourceTablePackage* ResourceTable::createPackage(const StringPiece16& name, Maybe<uint8_t> id) { ResourceTablePackage* package = findOrCreatePackage(name); if (id && !package->id) { package->id = id; return package; } if (id && package->id && package->id.value() != id.value()) { return nullptr; } return package; } ResourceTablePackage* ResourceTable::findOrCreatePackage(const StringPiece16& name) { const auto last = packages.end(); auto iter = std::lower_bound(packages.begin(), last, name, lessThanStructWithName<ResourceTablePackage>); if (iter != last && name == (*iter)->name) { return iter->get(); } std::unique_ptr<ResourceTablePackage> newPackage = util::make_unique<ResourceTablePackage>(); newPackage->name = name.toString(); return packages.emplace(iter, std::move(newPackage))->get(); } ResourceTableType* ResourceTablePackage::findType(ResourceType type) { const auto last = types.end(); auto iter = std::lower_bound(types.begin(), last, type, lessThanType); if (iter != last && (*iter)->type == type) { return iter->get(); } return nullptr; } ResourceTableType* ResourceTablePackage::findOrCreateType(ResourceType type) { const auto last = types.end(); auto iter = std::lower_bound(types.begin(), last, type, lessThanType); if (iter != last && (*iter)->type == type) { return iter->get(); } return types.emplace(iter, new ResourceTableType(type))->get(); } ResourceEntry* ResourceTableType::findEntry(const StringPiece16& name) { const auto last = entries.end(); auto iter = std::lower_bound(entries.begin(), last, name, lessThanStructWithName<ResourceEntry>); if (iter != last && name == (*iter)->name) { return iter->get(); } return nullptr; } ResourceEntry* ResourceTableType::findOrCreateEntry(const StringPiece16& name) { auto last = entries.end(); auto iter = std::lower_bound(entries.begin(), last, name, lessThanStructWithName<ResourceEntry>); if (iter != last && name == (*iter)->name) { return iter->get(); } return entries.emplace(iter, new ResourceEntry(name))->get(); } ResourceConfigValue* ResourceEntry::findValue(const ConfigDescription& config) { return findValue(config, StringPiece()); } struct ConfigKey { const ConfigDescription* config; const StringPiece& product; }; bool ltConfigKeyRef(const std::unique_ptr<ResourceConfigValue>& lhs, const ConfigKey& rhs) { int cmp = lhs->config.compare(*rhs.config); if (cmp == 0) { cmp = StringPiece(lhs->product).compare(rhs.product); } return cmp < 0; } ResourceConfigValue* ResourceEntry::findValue(const ConfigDescription& config, const StringPiece& product) { auto iter = std::lower_bound(values.begin(), values.end(), ConfigKey{ &config, product }, ltConfigKeyRef); if (iter != values.end()) { ResourceConfigValue* value = iter->get(); if (value->config == config && StringPiece(value->product) == product) { return value; } } return nullptr; } ResourceConfigValue* ResourceEntry::findOrCreateValue(const ConfigDescription& config, const StringPiece& product) { auto iter = std::lower_bound(values.begin(), values.end(), ConfigKey{ &config, product }, ltConfigKeyRef); if (iter != values.end()) { ResourceConfigValue* value = iter->get(); if (value->config == config && StringPiece(value->product) == product) { return value; } } ResourceConfigValue* newValue = values.insert( iter, util::make_unique<ResourceConfigValue>(config, product))->get(); return newValue; } std::vector<ResourceConfigValue*> ResourceEntry::findAllValues(const ConfigDescription& config) { std::vector<ResourceConfigValue*> results; auto iter = values.begin(); for (; iter != values.end(); ++iter) { ResourceConfigValue* value = iter->get(); if (value->config == config) { results.push_back(value); ++iter; break; } } for (; iter != values.end(); ++iter) { ResourceConfigValue* value = iter->get(); if (value->config == config) { results.push_back(value); } } return results; } /** * The default handler for collisions. A return value of -1 means keep the * existing value, 0 means fail, and +1 means take the incoming value. */ int ResourceTable::resolveValueCollision(Value* existing, Value* incoming) { Attribute* existingAttr = valueCast<Attribute>(existing); Attribute* incomingAttr = valueCast<Attribute>(incoming); if (!incomingAttr) { if (incoming->isWeak()) { // We're trying to add a weak resource but a resource // already exists. Keep the existing. return -1; } else if (existing->isWeak()) { // Override the weak resource with the new strong resource. return 1; } // The existing and incoming values are strong, this is an error // if the values are not both attributes. return 0; } if (!existingAttr) { if (existing->isWeak()) { // The existing value is not an attribute and it is weak, // so take the incoming attribute value. return 1; } // The existing value is not an attribute and it is strong, // so the incoming attribute value is an error. return 0; } assert(incomingAttr && existingAttr); // // Attribute specific handling. At this point we know both // values are attributes. Since we can declare and define // attributes all-over, we do special handling to see // which definition sticks. // if (existingAttr->typeMask == incomingAttr->typeMask) { // The two attributes are both DECLs, but they are plain attributes // with the same formats. // Keep the strongest one. return existingAttr->isWeak() ? 1 : -1; } if (existingAttr->isWeak() && existingAttr->typeMask == android::ResTable_map::TYPE_ANY) { // Any incoming attribute is better than this. return 1; } if (incomingAttr->isWeak() && incomingAttr->typeMask == android::ResTable_map::TYPE_ANY) { // The incoming attribute may be a USE instead of a DECL. // Keep the existing attribute. return -1; } return 0; } static constexpr const char16_t* kValidNameChars = u"._-"; static constexpr const char16_t* kValidNameMangledChars = u"._-$"; bool ResourceTable::addResource(const ResourceNameRef& name, const ConfigDescription& config, const StringPiece& product, std::unique_ptr<Value> value, IDiagnostics* diag) { return addResourceImpl(name, {}, config, product, std::move(value), kValidNameChars, resolveValueCollision, diag); } bool ResourceTable::addResource(const ResourceNameRef& name, const ResourceId resId, const ConfigDescription& config, const StringPiece& product, std::unique_ptr<Value> value, IDiagnostics* diag) { return addResourceImpl(name, resId, config, product, std::move(value), kValidNameChars, resolveValueCollision, diag); } bool ResourceTable::addFileReference(const ResourceNameRef& name, const ConfigDescription& config, const Source& source, const StringPiece16& path, IDiagnostics* diag) { return addFileReferenceImpl(name, config, source, path, nullptr, kValidNameChars, diag); } bool ResourceTable::addFileReferenceAllowMangled(const ResourceNameRef& name, const ConfigDescription& config, const Source& source, const StringPiece16& path, io::IFile* file, IDiagnostics* diag) { return addFileReferenceImpl(name, config, source, path, file, kValidNameMangledChars, diag); } bool ResourceTable::addFileReferenceImpl(const ResourceNameRef& name, const ConfigDescription& config, const Source& source, const StringPiece16& path, io::IFile* file, const char16_t* validChars, IDiagnostics* diag) { std::unique_ptr<FileReference> fileRef = util::make_unique<FileReference>( stringPool.makeRef(path)); fileRef->setSource(source); fileRef->file = file; return addResourceImpl(name, ResourceId{}, config, StringPiece{}, std::move(fileRef), kValidNameChars, resolveValueCollision, diag); } bool ResourceTable::addResourceAllowMangled(const ResourceNameRef& name, const ConfigDescription& config, const StringPiece& product, std::unique_ptr<Value> value, IDiagnostics* diag) { return addResourceImpl(name, ResourceId{}, config, product, std::move(value), kValidNameMangledChars, resolveValueCollision, diag); } bool ResourceTable::addResourceAllowMangled(const ResourceNameRef& name, const ResourceId id, const ConfigDescription& config, const StringPiece& product, std::unique_ptr<Value> value, IDiagnostics* diag) { return addResourceImpl(name, id, config, product, std::move(value), kValidNameMangledChars, resolveValueCollision, diag); } bool ResourceTable::addResourceImpl(const ResourceNameRef& name, const ResourceId resId, const ConfigDescription& config, const StringPiece& product, std::unique_ptr<Value> value, const char16_t* validChars, std::function<int(Value*,Value*)> conflictResolver, IDiagnostics* diag) { assert(value && "value can't be nullptr"); assert(diag && "diagnostics can't be nullptr"); auto badCharIter = util::findNonAlphaNumericAndNotInSet(name.entry, validChars); if (badCharIter != name.entry.end()) { diag->error(DiagMessage(value->getSource()) << "resource '" << name << "' has invalid entry name '" << name.entry << "'. Invalid character '" << StringPiece16(badCharIter, 1) << "'"); return false; } ResourceTablePackage* package = findOrCreatePackage(name.package); if (resId.isValid() && package->id && package->id.value() != resId.packageId()) { diag->error(DiagMessage(value->getSource()) << "trying to add resource '" << name << "' with ID " << resId << " but package '" << package->name << "' already has ID " << std::hex << (int) package->id.value() << std::dec); return false; } ResourceTableType* type = package->findOrCreateType(name.type); if (resId.isValid() && type->id && type->id.value() != resId.typeId()) { diag->error(DiagMessage(value->getSource()) << "trying to add resource '" << name << "' with ID " << resId << " but type '" << type->type << "' already has ID " << std::hex << (int) type->id.value() << std::dec); return false; } ResourceEntry* entry = type->findOrCreateEntry(name.entry); if (resId.isValid() && entry->id && entry->id.value() != resId.entryId()) { diag->error(DiagMessage(value->getSource()) << "trying to add resource '" << name << "' with ID " << resId << " but resource already has ID " << ResourceId(package->id.value(), type->id.value(), entry->id.value())); return false; } ResourceConfigValue* configValue = entry->findOrCreateValue(config, product); if (!configValue->value) { // Resource does not exist, add it now. configValue->value = std::move(value); } else { int collisionResult = conflictResolver(configValue->value.get(), value.get()); if (collisionResult > 0) { // Take the incoming value. configValue->value = std::move(value); } else if (collisionResult == 0) { diag->error(DiagMessage(value->getSource()) << "duplicate value for resource '" << name << "' " << "with config '" << config << "'"); diag->error(DiagMessage(configValue->value->getSource()) << "resource previously defined here"); return false; } } if (resId.isValid()) { package->id = resId.packageId(); type->id = resId.typeId(); entry->id = resId.entryId(); } return true; } bool ResourceTable::setSymbolState(const ResourceNameRef& name, const ResourceId resId, const Symbol& symbol, IDiagnostics* diag) { return setSymbolStateImpl(name, resId, symbol, kValidNameChars, diag); } bool ResourceTable::setSymbolStateAllowMangled(const ResourceNameRef& name, const ResourceId resId, const Symbol& symbol, IDiagnostics* diag) { return setSymbolStateImpl(name, resId, symbol, kValidNameMangledChars, diag); } bool ResourceTable::setSymbolStateImpl(const ResourceNameRef& name, const ResourceId resId, const Symbol& symbol, const char16_t* validChars, IDiagnostics* diag) { assert(diag && "diagnostics can't be nullptr"); auto badCharIter = util::findNonAlphaNumericAndNotInSet(name.entry, validChars); if (badCharIter != name.entry.end()) { diag->error(DiagMessage(symbol.source) << "resource '" << name << "' has invalid entry name '" << name.entry << "'. Invalid character '" << StringPiece16(badCharIter, 1) << "'"); return false; } ResourceTablePackage* package = findOrCreatePackage(name.package); if (resId.isValid() && package->id && package->id.value() != resId.packageId()) { diag->error(DiagMessage(symbol.source) << "trying to add resource '" << name << "' with ID " << resId << " but package '" << package->name << "' already has ID " << std::hex << (int) package->id.value() << std::dec); return false; } ResourceTableType* type = package->findOrCreateType(name.type); if (resId.isValid() && type->id && type->id.value() != resId.typeId()) { diag->error(DiagMessage(symbol.source) << "trying to add resource '" << name << "' with ID " << resId << " but type '" << type->type << "' already has ID " << std::hex << (int) type->id.value() << std::dec); return false; } ResourceEntry* entry = type->findOrCreateEntry(name.entry); if (resId.isValid() && entry->id && entry->id.value() != resId.entryId()) { diag->error(DiagMessage(symbol.source) << "trying to add resource '" << name << "' with ID " << resId << " but resource already has ID " << ResourceId(package->id.value(), type->id.value(), entry->id.value())); return false; } if (resId.isValid()) { package->id = resId.packageId(); type->id = resId.typeId(); entry->id = resId.entryId(); } // Only mark the type state as public, it doesn't care about being private. if (symbol.state == SymbolState::kPublic) { type->symbolStatus.state = SymbolState::kPublic; } if (symbol.state == SymbolState::kUndefined && entry->symbolStatus.state != SymbolState::kUndefined) { // We can't undefine a symbol (remove its visibility). Ignore. return true; } if (symbol.state == SymbolState::kPrivate && entry->symbolStatus.state == SymbolState::kPublic) { // We can't downgrade public to private. Ignore. return true; } entry->symbolStatus = std::move(symbol); return true; } Maybe<ResourceTable::SearchResult> ResourceTable::findResource(const ResourceNameRef& name) { ResourceTablePackage* package = findPackage(name.package); if (!package) { return {}; } ResourceTableType* type = package->findType(name.type); if (!type) { return {}; } ResourceEntry* entry = type->findEntry(name.entry); if (!entry) { return {}; } return SearchResult{ package, type, entry }; } } // namespace aapt