blob: 446fdd4dbc1b69148810e29aad7af148e87e52ce [file] [log] [blame]
Adam Lesinski6f6ceb72014-11-14 14:48:12 -08001/*
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#ifndef AAPT_RESOURCE_H
18#define AAPT_RESOURCE_H
19
Adam Lesinski6f6ceb72014-11-14 14:48:12 -080020#include <iomanip>
Adam Lesinskica2fc352015-04-03 12:08:26 -070021#include <limits>
Iurii Makhnodfbac392022-02-15 20:02:52 +000022#include <optional>
Adam Lesinskid0f116b2016-07-08 15:00:32 -070023#include <sstream>
Adam Lesinski6f6ceb72014-11-14 14:48:12 -080024#include <string>
25#include <tuple>
Adam Lesinski1ab598f2015-08-14 14:26:04 -070026#include <vector>
Adam Lesinski6f6ceb72014-11-14 14:48:12 -080027
MÃ¥rten Kongstad24c9aa62018-06-20 08:46:41 +020028#include "androidfw/ConfigDescription.h"
Jeremy Meyer56f36e82022-05-20 20:35:42 +000029#include "androidfw/Source.h"
Adam Lesinskid5083f62017-01-16 15:07:21 -080030#include "androidfw/StringPiece.h"
Adam Lesinskice5e56e2016-10-21 17:56:45 -070031#include "utils/JenkinsHash.h"
32
Adam Lesinski6f6ceb72014-11-14 14:48:12 -080033namespace aapt {
34
35/**
Iurii Makhnodfbac392022-02-15 20:02:52 +000036 * The various types of resource types available.
Adam Lesinski6f6ceb72014-11-14 14:48:12 -080037 */
38enum class ResourceType {
Adam Lesinskicacb28f2016-10-19 12:18:14 -070039 kAnim,
40 kAnimator,
41 kArray,
42 kAttr,
43 kAttrPrivate,
44 kBool,
45 kColor,
Adam Lesinski86d67df2017-01-31 13:47:27 -080046
47 // Not really a type, but it shows up in some CTS tests and
48 // we need to continue respecting it.
49 kConfigVarying,
50
Adam Lesinskicacb28f2016-10-19 12:18:14 -070051 kDimen,
52 kDrawable,
Adam Lesinskic0c36632016-10-19 18:37:53 -070053 kFont,
Adam Lesinskicacb28f2016-10-19 12:18:14 -070054 kFraction,
55 kId,
56 kInteger,
57 kInterpolator,
58 kLayout,
Ryan Mitchell326e35ff2021-04-12 07:50:42 -070059 kMacro,
Adam Lesinskicacb28f2016-10-19 12:18:14 -070060 kMenu,
61 kMipmap,
Adam Lesinski3b841242017-07-25 17:15:42 -070062 kNavigation,
Adam Lesinskicacb28f2016-10-19 12:18:14 -070063 kPlurals,
64 kRaw,
65 kString,
66 kStyle,
67 kStyleable,
68 kTransition,
69 kXml,
Adam Lesinski6f6ceb72014-11-14 14:48:12 -080070};
71
Jeremy Meyer211bec22024-06-04 14:22:03 -070072enum class FlagStatus { NoFlag = 0, Disabled = 1, Enabled = 2 };
73
Jeremy Meyer3d8d4a12024-08-23 17:29:03 -070074struct FeatureFlagAttribute {
75 std::string name;
76 bool negated = false;
77
78 std::string ToString() {
79 return (negated ? "!" : "") + name;
80 }
81
82 bool operator==(const FeatureFlagAttribute& o) const = default;
83};
84
Adam Lesinski93190b72017-11-03 15:20:17 -070085android::StringPiece to_string(ResourceType type);
Adam Lesinski6f6ceb72014-11-14 14:48:12 -080086
87/**
Adam Lesinski93190b72017-11-03 15:20:17 -070088 * Returns a pointer to a valid ResourceType, or nullptr if the string was invalid.
Adam Lesinski6f6ceb72014-11-14 14:48:12 -080089 */
Yurii Zubrytskyia5775142022-11-02 17:49:49 -070090const ResourceType* ParseResourceType(android::StringPiece str);
Adam Lesinski6f6ceb72014-11-14 14:48:12 -080091
92/**
Iurii Makhnodfbac392022-02-15 20:02:52 +000093 * Pair of type name as in ResourceTable and actual resource type.
94 * Corresponds to the 'type' in package:type/entry.
95 *
96 * This is to support resource types with custom names inside resource tables.
97 */
98struct ResourceNamedType {
99 std::string name;
100 ResourceType type = ResourceType::kRaw;
101
102 ResourceNamedType() = default;
Yurii Zubrytskyia5775142022-11-02 17:49:49 -0700103 ResourceNamedType(android::StringPiece n, ResourceType t);
Iurii Makhnodfbac392022-02-15 20:02:52 +0000104
105 int compare(const ResourceNamedType& other) const;
106
107 const std::string& to_string() const;
108};
109
110/**
111 * Same as ResourceNamedType, but uses StringPieces instead.
112 * Use this if you need to avoid copying and know that
113 * the lifetime of this object is shorter than that
114 * of the original string.
115 */
116struct ResourceNamedTypeRef {
117 android::StringPiece name;
118 ResourceType type = ResourceType::kRaw;
119
120 ResourceNamedTypeRef() = default;
121 ResourceNamedTypeRef(const ResourceNamedTypeRef&) = default;
122 ResourceNamedTypeRef(ResourceNamedTypeRef&&) = default;
123 ResourceNamedTypeRef(const ResourceNamedType& rhs); // NOLINT(google-explicit-constructor)
Yurii Zubrytskyia5775142022-11-02 17:49:49 -0700124 ResourceNamedTypeRef(android::StringPiece n, ResourceType t);
Iurii Makhnodfbac392022-02-15 20:02:52 +0000125 ResourceNamedTypeRef& operator=(const ResourceNamedTypeRef& rhs) = default;
126 ResourceNamedTypeRef& operator=(ResourceNamedTypeRef&& rhs) = default;
127 ResourceNamedTypeRef& operator=(const ResourceNamedType& rhs);
128
129 ResourceNamedType ToResourceNamedType() const;
130
Yurii Zubrytskyia5775142022-11-02 17:49:49 -0700131 std::string_view to_string() const;
Iurii Makhnodfbac392022-02-15 20:02:52 +0000132};
133
134ResourceNamedTypeRef ResourceNamedTypeWithDefaultName(ResourceType t);
135
Yurii Zubrytskyia5775142022-11-02 17:49:49 -0700136std::optional<ResourceNamedTypeRef> ParseResourceNamedType(android::StringPiece s);
Iurii Makhnodfbac392022-02-15 20:02:52 +0000137
138/**
Adam Lesinski6f6ceb72014-11-14 14:48:12 -0800139 * A resource's name. This can uniquely identify
140 * a resource in the ResourceTable.
141 */
142struct ResourceName {
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700143 std::string package;
Iurii Makhnocff10ce2022-02-15 19:33:50 +0000144 ResourceNamedType type;
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700145 std::string entry;
Adam Lesinski6f6ceb72014-11-14 14:48:12 -0800146
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700147 ResourceName() = default;
Yurii Zubrytskyia5775142022-11-02 17:49:49 -0700148 ResourceName(android::StringPiece p, const ResourceNamedTypeRef& t, android::StringPiece e);
149 ResourceName(android::StringPiece p, ResourceType t, android::StringPiece e);
Adam Lesinski9ba47d82015-10-13 11:37:10 -0700150
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700151 int compare(const ResourceName& other) const;
Adam Lesinski8197cc462016-08-19 12:16:49 -0700152
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700153 bool is_valid() const;
Adam Lesinski93190b72017-11-03 15:20:17 -0700154 std::string to_string() const;
Adam Lesinski6f6ceb72014-11-14 14:48:12 -0800155};
156
157/**
158 * Same as ResourceName, but uses StringPieces instead.
159 * Use this if you need to avoid copying and know that
160 * the lifetime of this object is shorter than that
161 * of the original string.
162 */
163struct ResourceNameRef {
Adam Lesinskid5083f62017-01-16 15:07:21 -0800164 android::StringPiece package;
Iurii Makhnocff10ce2022-02-15 19:33:50 +0000165 ResourceNamedTypeRef type;
Adam Lesinskid5083f62017-01-16 15:07:21 -0800166 android::StringPiece entry;
Adam Lesinski6f6ceb72014-11-14 14:48:12 -0800167
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700168 ResourceNameRef() = default;
169 ResourceNameRef(const ResourceNameRef&) = default;
170 ResourceNameRef(ResourceNameRef&&) = default;
Chih-Hung Hsieh1fc78e12018-12-20 13:37:44 -0800171 ResourceNameRef(const ResourceName& rhs); // NOLINT(google-explicit-constructor)
Yurii Zubrytskyia5775142022-11-02 17:49:49 -0700172 ResourceNameRef(android::StringPiece p, const ResourceNamedTypeRef& t, android::StringPiece e);
173 ResourceNameRef(android::StringPiece p, ResourceType t, android::StringPiece e);
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700174 ResourceNameRef& operator=(const ResourceNameRef& rhs) = default;
175 ResourceNameRef& operator=(ResourceNameRef&& rhs) = default;
176 ResourceNameRef& operator=(const ResourceName& rhs);
Adam Lesinski6f6ceb72014-11-14 14:48:12 -0800177
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700178 bool is_valid() const;
Adam Lesinski93190b72017-11-03 15:20:17 -0700179
180 ResourceName ToResourceName() const;
181 std::string to_string() const;
Adam Lesinski6f6ceb72014-11-14 14:48:12 -0800182};
183
Adam Lesinskib5dc4bd2017-02-22 19:29:29 -0800184constexpr const uint8_t kAppPackageId = 0x7fu;
185constexpr const uint8_t kFrameworkPackageId = 0x01u;
186
Adam Lesinski6f6ceb72014-11-14 14:48:12 -0800187/**
188 * A binary identifier representing a resource. Internally it
189 * is a 32bit integer split as follows:
190 *
191 * 0xPPTTEEEE
192 *
193 * PP: 8 bit package identifier. 0x01 is reserved for system
194 * and 0x7f is reserved for the running app.
195 * TT: 8 bit type identifier. 0x00 is invalid.
196 * EEEE: 16 bit entry identifier.
197 */
198struct ResourceId {
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700199 uint32_t id;
Adam Lesinski6f6ceb72014-11-14 14:48:12 -0800200
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700201 ResourceId();
Ryan Mitchell9634efb2021-03-19 14:53:17 -0700202 ResourceId(const ResourceId& rhs) = default;
Chih-Hung Hsieh1fc78e12018-12-20 13:37:44 -0800203 ResourceId(uint32_t res_id); // NOLINT(google-explicit-constructor)
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700204 ResourceId(uint8_t p, uint8_t t, uint16_t e);
Adam Lesinski6f6ceb72014-11-14 14:48:12 -0800205
Ryan Mitchellcd78feb2019-12-18 15:20:48 -0800206 // Returns true if the ID is a valid ID that is not dynamic (package ID cannot be 0)
207 bool is_valid_static() const;
Adam Lesinskiceb9b2f2017-02-16 12:05:42 -0800208
209 // Returns true if the ID is a valid ID or dynamic ID (package ID can be 0).
Ryan Mitchellcd78feb2019-12-18 15:20:48 -0800210 bool is_valid() const;
Adam Lesinskiceb9b2f2017-02-16 12:05:42 -0800211
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700212 uint8_t package_id() const;
213 uint8_t type_id() const;
214 uint16_t entry_id() const;
Adam Lesinski93190b72017-11-03 15:20:17 -0700215
216 std::string to_string() const;
Adam Lesinski6f6ceb72014-11-14 14:48:12 -0800217};
218
Adam Lesinski1ab598f2015-08-14 14:26:04 -0700219struct SourcedResourceName {
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700220 ResourceName name;
221 size_t line;
Adam Lesinski1ab598f2015-08-14 14:26:04 -0700222};
223
224struct ResourceFile {
Adam Lesinski00451162017-10-03 07:44:08 -0700225 enum class Type {
226 kUnknown,
227 kPng,
228 kBinaryXml,
229 kProtoXml,
230 };
231
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700232 // Name
233 ResourceName name;
Adam Lesinski1ab598f2015-08-14 14:26:04 -0700234
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700235 // Configuration
MÃ¥rten Kongstad24c9aa62018-06-20 08:46:41 +0200236 android::ConfigDescription config;
Adam Lesinski1ab598f2015-08-14 14:26:04 -0700237
Adam Lesinski00451162017-10-03 07:44:08 -0700238 // Type
239 Type type;
240
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700241 // Source
Jeremy Meyer56f36e82022-05-20 20:35:42 +0000242 android::Source source;
Adam Lesinski1ab598f2015-08-14 14:26:04 -0700243
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700244 // Exported symbols
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700245 std::vector<SourcedResourceName> exported_symbols;
Adam Lesinski1ab598f2015-08-14 14:26:04 -0700246};
247
Adam Lesinskia6fe3452015-12-09 15:20:52 -0800248/**
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700249 * Useful struct used as a key to represent a unique resource in associative
250 * containers.
Adam Lesinskia6fe3452015-12-09 15:20:52 -0800251 */
252struct ResourceKey {
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700253 ResourceName name;
MÃ¥rten Kongstad24c9aa62018-06-20 08:46:41 +0200254 android::ConfigDescription config;
Adam Lesinskia6fe3452015-12-09 15:20:52 -0800255};
256
257bool operator<(const ResourceKey& a, const ResourceKey& b);
258
259/**
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700260 * Useful struct used as a key to represent a unique resource in associative
261 * containers.
Adam Lesinskia6fe3452015-12-09 15:20:52 -0800262 * Holds a reference to the name, so that name better live longer than this key!
263 */
264struct ResourceKeyRef {
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700265 ResourceNameRef name;
MÃ¥rten Kongstad24c9aa62018-06-20 08:46:41 +0200266 android::ConfigDescription config;
Adam Lesinskia6fe3452015-12-09 15:20:52 -0800267
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700268 ResourceKeyRef() = default;
MÃ¥rten Kongstad24c9aa62018-06-20 08:46:41 +0200269 ResourceKeyRef(const ResourceNameRef& n, const android::ConfigDescription& c)
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700270 : name(n), config(c) {}
Adam Lesinskia6fe3452015-12-09 15:20:52 -0800271
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700272 /**
273 * Prevent taking a reference to a temporary. This is bad.
274 */
MÃ¥rten Kongstad24c9aa62018-06-20 08:46:41 +0200275 ResourceKeyRef(ResourceName&& n, const android::ConfigDescription& c) = delete;
Adam Lesinskia6fe3452015-12-09 15:20:52 -0800276};
277
278bool operator<(const ResourceKeyRef& a, const ResourceKeyRef& b);
279
Adam Lesinski6f6ceb72014-11-14 14:48:12 -0800280//
281// ResourceId implementation.
282//
283
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700284inline ResourceId::ResourceId() : id(0) {}
Adam Lesinski6f6ceb72014-11-14 14:48:12 -0800285
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700286inline ResourceId::ResourceId(uint32_t res_id) : id(res_id) {}
Adam Lesinski6f6ceb72014-11-14 14:48:12 -0800287
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700288inline ResourceId::ResourceId(uint8_t p, uint8_t t, uint16_t e)
289 : id((p << 24) | (t << 16) | e) {}
Adam Lesinski6f6ceb72014-11-14 14:48:12 -0800290
Ryan Mitchellcd78feb2019-12-18 15:20:48 -0800291inline bool ResourceId::is_valid_static() const {
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700292 return (id & 0xff000000u) != 0 && (id & 0x00ff0000u) != 0;
Adam Lesinski6f6ceb72014-11-14 14:48:12 -0800293}
294
Ryan Mitchellcd78feb2019-12-18 15:20:48 -0800295inline bool ResourceId::is_valid() const {
Adam Lesinski93190b72017-11-03 15:20:17 -0700296 return (id & 0x00ff0000u) != 0;
297}
Adam Lesinskiceb9b2f2017-02-16 12:05:42 -0800298
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700299inline uint8_t ResourceId::package_id() const {
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700300 return static_cast<uint8_t>(id >> 24);
Adam Lesinski6f6ceb72014-11-14 14:48:12 -0800301}
302
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700303inline uint8_t ResourceId::type_id() const {
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700304 return static_cast<uint8_t>(id >> 16);
Adam Lesinski6f6ceb72014-11-14 14:48:12 -0800305}
306
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700307inline uint16_t ResourceId::entry_id() const {
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700308 return static_cast<uint16_t>(id);
Adam Lesinski6f6ceb72014-11-14 14:48:12 -0800309}
310
Adam Lesinski74605cd2016-03-03 15:39:50 -0800311inline bool operator<(const ResourceId& lhs, const ResourceId& rhs) {
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700312 return lhs.id < rhs.id;
Adam Lesinski6f6ceb72014-11-14 14:48:12 -0800313}
314
Adam Lesinski74605cd2016-03-03 15:39:50 -0800315inline bool operator>(const ResourceId& lhs, const ResourceId& rhs) {
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700316 return lhs.id > rhs.id;
Adam Lesinski24aad162015-04-24 19:19:30 -0700317}
318
Adam Lesinski74605cd2016-03-03 15:39:50 -0800319inline bool operator==(const ResourceId& lhs, const ResourceId& rhs) {
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700320 return lhs.id == rhs.id;
Adam Lesinski74605cd2016-03-03 15:39:50 -0800321}
322
323inline bool operator!=(const ResourceId& lhs, const ResourceId& rhs) {
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700324 return lhs.id != rhs.id;
Adam Lesinski74605cd2016-03-03 15:39:50 -0800325}
326
Adam Lesinski93190b72017-11-03 15:20:17 -0700327inline ::std::ostream& operator<<(::std::ostream& out, const ResourceId& res_id) {
328 return out << res_id.to_string();
Adam Lesinski6f6ceb72014-11-14 14:48:12 -0800329}
330
Adam Lesinskia693c4a2017-11-09 11:29:39 -0800331// For generic code to call 'using std::to_string; to_string(T);'.
332inline std::string to_string(const ResourceId& id) {
333 return id.to_string();
334}
335
Clark DuVall8f51d6b2020-04-22 13:19:28 -0700336// Helper to compare resource IDs, moving dynamic IDs after framework IDs.
337inline bool cmp_ids_dynamic_after_framework(const ResourceId& a, const ResourceId& b) {
338 // If one of a and b is from the framework package (package ID 0x01), and the
339 // other is a dynamic ID (package ID 0x00), then put the dynamic ID after the
340 // framework ID. This ensures that when AssetManager resolves the dynamic IDs,
341 // they will be in sorted order as expected by AssetManager.
342 if ((a.package_id() == kFrameworkPackageId && b.package_id() == 0x00) ||
343 (a.package_id() == 0x00 && b.package_id() == kFrameworkPackageId)) {
344 return b < a;
345 }
346 return a < b;
347}
348
Adam Lesinski6f6ceb72014-11-14 14:48:12 -0800349//
350// ResourceType implementation.
351//
352
Adam Lesinski93190b72017-11-03 15:20:17 -0700353inline ::std::ostream& operator<<(::std::ostream& out, const ResourceType& val) {
354 return out << to_string(val);
Adam Lesinski6f6ceb72014-11-14 14:48:12 -0800355}
356
357//
Iurii Makhnodfbac392022-02-15 20:02:52 +0000358// ResourceNamedType implementation.
359//
Yurii Zubrytskyia5775142022-11-02 17:49:49 -0700360inline ResourceNamedType::ResourceNamedType(android::StringPiece n, ResourceType t)
361 : name(n), type(t) {
Iurii Makhnodfbac392022-02-15 20:02:52 +0000362}
363
364inline int ResourceNamedType::compare(const ResourceNamedType& other) const {
365 int cmp = static_cast<int>(type) - static_cast<int>(other.type);
366 if (cmp != 0) return cmp;
367 cmp = name.compare(other.name);
368 return cmp;
369}
370
371inline const std::string& ResourceNamedType::to_string() const {
372 return name;
373}
374
375inline bool operator<(const ResourceNamedType& lhs, const ResourceNamedType& rhs) {
376 return lhs.compare(rhs) < 0;
377}
378
379inline bool operator==(const ResourceNamedType& lhs, const ResourceNamedType& rhs) {
380 return lhs.compare(rhs) == 0;
381}
382
383inline bool operator!=(const ResourceNamedType& lhs, const ResourceNamedType& rhs) {
384 return lhs.compare(rhs) != 0;
385}
386
387inline ::std::ostream& operator<<(::std::ostream& out, const ResourceNamedType& val) {
388 return out << val.to_string();
389}
390
391//
392// ResourceNamedTypeRef implementation.
393//
Yurii Zubrytskyia5775142022-11-02 17:49:49 -0700394inline ResourceNamedTypeRef::ResourceNamedTypeRef(android::StringPiece n, ResourceType t)
Iurii Makhnodfbac392022-02-15 20:02:52 +0000395 : name(n), type(t) {
396}
397
398inline ResourceNamedTypeRef::ResourceNamedTypeRef(const ResourceNamedType& rhs)
399 : name(rhs.name), type(rhs.type) {
400}
401
402inline ResourceNamedTypeRef& ResourceNamedTypeRef::operator=(const ResourceNamedType& rhs) {
403 name = rhs.name;
404 type = rhs.type;
405 return *this;
406}
407
408inline ResourceNamedType ResourceNamedTypeRef::ToResourceNamedType() const {
409 return ResourceNamedType(name, type);
410}
411
Yurii Zubrytskyia5775142022-11-02 17:49:49 -0700412inline std::string_view ResourceNamedTypeRef::to_string() const {
413 return name;
Iurii Makhnodfbac392022-02-15 20:02:52 +0000414}
415
416inline bool operator<(const ResourceNamedTypeRef& lhs, const ResourceNamedTypeRef& rhs) {
417 return std::tie(lhs.type, lhs.name) < std::tie(rhs.type, rhs.name);
418}
419
420inline bool operator==(const ResourceNamedTypeRef& lhs, const ResourceNamedTypeRef& rhs) {
421 return std::tie(lhs.type, lhs.name) == std::tie(rhs.type, rhs.name);
422}
423
424inline bool operator!=(const ResourceNamedTypeRef& lhs, const ResourceNamedTypeRef& rhs) {
425 return std::tie(lhs.type, lhs.name) != std::tie(rhs.type, rhs.name);
426}
427
428inline ::std::ostream& operator<<(::std::ostream& out, const ResourceNamedTypeRef& val) {
429 return out << val.name;
430}
431
432//
Adam Lesinski6f6ceb72014-11-14 14:48:12 -0800433// ResourceName implementation.
434//
435
Yurii Zubrytskyia5775142022-11-02 17:49:49 -0700436inline ResourceName::ResourceName(android::StringPiece p, const ResourceNamedTypeRef& t,
437 android::StringPiece e)
438 : package(p), type(t.ToResourceNamedType()), entry(e) {
Iurii Makhnocff10ce2022-02-15 19:33:50 +0000439}
440
Yurii Zubrytskyia5775142022-11-02 17:49:49 -0700441inline ResourceName::ResourceName(android::StringPiece p, ResourceType t, android::StringPiece e)
Iurii Makhnocff10ce2022-02-15 19:33:50 +0000442 : ResourceName(p, ResourceNamedTypeWithDefaultName(t), e) {
443}
Adam Lesinski9ba47d82015-10-13 11:37:10 -0700444
Adam Lesinski8197cc462016-08-19 12:16:49 -0700445inline int ResourceName::compare(const ResourceName& other) const {
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700446 int cmp = package.compare(other.package);
447 if (cmp != 0) return cmp;
Iurii Makhnocff10ce2022-02-15 19:33:50 +0000448 cmp = type.compare(other.type);
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700449 if (cmp != 0) return cmp;
450 cmp = entry.compare(other.entry);
451 return cmp;
Adam Lesinski8197cc462016-08-19 12:16:49 -0700452}
453
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700454inline bool ResourceName::is_valid() const {
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700455 return !package.empty() && !entry.empty();
Adam Lesinski6f6ceb72014-11-14 14:48:12 -0800456}
457
Adam Lesinski74605cd2016-03-03 15:39:50 -0800458inline bool operator<(const ResourceName& lhs, const ResourceName& rhs) {
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700459 return std::tie(lhs.package, lhs.type, lhs.entry) <
460 std::tie(rhs.package, rhs.type, rhs.entry);
Adam Lesinski6f6ceb72014-11-14 14:48:12 -0800461}
462
Adam Lesinski74605cd2016-03-03 15:39:50 -0800463inline bool operator==(const ResourceName& lhs, const ResourceName& rhs) {
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700464 return std::tie(lhs.package, lhs.type, lhs.entry) ==
465 std::tie(rhs.package, rhs.type, rhs.entry);
Adam Lesinski6f6ceb72014-11-14 14:48:12 -0800466}
467
Adam Lesinski74605cd2016-03-03 15:39:50 -0800468inline bool operator!=(const ResourceName& lhs, const ResourceName& rhs) {
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700469 return std::tie(lhs.package, lhs.type, lhs.entry) !=
470 std::tie(rhs.package, rhs.type, rhs.entry);
Adam Lesinski6f6ceb72014-11-14 14:48:12 -0800471}
472
Adam Lesinski93190b72017-11-03 15:20:17 -0700473inline ::std::ostream& operator<<(::std::ostream& out, const ResourceName& name) {
474 return out << name.to_string();
Adam Lesinskid0f116b2016-07-08 15:00:32 -0700475}
Adam Lesinski769de982015-04-10 19:43:55 -0700476
Adam Lesinski6f6ceb72014-11-14 14:48:12 -0800477//
478// ResourceNameRef implementation.
479//
480
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700481inline ResourceNameRef::ResourceNameRef(const ResourceName& rhs)
482 : package(rhs.package), type(rhs.type), entry(rhs.entry) {}
Adam Lesinski6f6ceb72014-11-14 14:48:12 -0800483
Yurii Zubrytskyia5775142022-11-02 17:49:49 -0700484inline ResourceNameRef::ResourceNameRef(android::StringPiece p, const ResourceNamedTypeRef& t,
485 android::StringPiece e)
Iurii Makhnocff10ce2022-02-15 19:33:50 +0000486 : package(p), type(t), entry(e) {
487}
488
Yurii Zubrytskyia5775142022-11-02 17:49:49 -0700489inline ResourceNameRef::ResourceNameRef(android::StringPiece p, ResourceType t,
490 android::StringPiece e)
Iurii Makhnocff10ce2022-02-15 19:33:50 +0000491 : ResourceNameRef(p, ResourceNamedTypeWithDefaultName(t), e) {
492}
Adam Lesinski6f6ceb72014-11-14 14:48:12 -0800493
494inline ResourceNameRef& ResourceNameRef::operator=(const ResourceName& rhs) {
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700495 package = rhs.package;
496 type = rhs.type;
497 entry = rhs.entry;
498 return *this;
Adam Lesinski6f6ceb72014-11-14 14:48:12 -0800499}
500
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700501inline ResourceName ResourceNameRef::ToResourceName() const {
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700502 return ResourceName(package, type, entry);
Adam Lesinski6f6ceb72014-11-14 14:48:12 -0800503}
504
Adam Lesinskice5e56e2016-10-21 17:56:45 -0700505inline bool ResourceNameRef::is_valid() const {
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700506 return !package.empty() && !entry.empty();
Adam Lesinski6f6ceb72014-11-14 14:48:12 -0800507}
508
Adam Lesinski74605cd2016-03-03 15:39:50 -0800509inline bool operator<(const ResourceNameRef& lhs, const ResourceNameRef& rhs) {
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700510 return std::tie(lhs.package, lhs.type, lhs.entry) <
511 std::tie(rhs.package, rhs.type, rhs.entry);
Adam Lesinski6f6ceb72014-11-14 14:48:12 -0800512}
513
Adam Lesinski74605cd2016-03-03 15:39:50 -0800514inline bool operator==(const ResourceNameRef& lhs, const ResourceNameRef& rhs) {
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700515 return std::tie(lhs.package, lhs.type, lhs.entry) ==
516 std::tie(rhs.package, rhs.type, rhs.entry);
Adam Lesinski6f6ceb72014-11-14 14:48:12 -0800517}
518
Adam Lesinski74605cd2016-03-03 15:39:50 -0800519inline bool operator!=(const ResourceNameRef& lhs, const ResourceNameRef& rhs) {
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700520 return std::tie(lhs.package, lhs.type, lhs.entry) !=
521 std::tie(rhs.package, rhs.type, rhs.entry);
Adam Lesinski6f6ceb72014-11-14 14:48:12 -0800522}
523
Adam Lesinski93190b72017-11-03 15:20:17 -0700524inline ::std::ostream& operator<<(::std::ostream& out, const ResourceNameRef& name) {
525 return out << name.to_string();
Adam Lesinski6f6ceb72014-11-14 14:48:12 -0800526}
527
Adam Lesinski74605cd2016-03-03 15:39:50 -0800528inline bool operator<(const ResourceName& lhs, const ResourceNameRef& b) {
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700529 return ResourceNameRef(lhs) < b;
Adam Lesinski74605cd2016-03-03 15:39:50 -0800530}
531
532inline bool operator!=(const ResourceName& lhs, const ResourceNameRef& rhs) {
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700533 return ResourceNameRef(lhs) != rhs;
Adam Lesinski74605cd2016-03-03 15:39:50 -0800534}
535
Adam Lesinski93190b72017-11-03 15:20:17 -0700536inline bool operator==(const SourcedResourceName& lhs, const SourcedResourceName& rhs) {
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700537 return lhs.name == rhs.name && lhs.line == rhs.line;
Adam Lesinski74605cd2016-03-03 15:39:50 -0800538}
539
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700540} // namespace aapt
Adam Lesinski6f6ceb72014-11-14 14:48:12 -0800541
Adam Lesinskibf0bd0f2016-06-01 15:31:50 -0700542namespace std {
543
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700544template <>
545struct hash<aapt::ResourceName> {
546 size_t operator()(const aapt::ResourceName& name) const {
547 android::hash_t h = 0;
Adam Lesinskic744ae82017-05-17 19:28:38 -0700548 h = android::JenkinsHashMix(h, static_cast<uint32_t>(hash<string>()(name.package)));
Iurii Makhnocff10ce2022-02-15 19:33:50 +0000549 h = android::JenkinsHashMix(h, static_cast<uint32_t>(hash<string>()(name.type.name)));
Adam Lesinskic744ae82017-05-17 19:28:38 -0700550 h = android::JenkinsHashMix(h, static_cast<uint32_t>(hash<string>()(name.entry)));
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700551 return static_cast<size_t>(h);
552 }
Adam Lesinskibf0bd0f2016-06-01 15:31:50 -0700553};
554
Adam Lesinskic744ae82017-05-17 19:28:38 -0700555template <>
556struct hash<aapt::ResourceId> {
557 size_t operator()(const aapt::ResourceId& id) const {
558 return id.id;
559 }
560};
561
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700562} // namespace std
Adam Lesinskibf0bd0f2016-06-01 15:31:50 -0700563
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700564#endif // AAPT_RESOURCE_H