blob: 9530c1750c79baa15c3a42e278f4c40f6406236d [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_TABLE_H
18#define AAPT_RESOURCE_TABLE_H
19
Adam Lesinski458b8772016-04-25 14:20:21 -070020#include <functional>
Adam Lesinskie4bb9eb2016-02-12 22:18:51 -080021#include <map>
Adam Lesinski6f6ceb72014-11-14 14:48:12 -080022#include <memory>
23#include <string>
24#include <tuple>
Adam Lesinskie4bb9eb2016-02-12 22:18:51 -080025#include <unordered_map>
Adam Lesinski6f6ceb72014-11-14 14:48:12 -080026#include <vector>
27
Jeremy Meyer56f36e82022-05-20 20:35:42 +000028#include "Resource.h"
29#include "ResourceValues.h"
30#include "android-base/macros.h"
31#include "androidfw/ConfigDescription.h"
32#include "androidfw/IDiagnostics.h"
33#include "androidfw/Source.h"
34#include "androidfw/StringPiece.h"
35#include "androidfw/StringPool.h"
36#include "io/File.h"
37
Winson62ac8b52019-12-04 08:36:48 -080038using PolicyFlags = android::ResTable_overlayable_policy_header::PolicyFlags;
39
Adam Lesinski6f6ceb72014-11-14 14:48:12 -080040namespace aapt {
41
Adam Lesinski71be7052017-12-12 16:48:07 -080042// The Public status of a resource.
43struct Visibility {
44 enum class Level {
45 kUndefined,
46 kPrivate,
47 kPublic,
48 };
49
50 Level level = Level::kUndefined;
Jeremy Meyer56f36e82022-05-20 20:35:42 +000051 android::Source source;
Adam Lesinski71be7052017-12-12 16:48:07 -080052 std::string comment;
Ryan Mitchell2e9bec12021-03-22 09:31:00 -070053
54 // Indicates that the resource id may change across builds and that the public R.java identifier
55 // for this resource should not be final. This is set to `true` for resources in `staging-group`
56 // tags.
57 bool staged_api = false;
Adam Lesinski9e10ac72015-10-16 14:37:48 -070058};
59
Adam Lesinski71be7052017-12-12 16:48:07 -080060// Represents <add-resource> in an overlay.
61struct AllowNew {
Jeremy Meyer56f36e82022-05-20 20:35:42 +000062 android::Source source;
Adam Lesinski71be7052017-12-12 16:48:07 -080063 std::string comment;
64};
Adam Lesinski4488f1c2017-05-26 17:33:38 -070065
Ryan Mitchell2fedba92021-04-23 07:47:38 -070066// Represents the staged resource id of a finalized resource.
67struct StagedId {
68 ResourceId id;
Jeremy Meyer56f36e82022-05-20 20:35:42 +000069 android::Source source;
Ryan Mitchell2fedba92021-04-23 07:47:38 -070070};
71
Adam Lesinski71be7052017-12-12 16:48:07 -080072struct Overlayable {
Ryan Mitchell54237ff2018-12-13 15:44:29 -080073 Overlayable() = default;
Yurii Zubrytskyia5775142022-11-02 17:49:49 -070074 Overlayable(android::StringPiece name, android::StringPiece actor) : name(name), actor(actor) {
75 }
76 Overlayable(android::StringPiece name, android::StringPiece actor, const android::Source& source)
77 : name(name), actor(actor), source(source) {
78 }
Ryan Mitchell54237ff2018-12-13 15:44:29 -080079
80 static const char* kActorScheme;
81 std::string name;
82 std::string actor;
Jeremy Meyer56f36e82022-05-20 20:35:42 +000083 android::Source source;
Ryan Mitchell54237ff2018-12-13 15:44:29 -080084};
85
86// Represents a declaration that a resource is overlayable at runtime.
87struct OverlayableItem {
88 explicit OverlayableItem(const std::shared_ptr<Overlayable>& overlayable)
89 : overlayable(overlayable) {}
Ryan Mitchell54237ff2018-12-13 15:44:29 -080090 std::shared_ptr<Overlayable> overlayable;
Winson62ac8b52019-12-04 08:36:48 -080091 PolicyFlags policies = PolicyFlags::NONE;
Adam Lesinskicacb28f2016-10-19 12:18:14 -070092 std::string comment;
Jeremy Meyer56f36e82022-05-20 20:35:42 +000093 android::Source source;
Adam Lesinski6f6ceb72014-11-14 14:48:12 -080094};
95
Adam Lesinskie4bb9eb2016-02-12 22:18:51 -080096class ResourceConfigValue {
Adam Lesinskicacb28f2016-10-19 12:18:14 -070097 public:
Adam Lesinski71be7052017-12-12 16:48:07 -080098 // The configuration for which this value is defined.
MÃ¥rten Kongstad24c9aa62018-06-20 08:46:41 +020099 const android::ConfigDescription config;
Adam Lesinskie4bb9eb2016-02-12 22:18:51 -0800100
Adam Lesinski71be7052017-12-12 16:48:07 -0800101 // The product for which this value is defined.
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700102 const std::string product;
Adam Lesinskie4bb9eb2016-02-12 22:18:51 -0800103
Adam Lesinski71be7052017-12-12 16:48:07 -0800104 // The actual Value.
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700105 std::unique_ptr<Value> value;
Adam Lesinskie4bb9eb2016-02-12 22:18:51 -0800106
Jeremy Meyer211bec22024-06-04 14:22:03 -0700107 FlagStatus flag_status;
108
Yurii Zubrytskyia5775142022-11-02 17:49:49 -0700109 ResourceConfigValue(const android::ConfigDescription& config, android::StringPiece product)
110 : config(config), product(product) {
111 }
Adam Lesinskie4bb9eb2016-02-12 22:18:51 -0800112
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700113 private:
114 DISALLOW_COPY_AND_ASSIGN(ResourceConfigValue);
Adam Lesinski6f6ceb72014-11-14 14:48:12 -0800115};
116
Adam Lesinski73bff1e2017-12-08 16:06:10 -0800117// Represents a resource entry, which may have varying values for each defined configuration.
Adam Lesinskie4bb9eb2016-02-12 22:18:51 -0800118class ResourceEntry {
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700119 public:
Adam Lesinski71be7052017-12-12 16:48:07 -0800120 // The name of the resource. Immutable, as this determines the order of this resource
121 // when doing lookups.
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700122 const std::string name;
Adam Lesinski6f6ceb72014-11-14 14:48:12 -0800123
Adam Lesinski71be7052017-12-12 16:48:07 -0800124 // The entry ID for this resource (the EEEE in 0xPPTTEEEE).
Ryan Mitchell4382e442021-07-14 12:53:01 -0700125 std::optional<ResourceId> id;
Adam Lesinski6f6ceb72014-11-14 14:48:12 -0800126
Adam Lesinski71be7052017-12-12 16:48:07 -0800127 // Whether this resource is public (and must maintain the same entry ID across builds).
128 Visibility visibility;
Adam Lesinski6f6ceb72014-11-14 14:48:12 -0800129
Ryan Mitchell4382e442021-07-14 12:53:01 -0700130 std::optional<AllowNew> allow_new;
Adam Lesinski71be7052017-12-12 16:48:07 -0800131
Ryan Mitchelle4e989c2018-10-29 02:21:50 -0700132 // The declarations of this resource as overlayable for RROs
Ryan Mitchell4382e442021-07-14 12:53:01 -0700133 std::optional<OverlayableItem> overlayable_item;
Adam Lesinski71be7052017-12-12 16:48:07 -0800134
Ryan Mitchell2fedba92021-04-23 07:47:38 -0700135 // The staged resource id for a finalized resource.
Ryan Mitchell4382e442021-07-14 12:53:01 -0700136 std::optional<StagedId> staged_id;
Ryan Mitchell2fedba92021-04-23 07:47:38 -0700137
Adam Lesinski71be7052017-12-12 16:48:07 -0800138 // The resource's values for each configuration.
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700139 std::vector<std::unique_ptr<ResourceConfigValue>> values;
Adam Lesinski6f6ceb72014-11-14 14:48:12 -0800140
Yurii Zubrytskyia5775142022-11-02 17:49:49 -0700141 explicit ResourceEntry(android::StringPiece name) : name(name) {
142 }
Adam Lesinskie4bb9eb2016-02-12 22:18:51 -0800143
MÃ¥rten Kongstad24c9aa62018-06-20 08:46:41 +0200144 ResourceConfigValue* FindValue(const android::ConfigDescription& config,
Ryan Mitchell9634efb2021-03-19 14:53:17 -0700145 android::StringPiece product = {});
146 const ResourceConfigValue* FindValue(const android::ConfigDescription& config,
147 android::StringPiece product = {}) const;
Adam Lesinski34a16872018-02-23 16:18:10 -0800148
MÃ¥rten Kongstad24c9aa62018-06-20 08:46:41 +0200149 ResourceConfigValue* FindOrCreateValue(const android::ConfigDescription& config,
Yurii Zubrytskyia5775142022-11-02 17:49:49 -0700150 android::StringPiece product);
MÃ¥rten Kongstad24c9aa62018-06-20 08:46:41 +0200151 std::vector<ResourceConfigValue*> FindAllValues(const android::ConfigDescription& config);
Adam Lesinski34a16872018-02-23 16:18:10 -0800152
153 template <typename Func>
154 std::vector<ResourceConfigValue*> FindValuesIf(Func f) {
155 std::vector<ResourceConfigValue*> results;
156 for (auto& config_value : values) {
157 if (f(config_value.get())) {
158 results.push_back(config_value.get());
159 }
160 }
161 return results;
162 }
163
164 bool HasDefaultValue() const;
Adam Lesinski458b8772016-04-25 14:20:21 -0700165
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700166 private:
167 DISALLOW_COPY_AND_ASSIGN(ResourceEntry);
Adam Lesinski6f6ceb72014-11-14 14:48:12 -0800168};
169
Adam Lesinski71be7052017-12-12 16:48:07 -0800170// Represents a resource type (eg. string, drawable, layout, etc.) containing resource entries.
Adam Lesinskie4bb9eb2016-02-12 22:18:51 -0800171class ResourceTableType {
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700172 public:
Adam Lesinski71be7052017-12-12 16:48:07 -0800173 // The logical type of resource (string, drawable, layout, etc.).
Iurii Makhnof0c5ff42022-02-22 13:31:02 +0000174 const ResourceNamedType named_type;
Adam Lesinski6f6ceb72014-11-14 14:48:12 -0800175
Adam Lesinski71be7052017-12-12 16:48:07 -0800176 // Whether this type is public (and must maintain the same type ID across builds).
177 Visibility::Level visibility_level = Visibility::Level::kUndefined;
Adam Lesinski6f6ceb72014-11-14 14:48:12 -0800178
Adam Lesinski71be7052017-12-12 16:48:07 -0800179 // List of resources for this type.
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700180 std::vector<std::unique_ptr<ResourceEntry>> entries;
Adam Lesinski6f6ceb72014-11-14 14:48:12 -0800181
Iurii Makhnof0c5ff42022-02-22 13:31:02 +0000182 explicit ResourceTableType(const ResourceNamedTypeRef& type)
183 : named_type(type.ToResourceNamedType()) {
184 }
Adam Lesinski1ab598f2015-08-14 14:26:04 -0700185
Yurii Zubrytskyia5775142022-11-02 17:49:49 -0700186 ResourceEntry* CreateEntry(android::StringPiece name);
187 ResourceEntry* FindEntry(android::StringPiece name) const;
188 ResourceEntry* FindOrCreateEntry(android::StringPiece name);
Adam Lesinskie4bb9eb2016-02-12 22:18:51 -0800189
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700190 private:
191 DISALLOW_COPY_AND_ASSIGN(ResourceTableType);
Adam Lesinski1ab598f2015-08-14 14:26:04 -0700192};
193
Adam Lesinskie4bb9eb2016-02-12 22:18:51 -0800194class ResourceTablePackage {
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700195 public:
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700196 std::string name;
Adam Lesinski1ab598f2015-08-14 14:26:04 -0700197
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700198 std::vector<std::unique_ptr<ResourceTableType>> types;
Adam Lesinski1ab598f2015-08-14 14:26:04 -0700199
Yurii Zubrytskyia5775142022-11-02 17:49:49 -0700200 explicit ResourceTablePackage(android::StringPiece name) : name(name) {
Ryan Mitchell9634efb2021-03-19 14:53:17 -0700201 }
202
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700203 ResourceTablePackage() = default;
Iurii Makhnof0c5ff42022-02-22 13:31:02 +0000204 ResourceTableType* FindTypeWithDefaultName(const ResourceType type) const;
205 ResourceTableType* FindType(const ResourceNamedTypeRef& type) const;
206 ResourceTableType* FindOrCreateType(const ResourceNamedTypeRef& type);
Adam Lesinskie4bb9eb2016-02-12 22:18:51 -0800207
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700208 private:
209 DISALLOW_COPY_AND_ASSIGN(ResourceTablePackage);
Adam Lesinski6f6ceb72014-11-14 14:48:12 -0800210};
211
Ryan Mitchell2fedba92021-04-23 07:47:38 -0700212struct ResourceTableEntryView {
213 std::string name;
Ryan Mitchell4382e442021-07-14 12:53:01 -0700214 std::optional<uint16_t> id;
Ryan Mitchell2fedba92021-04-23 07:47:38 -0700215 Visibility visibility;
Ryan Mitchell4382e442021-07-14 12:53:01 -0700216 std::optional<AllowNew> allow_new;
217 std::optional<OverlayableItem> overlayable_item;
218 std::optional<StagedId> staged_id;
Ryan Mitchell2fedba92021-04-23 07:47:38 -0700219 std::vector<const ResourceConfigValue*> values;
220
221 const ResourceConfigValue* FindValue(const android::ConfigDescription& config,
222 android::StringPiece product = {}) const;
223};
224
Ryan Mitchell9634efb2021-03-19 14:53:17 -0700225struct ResourceTableTypeView {
Iurii Makhnof0c5ff42022-02-22 13:31:02 +0000226 ResourceNamedType named_type;
Ryan Mitchell4382e442021-07-14 12:53:01 -0700227 std::optional<uint8_t> id;
Ryan Mitchell9634efb2021-03-19 14:53:17 -0700228 Visibility::Level visibility_level = Visibility::Level::kUndefined;
229
230 // Entries sorted in ascending entry id order. If ids have not been assigned, the entries are
Ryan Mitchell2fedba92021-04-23 07:47:38 -0700231 // sorted lexicographically.
232 std::vector<ResourceTableEntryView> entries;
Ryan Mitchell9634efb2021-03-19 14:53:17 -0700233};
234
235struct ResourceTablePackageView {
236 std::string name;
Ryan Mitchell4382e442021-07-14 12:53:01 -0700237 std::optional<uint8_t> id;
Ryan Mitchell9634efb2021-03-19 14:53:17 -0700238 // Types sorted in ascending type id order. If ids have not been assigned, the types are sorted by
239 // their declaration order in the ResourceType enum.
240 std::vector<ResourceTableTypeView> types;
241};
242
Ryan Mitchell2fedba92021-04-23 07:47:38 -0700243struct ResourceTableViewOptions {
244 bool create_alias_entries = false;
245};
246
Ryan Mitchell9634efb2021-03-19 14:53:17 -0700247struct ResourceTableView {
248 // Packages sorted in ascending package id order. If ids have not been assigned, the packages are
249 // sorted lexicographically.
250 std::vector<ResourceTablePackageView> packages;
251};
252
253enum class OnIdConflict {
254 // If the resource entry already exists but has a different resource id, the resource value will
255 // not be added to the table.
256 ERROR,
257
258 // If the resource entry already exists but has a different resource id, create a new resource
259 // with this resource name and id combination.
260 CREATE_ENTRY,
261};
262
263struct NewResource {
264 ResourceName name;
265 std::unique_ptr<Value> value;
266 android::ConfigDescription config;
267 std::string product;
268 std::optional<std::pair<ResourceId, OnIdConflict>> id;
269 std::optional<Visibility> visibility;
270 std::optional<OverlayableItem> overlayable;
271 std::optional<AllowNew> allow_new;
Ryan Mitchell2fedba92021-04-23 07:47:38 -0700272 std::optional<StagedId> staged_id;
Ryan Mitchell9634efb2021-03-19 14:53:17 -0700273 bool allow_mangled = false;
Jeremy Meyer211bec22024-06-04 14:22:03 -0700274 FlagStatus flag_status;
Ryan Mitchell9634efb2021-03-19 14:53:17 -0700275};
276
277struct NewResourceBuilder {
278 explicit NewResourceBuilder(const ResourceNameRef& name);
279 explicit NewResourceBuilder(const std::string& name);
280 NewResourceBuilder& SetValue(std::unique_ptr<Value> value, android::ConfigDescription config = {},
281 std::string product = {});
282 NewResourceBuilder& SetId(ResourceId id, OnIdConflict on_conflict = OnIdConflict::ERROR);
283 NewResourceBuilder& SetVisibility(Visibility id);
284 NewResourceBuilder& SetOverlayable(OverlayableItem overlayable);
285 NewResourceBuilder& SetAllowNew(AllowNew allow_new);
Ryan Mitchell2fedba92021-04-23 07:47:38 -0700286 NewResourceBuilder& SetStagedId(StagedId id);
Ryan Mitchell9634efb2021-03-19 14:53:17 -0700287 NewResourceBuilder& SetAllowMangled(bool allow_mangled);
Jeremy Meyer211bec22024-06-04 14:22:03 -0700288 NewResourceBuilder& SetFlagStatus(FlagStatus flag_status);
Ryan Mitchell9634efb2021-03-19 14:53:17 -0700289 NewResource Build();
290
291 private:
292 NewResource res_;
293};
294
Adam Lesinski71be7052017-12-12 16:48:07 -0800295// The container and index for all resources defined for an app.
Adam Lesinski6f6ceb72014-11-14 14:48:12 -0800296class ResourceTable {
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700297 public:
Ryan Mitchell9634efb2021-03-19 14:53:17 -0700298 enum class Validation {
299 kEnabled,
300 kDisabled,
301 };
Adam Lesinski6f6ceb72014-11-14 14:48:12 -0800302
Ryan Mitchell83a37ad2018-08-06 16:32:24 -0700303 enum class CollisionResult { kKeepBoth, kKeepOriginal, kConflict, kTakeNew };
Adam Lesinski5c3464c2016-08-24 16:03:48 -0700304
Ryan Mitchell9634efb2021-03-19 14:53:17 -0700305 ResourceTable() = default;
306 explicit ResourceTable(Validation validation);
Adam Lesinski5c3464c2016-08-24 16:03:48 -0700307
Jeremy Meyer56f36e82022-05-20 20:35:42 +0000308 bool AddResource(NewResource&& res, android::IDiagnostics* diag);
Adam Lesinski6f6ceb72014-11-14 14:48:12 -0800309
Ryan Mitchell9634efb2021-03-19 14:53:17 -0700310 // Retrieves a sorted a view of the packages, types, and entries sorted in ascending resource id
311 // order.
Ryan Mitchell2fedba92021-04-23 07:47:38 -0700312 ResourceTableView GetPartitionedView(const ResourceTableViewOptions& options = {}) const;
Adam Lesinski6f6ceb72014-11-14 14:48:12 -0800313
Yurii Zubrytskyicf91ab82023-04-24 18:34:13 -0700314 using ReferencedPackages = std::map<uint8_t, std::string>;
315 const ReferencedPackages& GetReferencedPackages() const {
316 return included_packages_;
317 }
318
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700319 struct SearchResult {
320 ResourceTablePackage* package;
321 ResourceTableType* type;
322 ResourceEntry* entry;
323 };
Adam Lesinski1ab598f2015-08-14 14:26:04 -0700324
Ryan Mitchell4382e442021-07-14 12:53:01 -0700325 std::optional<SearchResult> FindResource(const ResourceNameRef& name) const;
326 std::optional<SearchResult> FindResource(const ResourceNameRef& name, ResourceId id) const;
Ryan Mitchell2fedba92021-04-23 07:47:38 -0700327 bool RemoveResource(const ResourceNameRef& name, ResourceId id) const;
Adam Lesinski1ab598f2015-08-14 14:26:04 -0700328
Adam Lesinski71be7052017-12-12 16:48:07 -0800329 // Returns the package struct with the given name, or nullptr if such a package does not
330 // exist. The empty string is a valid package and typically is used to represent the
331 // 'current' package before it is known to the ResourceTable.
Yurii Zubrytskyia5775142022-11-02 17:49:49 -0700332 ResourceTablePackage* FindPackage(android::StringPiece name) const;
333 ResourceTablePackage* FindOrCreatePackage(android::StringPiece name);
David Chaloupkae3c1a4a2018-01-18 13:44:36 +0000334
Shane Farmer0a5b2012017-06-22 12:24:12 -0700335 std::unique_ptr<ResourceTable> Clone() const;
336
Jeremy Meyer211bec22024-06-04 14:22:03 -0700337 // When a collision of resources occurs, these methods decide which value to keep.
338 static CollisionResult ResolveFlagCollision(FlagStatus existing, FlagStatus incoming);
Ryan Mitchell9634efb2021-03-19 14:53:17 -0700339 static CollisionResult ResolveValueCollision(Value* existing, Value* incoming);
340
Adam Lesinski71be7052017-12-12 16:48:07 -0800341 // The string pool used by this resource table. Values that reference strings must use
342 // this pool to create their strings.
343 // NOTE: `string_pool` must come before `packages` so that it is destroyed after.
344 // When `string_pool` references are destroyed (as they will be when `packages` is destroyed),
345 // they decrement a refCount, which would cause invalid memory access if the pool was already
346 // destroyed.
Jeremy Meyer56f36e82022-05-20 20:35:42 +0000347 android::StringPool string_pool;
Adam Lesinski6f6ceb72014-11-14 14:48:12 -0800348
David Chaloupkae3c1a4a2018-01-18 13:44:36 +0000349 // The list of packages in this table, sorted alphabetically by package name and increasing
350 // package ID (missing ID being the lowest).
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700351 std::vector<std::unique_ptr<ResourceTablePackage>> packages;
Adam Lesinski6f6ceb72014-11-14 14:48:12 -0800352
Adam Lesinskiceb9b2f2017-02-16 12:05:42 -0800353 // Set of dynamic packages that this table may reference. Their package names get encoded
354 // into the resources.arsc along with their compile-time assigned IDs.
Yurii Zubrytskyicf91ab82023-04-24 18:34:13 -0700355 ReferencedPackages included_packages_;
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700356
357 private:
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700358 DISALLOW_COPY_AND_ASSIGN(ResourceTable);
Ryan Mitchell9634efb2021-03-19 14:53:17 -0700359
360 Validation validation_ = Validation::kEnabled;
Adam Lesinski6f6ceb72014-11-14 14:48:12 -0800361};
362
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700363} // namespace aapt
Adam Lesinski6f6ceb72014-11-14 14:48:12 -0800364
Adam Lesinskicacb28f2016-10-19 12:18:14 -0700365#endif // AAPT_RESOURCE_TABLE_H