/* * 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. */ #ifndef AAPT_RESOURCE_H #define AAPT_RESOURCE_H #include #include #include #include #include #include #include #include "androidfw/ConfigDescription.h" #include "androidfw/Source.h" #include "androidfw/StringPiece.h" #include "utils/JenkinsHash.h" namespace aapt { /** * The various types of resource types available. */ enum class ResourceType { kAnim, kAnimator, kArray, kAttr, kAttrPrivate, kBool, kColor, // Not really a type, but it shows up in some CTS tests and // we need to continue respecting it. kConfigVarying, kDimen, kDrawable, kFont, kFraction, kId, kInteger, kInterpolator, kLayout, kMacro, kMenu, kMipmap, kNavigation, kPlurals, kRaw, kString, kStyle, kStyleable, kTransition, kXml, }; android::StringPiece to_string(ResourceType type); /** * Returns a pointer to a valid ResourceType, or nullptr if the string was invalid. */ const ResourceType* ParseResourceType(android::StringPiece str); /** * Pair of type name as in ResourceTable and actual resource type. * Corresponds to the 'type' in package:type/entry. * * This is to support resource types with custom names inside resource tables. */ struct ResourceNamedType { std::string name; ResourceType type = ResourceType::kRaw; ResourceNamedType() = default; ResourceNamedType(android::StringPiece n, ResourceType t); int compare(const ResourceNamedType& other) const; const std::string& to_string() const; }; /** * Same as ResourceNamedType, but uses StringPieces instead. * Use this if you need to avoid copying and know that * the lifetime of this object is shorter than that * of the original string. */ struct ResourceNamedTypeRef { android::StringPiece name; ResourceType type = ResourceType::kRaw; ResourceNamedTypeRef() = default; ResourceNamedTypeRef(const ResourceNamedTypeRef&) = default; ResourceNamedTypeRef(ResourceNamedTypeRef&&) = default; ResourceNamedTypeRef(const ResourceNamedType& rhs); // NOLINT(google-explicit-constructor) ResourceNamedTypeRef(android::StringPiece n, ResourceType t); ResourceNamedTypeRef& operator=(const ResourceNamedTypeRef& rhs) = default; ResourceNamedTypeRef& operator=(ResourceNamedTypeRef&& rhs) = default; ResourceNamedTypeRef& operator=(const ResourceNamedType& rhs); ResourceNamedType ToResourceNamedType() const; std::string_view to_string() const; }; ResourceNamedTypeRef ResourceNamedTypeWithDefaultName(ResourceType t); std::optional ParseResourceNamedType(android::StringPiece s); /** * A resource's name. This can uniquely identify * a resource in the ResourceTable. */ struct ResourceName { std::string package; ResourceNamedType type; std::string entry; ResourceName() = default; ResourceName(android::StringPiece p, const ResourceNamedTypeRef& t, android::StringPiece e); ResourceName(android::StringPiece p, ResourceType t, android::StringPiece e); int compare(const ResourceName& other) const; bool is_valid() const; std::string to_string() const; }; /** * Same as ResourceName, but uses StringPieces instead. * Use this if you need to avoid copying and know that * the lifetime of this object is shorter than that * of the original string. */ struct ResourceNameRef { android::StringPiece package; ResourceNamedTypeRef type; android::StringPiece entry; ResourceNameRef() = default; ResourceNameRef(const ResourceNameRef&) = default; ResourceNameRef(ResourceNameRef&&) = default; ResourceNameRef(const ResourceName& rhs); // NOLINT(google-explicit-constructor) ResourceNameRef(android::StringPiece p, const ResourceNamedTypeRef& t, android::StringPiece e); ResourceNameRef(android::StringPiece p, ResourceType t, android::StringPiece e); ResourceNameRef& operator=(const ResourceNameRef& rhs) = default; ResourceNameRef& operator=(ResourceNameRef&& rhs) = default; ResourceNameRef& operator=(const ResourceName& rhs); bool is_valid() const; ResourceName ToResourceName() const; std::string to_string() const; }; constexpr const uint8_t kAppPackageId = 0x7fu; constexpr const uint8_t kFrameworkPackageId = 0x01u; /** * A binary identifier representing a resource. Internally it * is a 32bit integer split as follows: * * 0xPPTTEEEE * * PP: 8 bit package identifier. 0x01 is reserved for system * and 0x7f is reserved for the running app. * TT: 8 bit type identifier. 0x00 is invalid. * EEEE: 16 bit entry identifier. */ struct ResourceId { uint32_t id; ResourceId(); ResourceId(const ResourceId& rhs) = default; ResourceId(uint32_t res_id); // NOLINT(google-explicit-constructor) ResourceId(uint8_t p, uint8_t t, uint16_t e); // Returns true if the ID is a valid ID that is not dynamic (package ID cannot be 0) bool is_valid_static() const; // Returns true if the ID is a valid ID or dynamic ID (package ID can be 0). bool is_valid() const; uint8_t package_id() const; uint8_t type_id() const; uint16_t entry_id() const; std::string to_string() const; }; struct SourcedResourceName { ResourceName name; size_t line; }; struct ResourceFile { enum class Type { kUnknown, kPng, kBinaryXml, kProtoXml, }; // Name ResourceName name; // Configuration android::ConfigDescription config; // Type Type type; // Source android::Source source; // Exported symbols std::vector exported_symbols; }; /** * Useful struct used as a key to represent a unique resource in associative * containers. */ struct ResourceKey { ResourceName name; android::ConfigDescription config; }; bool operator<(const ResourceKey& a, const ResourceKey& b); /** * Useful struct used as a key to represent a unique resource in associative * containers. * Holds a reference to the name, so that name better live longer than this key! */ struct ResourceKeyRef { ResourceNameRef name; android::ConfigDescription config; ResourceKeyRef() = default; ResourceKeyRef(const ResourceNameRef& n, const android::ConfigDescription& c) : name(n), config(c) {} /** * Prevent taking a reference to a temporary. This is bad. */ ResourceKeyRef(ResourceName&& n, const android::ConfigDescription& c) = delete; }; bool operator<(const ResourceKeyRef& a, const ResourceKeyRef& b); // // ResourceId implementation. // inline ResourceId::ResourceId() : id(0) {} inline ResourceId::ResourceId(uint32_t res_id) : id(res_id) {} inline ResourceId::ResourceId(uint8_t p, uint8_t t, uint16_t e) : id((p << 24) | (t << 16) | e) {} inline bool ResourceId::is_valid_static() const { return (id & 0xff000000u) != 0 && (id & 0x00ff0000u) != 0; } inline bool ResourceId::is_valid() const { return (id & 0x00ff0000u) != 0; } inline uint8_t ResourceId::package_id() const { return static_cast(id >> 24); } inline uint8_t ResourceId::type_id() const { return static_cast(id >> 16); } inline uint16_t ResourceId::entry_id() const { return static_cast(id); } inline bool operator<(const ResourceId& lhs, const ResourceId& rhs) { return lhs.id < rhs.id; } inline bool operator>(const ResourceId& lhs, const ResourceId& rhs) { return lhs.id > rhs.id; } inline bool operator==(const ResourceId& lhs, const ResourceId& rhs) { return lhs.id == rhs.id; } inline bool operator!=(const ResourceId& lhs, const ResourceId& rhs) { return lhs.id != rhs.id; } inline ::std::ostream& operator<<(::std::ostream& out, const ResourceId& res_id) { return out << res_id.to_string(); } // For generic code to call 'using std::to_string; to_string(T);'. inline std::string to_string(const ResourceId& id) { return id.to_string(); } // Helper to compare resource IDs, moving dynamic IDs after framework IDs. inline bool cmp_ids_dynamic_after_framework(const ResourceId& a, const ResourceId& b) { // If one of a and b is from the framework package (package ID 0x01), and the // other is a dynamic ID (package ID 0x00), then put the dynamic ID after the // framework ID. This ensures that when AssetManager resolves the dynamic IDs, // they will be in sorted order as expected by AssetManager. if ((a.package_id() == kFrameworkPackageId && b.package_id() == 0x00) || (a.package_id() == 0x00 && b.package_id() == kFrameworkPackageId)) { return b < a; } return a < b; } // // ResourceType implementation. // inline ::std::ostream& operator<<(::std::ostream& out, const ResourceType& val) { return out << to_string(val); } // // ResourceNamedType implementation. // inline ResourceNamedType::ResourceNamedType(android::StringPiece n, ResourceType t) : name(n), type(t) { } inline int ResourceNamedType::compare(const ResourceNamedType& other) const { int cmp = static_cast(type) - static_cast(other.type); if (cmp != 0) return cmp; cmp = name.compare(other.name); return cmp; } inline const std::string& ResourceNamedType::to_string() const { return name; } inline bool operator<(const ResourceNamedType& lhs, const ResourceNamedType& rhs) { return lhs.compare(rhs) < 0; } inline bool operator==(const ResourceNamedType& lhs, const ResourceNamedType& rhs) { return lhs.compare(rhs) == 0; } inline bool operator!=(const ResourceNamedType& lhs, const ResourceNamedType& rhs) { return lhs.compare(rhs) != 0; } inline ::std::ostream& operator<<(::std::ostream& out, const ResourceNamedType& val) { return out << val.to_string(); } // // ResourceNamedTypeRef implementation. // inline ResourceNamedTypeRef::ResourceNamedTypeRef(android::StringPiece n, ResourceType t) : name(n), type(t) { } inline ResourceNamedTypeRef::ResourceNamedTypeRef(const ResourceNamedType& rhs) : name(rhs.name), type(rhs.type) { } inline ResourceNamedTypeRef& ResourceNamedTypeRef::operator=(const ResourceNamedType& rhs) { name = rhs.name; type = rhs.type; return *this; } inline ResourceNamedType ResourceNamedTypeRef::ToResourceNamedType() const { return ResourceNamedType(name, type); } inline std::string_view ResourceNamedTypeRef::to_string() const { return name; } inline bool operator<(const ResourceNamedTypeRef& lhs, const ResourceNamedTypeRef& rhs) { return std::tie(lhs.type, lhs.name) < std::tie(rhs.type, rhs.name); } inline bool operator==(const ResourceNamedTypeRef& lhs, const ResourceNamedTypeRef& rhs) { return std::tie(lhs.type, lhs.name) == std::tie(rhs.type, rhs.name); } inline bool operator!=(const ResourceNamedTypeRef& lhs, const ResourceNamedTypeRef& rhs) { return std::tie(lhs.type, lhs.name) != std::tie(rhs.type, rhs.name); } inline ::std::ostream& operator<<(::std::ostream& out, const ResourceNamedTypeRef& val) { return out << val.name; } // // ResourceName implementation. // inline ResourceName::ResourceName(android::StringPiece p, const ResourceNamedTypeRef& t, android::StringPiece e) : package(p), type(t.ToResourceNamedType()), entry(e) { } inline ResourceName::ResourceName(android::StringPiece p, ResourceType t, android::StringPiece e) : ResourceName(p, ResourceNamedTypeWithDefaultName(t), e) { } inline int ResourceName::compare(const ResourceName& other) const { int cmp = package.compare(other.package); if (cmp != 0) return cmp; cmp = type.compare(other.type); if (cmp != 0) return cmp; cmp = entry.compare(other.entry); return cmp; } inline bool ResourceName::is_valid() const { return !package.empty() && !entry.empty(); } inline bool operator<(const ResourceName& lhs, const ResourceName& rhs) { return std::tie(lhs.package, lhs.type, lhs.entry) < std::tie(rhs.package, rhs.type, rhs.entry); } inline bool operator==(const ResourceName& lhs, const ResourceName& rhs) { return std::tie(lhs.package, lhs.type, lhs.entry) == std::tie(rhs.package, rhs.type, rhs.entry); } inline bool operator!=(const ResourceName& lhs, const ResourceName& rhs) { return std::tie(lhs.package, lhs.type, lhs.entry) != std::tie(rhs.package, rhs.type, rhs.entry); } inline ::std::ostream& operator<<(::std::ostream& out, const ResourceName& name) { return out << name.to_string(); } // // ResourceNameRef implementation. // inline ResourceNameRef::ResourceNameRef(const ResourceName& rhs) : package(rhs.package), type(rhs.type), entry(rhs.entry) {} inline ResourceNameRef::ResourceNameRef(android::StringPiece p, const ResourceNamedTypeRef& t, android::StringPiece e) : package(p), type(t), entry(e) { } inline ResourceNameRef::ResourceNameRef(android::StringPiece p, ResourceType t, android::StringPiece e) : ResourceNameRef(p, ResourceNamedTypeWithDefaultName(t), e) { } inline ResourceNameRef& ResourceNameRef::operator=(const ResourceName& rhs) { package = rhs.package; type = rhs.type; entry = rhs.entry; return *this; } inline ResourceName ResourceNameRef::ToResourceName() const { return ResourceName(package, type, entry); } inline bool ResourceNameRef::is_valid() const { return !package.empty() && !entry.empty(); } inline bool operator<(const ResourceNameRef& lhs, const ResourceNameRef& rhs) { return std::tie(lhs.package, lhs.type, lhs.entry) < std::tie(rhs.package, rhs.type, rhs.entry); } inline bool operator==(const ResourceNameRef& lhs, const ResourceNameRef& rhs) { return std::tie(lhs.package, lhs.type, lhs.entry) == std::tie(rhs.package, rhs.type, rhs.entry); } inline bool operator!=(const ResourceNameRef& lhs, const ResourceNameRef& rhs) { return std::tie(lhs.package, lhs.type, lhs.entry) != std::tie(rhs.package, rhs.type, rhs.entry); } inline ::std::ostream& operator<<(::std::ostream& out, const ResourceNameRef& name) { return out << name.to_string(); } inline bool operator<(const ResourceName& lhs, const ResourceNameRef& b) { return ResourceNameRef(lhs) < b; } inline bool operator!=(const ResourceName& lhs, const ResourceNameRef& rhs) { return ResourceNameRef(lhs) != rhs; } inline bool operator==(const SourcedResourceName& lhs, const SourcedResourceName& rhs) { return lhs.name == rhs.name && lhs.line == rhs.line; } } // namespace aapt namespace std { template <> struct hash { size_t operator()(const aapt::ResourceName& name) const { android::hash_t h = 0; h = android::JenkinsHashMix(h, static_cast(hash()(name.package))); h = android::JenkinsHashMix(h, static_cast(hash()(name.type.name))); h = android::JenkinsHashMix(h, static_cast(hash()(name.entry))); return static_cast(h); } }; template <> struct hash { size_t operator()(const aapt::ResourceId& id) const { return id.id; } }; } // namespace std #endif // AAPT_RESOURCE_H