blob: 22d587a7f5c41aebe03ac13938f5c899db4d144d [file] [log] [blame]
Adam Lesinski7ad11102016-10-28 16:39:15 -07001/*
2 * Copyright (C) 2016 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 "androidfw/LoadedArsc.h"
18
Adam Lesinskibebfcc42018-02-12 14:27:46 -080019#include "android-base/file.h"
20#include "androidfw/ResourceUtils.h"
21
Adam Lesinski7ad11102016-10-28 16:39:15 -070022#include "TestHelpers.h"
23#include "data/basic/R.h"
Adam Lesinskida431a22016-12-29 16:08:16 -050024#include "data/libclient/R.h"
Ryan Mitchell75e20dd2018-11-06 16:39:36 -080025#include "data/overlayable/R.h"
Adam Lesinski73f6f9d2017-11-14 10:18:05 -080026#include "data/sparse/R.h"
Adam Lesinski7ad11102016-10-28 16:39:15 -070027#include "data/styles/R.h"
28
29namespace app = com::android::app;
30namespace basic = com::android::basic;
Adam Lesinskida431a22016-12-29 16:08:16 -050031namespace libclient = com::android::libclient;
Ryan Mitchell75e20dd2018-11-06 16:39:36 -080032namespace overlayable = com::android::overlayable;
Adam Lesinski73f6f9d2017-11-14 10:18:05 -080033namespace sparse = com::android::sparse;
Adam Lesinski7ad11102016-10-28 16:39:15 -070034
Adam Lesinskibebfcc42018-02-12 14:27:46 -080035using ::android::base::ReadFileToString;
36using ::testing::Eq;
37using ::testing::Ge;
38using ::testing::IsNull;
39using ::testing::NotNull;
40using ::testing::SizeIs;
41using ::testing::StrEq;
42
Adam Lesinski7ad11102016-10-28 16:39:15 -070043namespace android {
44
45TEST(LoadedArscTest, LoadSinglePackageArsc) {
Adam Lesinski7ad11102016-10-28 16:39:15 -070046 std::string contents;
47 ASSERT_TRUE(ReadFileFromZipToString(GetTestDataPath() + "/styles/styles.apk", "resources.arsc",
48 &contents));
49
Adam Lesinski970bd8d2017-09-25 13:21:55 -070050 std::unique_ptr<const LoadedArsc> loaded_arsc = LoadedArsc::Load(StringPiece(contents));
Adam Lesinskibebfcc42018-02-12 14:27:46 -080051 ASSERT_THAT(loaded_arsc, NotNull());
Adam Lesinski7ad11102016-10-28 16:39:15 -070052
Adam Lesinskibebfcc42018-02-12 14:27:46 -080053 const LoadedPackage* package =
54 loaded_arsc->GetPackageById(get_package_id(app::R::string::string_one));
55 ASSERT_THAT(package, NotNull());
56 EXPECT_THAT(package->GetPackageName(), StrEq("com.android.app"));
57 EXPECT_THAT(package->GetPackageId(), Eq(0x7f));
Adam Lesinskida431a22016-12-29 16:08:16 -050058
Adam Lesinskibebfcc42018-02-12 14:27:46 -080059 const uint8_t type_index = get_type_id(app::R::string::string_one) - 1;
60 const uint16_t entry_index = get_entry_id(app::R::string::string_one);
Adam Lesinski7ad11102016-10-28 16:39:15 -070061
Adam Lesinskibebfcc42018-02-12 14:27:46 -080062 const TypeSpec* type_spec = package->GetTypeSpecByTypeIndex(type_index);
63 ASSERT_THAT(type_spec, NotNull());
64 ASSERT_THAT(type_spec->type_count, Ge(1u));
Adam Lesinski7ad11102016-10-28 16:39:15 -070065
Adam Lesinskibebfcc42018-02-12 14:27:46 -080066 const ResTable_type* type = type_spec->types[0];
67 ASSERT_THAT(type, NotNull());
68 ASSERT_THAT(LoadedPackage::GetEntry(type, entry_index), NotNull());
Adam Lesinski7ad11102016-10-28 16:39:15 -070069}
70
Adam Lesinski73f6f9d2017-11-14 10:18:05 -080071TEST(LoadedArscTest, LoadSparseEntryApp) {
72 std::string contents;
73 ASSERT_TRUE(ReadFileFromZipToString(GetTestDataPath() + "/sparse/sparse.apk", "resources.arsc",
74 &contents));
75
76 std::unique_ptr<const LoadedArsc> loaded_arsc = LoadedArsc::Load(StringPiece(contents));
Adam Lesinskibebfcc42018-02-12 14:27:46 -080077 ASSERT_THAT(loaded_arsc, NotNull());
Adam Lesinski73f6f9d2017-11-14 10:18:05 -080078
Adam Lesinskibebfcc42018-02-12 14:27:46 -080079 const LoadedPackage* package =
80 loaded_arsc->GetPackageById(get_package_id(sparse::R::integer::foo_9));
81 ASSERT_THAT(package, NotNull());
Adam Lesinski73f6f9d2017-11-14 10:18:05 -080082
Adam Lesinskibebfcc42018-02-12 14:27:46 -080083 const uint8_t type_index = get_type_id(sparse::R::integer::foo_9) - 1;
84 const uint16_t entry_index = get_entry_id(sparse::R::integer::foo_9);
85
86 const TypeSpec* type_spec = package->GetTypeSpecByTypeIndex(type_index);
87 ASSERT_THAT(type_spec, NotNull());
88 ASSERT_THAT(type_spec->type_count, Ge(1u));
89
90 const ResTable_type* type = type_spec->types[0];
91 ASSERT_THAT(type, NotNull());
92 ASSERT_THAT(LoadedPackage::GetEntry(type, entry_index), NotNull());
Adam Lesinski73f6f9d2017-11-14 10:18:05 -080093}
94
Adam Lesinskida431a22016-12-29 16:08:16 -050095TEST(LoadedArscTest, LoadSharedLibrary) {
96 std::string contents;
97 ASSERT_TRUE(ReadFileFromZipToString(GetTestDataPath() + "/lib_one/lib_one.apk", "resources.arsc",
98 &contents));
99
Adam Lesinski970bd8d2017-09-25 13:21:55 -0700100 std::unique_ptr<const LoadedArsc> loaded_arsc = LoadedArsc::Load(StringPiece(contents));
Adam Lesinskibebfcc42018-02-12 14:27:46 -0800101 ASSERT_THAT(loaded_arsc, NotNull());
Adam Lesinskida431a22016-12-29 16:08:16 -0500102
103 const auto& packages = loaded_arsc->GetPackages();
Adam Lesinskibebfcc42018-02-12 14:27:46 -0800104 ASSERT_THAT(packages, SizeIs(1u));
Adam Lesinskida431a22016-12-29 16:08:16 -0500105 EXPECT_TRUE(packages[0]->IsDynamic());
Adam Lesinskibebfcc42018-02-12 14:27:46 -0800106 EXPECT_THAT(packages[0]->GetPackageName(), StrEq("com.android.lib_one"));
107 EXPECT_THAT(packages[0]->GetPackageId(), Eq(0));
Adam Lesinskida431a22016-12-29 16:08:16 -0500108
109 const auto& dynamic_pkg_map = packages[0]->GetDynamicPackageMap();
110
111 // The library has no dependencies.
112 ASSERT_TRUE(dynamic_pkg_map.empty());
113}
114
115TEST(LoadedArscTest, LoadAppLinkedAgainstSharedLibrary) {
116 std::string contents;
117 ASSERT_TRUE(ReadFileFromZipToString(GetTestDataPath() + "/libclient/libclient.apk",
118 "resources.arsc", &contents));
119
Adam Lesinski970bd8d2017-09-25 13:21:55 -0700120 std::unique_ptr<const LoadedArsc> loaded_arsc = LoadedArsc::Load(StringPiece(contents));
Adam Lesinskibebfcc42018-02-12 14:27:46 -0800121 ASSERT_THAT(loaded_arsc, NotNull());
Adam Lesinskida431a22016-12-29 16:08:16 -0500122
123 const auto& packages = loaded_arsc->GetPackages();
Adam Lesinskibebfcc42018-02-12 14:27:46 -0800124 ASSERT_THAT(packages, SizeIs(1u));
Adam Lesinskida431a22016-12-29 16:08:16 -0500125 EXPECT_FALSE(packages[0]->IsDynamic());
Adam Lesinskibebfcc42018-02-12 14:27:46 -0800126 EXPECT_THAT(packages[0]->GetPackageName(), StrEq("com.android.libclient"));
127 EXPECT_THAT(packages[0]->GetPackageId(), Eq(0x7f));
Adam Lesinskida431a22016-12-29 16:08:16 -0500128
129 const auto& dynamic_pkg_map = packages[0]->GetDynamicPackageMap();
130
131 // The library has two dependencies.
Adam Lesinskibebfcc42018-02-12 14:27:46 -0800132 ASSERT_THAT(dynamic_pkg_map, SizeIs(2u));
133 EXPECT_THAT(dynamic_pkg_map[0].package_name, StrEq("com.android.lib_one"));
134 EXPECT_THAT(dynamic_pkg_map[0].package_id, Eq(0x02));
Adam Lesinskida431a22016-12-29 16:08:16 -0500135
Adam Lesinskibebfcc42018-02-12 14:27:46 -0800136 EXPECT_THAT(dynamic_pkg_map[1].package_name, StrEq("com.android.lib_two"));
137 EXPECT_THAT(dynamic_pkg_map[1].package_id, Eq(0x03));
Adam Lesinskida431a22016-12-29 16:08:16 -0500138}
139
140TEST(LoadedArscTest, LoadAppAsSharedLibrary) {
141 std::string contents;
142 ASSERT_TRUE(ReadFileFromZipToString(GetTestDataPath() + "/appaslib/appaslib.apk",
143 "resources.arsc", &contents));
144
Adam Lesinski970bd8d2017-09-25 13:21:55 -0700145 std::unique_ptr<const LoadedArsc> loaded_arsc =
146 LoadedArsc::Load(StringPiece(contents), nullptr /*loaded_idmap*/, false /*system*/,
147 true /*load_as_shared_library*/);
Adam Lesinskibebfcc42018-02-12 14:27:46 -0800148 ASSERT_THAT(loaded_arsc, NotNull());
Adam Lesinskida431a22016-12-29 16:08:16 -0500149
150 const auto& packages = loaded_arsc->GetPackages();
Adam Lesinskibebfcc42018-02-12 14:27:46 -0800151 ASSERT_THAT(packages, SizeIs(1u));
Adam Lesinskida431a22016-12-29 16:08:16 -0500152 EXPECT_TRUE(packages[0]->IsDynamic());
Adam Lesinskibebfcc42018-02-12 14:27:46 -0800153 EXPECT_THAT(packages[0]->GetPackageId(), Eq(0x7f));
Adam Lesinskida431a22016-12-29 16:08:16 -0500154}
155
Adam Lesinskic6aada92017-01-13 15:34:14 -0800156TEST(LoadedArscTest, LoadFeatureSplit) {
157 std::string contents;
158 ASSERT_TRUE(ReadFileFromZipToString(GetTestDataPath() + "/feature/feature.apk", "resources.arsc",
159 &contents));
Adam Lesinski970bd8d2017-09-25 13:21:55 -0700160 std::unique_ptr<const LoadedArsc> loaded_arsc = LoadedArsc::Load(StringPiece(contents));
Adam Lesinskibebfcc42018-02-12 14:27:46 -0800161 ASSERT_THAT(loaded_arsc, NotNull());
Adam Lesinskic6aada92017-01-13 15:34:14 -0800162
Adam Lesinskibebfcc42018-02-12 14:27:46 -0800163 const LoadedPackage* package =
164 loaded_arsc->GetPackageById(get_package_id(basic::R::string::test3));
165 ASSERT_THAT(package, NotNull());
Adam Lesinskic6aada92017-01-13 15:34:14 -0800166
Adam Lesinskibebfcc42018-02-12 14:27:46 -0800167 uint8_t type_index = get_type_id(basic::R::string::test3) - 1;
168 uint8_t entry_index = get_entry_id(basic::R::string::test3);
169
170 const TypeSpec* type_spec = package->GetTypeSpecByTypeIndex(type_index);
171 ASSERT_THAT(type_spec, NotNull());
172 ASSERT_THAT(type_spec->type_count, Ge(1u));
173 ASSERT_THAT(type_spec->types[0], NotNull());
Adam Lesinskic6aada92017-01-13 15:34:14 -0800174
175 size_t len;
Adam Lesinskibebfcc42018-02-12 14:27:46 -0800176 const char16_t* type_name16 =
177 package->GetTypeStringPool()->stringAt(type_spec->type_spec->id - 1, &len);
178 ASSERT_THAT(type_name16, NotNull());
179 EXPECT_THAT(util::Utf16ToUtf8(StringPiece16(type_name16, len)), StrEq("string"));
Adam Lesinskic6aada92017-01-13 15:34:14 -0800180
Adam Lesinskibebfcc42018-02-12 14:27:46 -0800181 ASSERT_THAT(LoadedPackage::GetEntry(type_spec->types[0], entry_index), NotNull());
182}
183
184// AAPT(2) generates resource tables with chunks in a certain order. The rule is that
185// a RES_TABLE_TYPE_TYPE with id `i` must always be preceded by a RES_TABLE_TYPE_SPEC_TYPE with
186// id `i`. The RES_TABLE_TYPE_SPEC_TYPE does not need to be directly preceding, however.
187//
188// AAPT(2) generates something like:
189// RES_TABLE_TYPE_SPEC_TYPE id=1
190// RES_TABLE_TYPE_TYPE id=1
191// RES_TABLE_TYPE_SPEC_TYPE id=2
192// RES_TABLE_TYPE_TYPE id=2
193//
194// But the following is valid too:
195// RES_TABLE_TYPE_SPEC_TYPE id=1
196// RES_TABLE_TYPE_SPEC_TYPE id=2
197// RES_TABLE_TYPE_TYPE id=1
198// RES_TABLE_TYPE_TYPE id=2
199//
200TEST(LoadedArscTest, LoadOutOfOrderTypeSpecs) {
201 std::string contents;
202 ASSERT_TRUE(
203 ReadFileFromZipToString(GetTestDataPath() + "/out_of_order_types/out_of_order_types.apk",
204 "resources.arsc", &contents));
205
206 std::unique_ptr<const LoadedArsc> loaded_arsc = LoadedArsc::Load(StringPiece(contents));
207 ASSERT_THAT(loaded_arsc, NotNull());
208
209 ASSERT_THAT(loaded_arsc->GetPackages(), SizeIs(1u));
210 const auto& package = loaded_arsc->GetPackages()[0];
211 ASSERT_THAT(package, NotNull());
212
213 const TypeSpec* type_spec = package->GetTypeSpecByTypeIndex(0);
214 ASSERT_THAT(type_spec, NotNull());
215 ASSERT_THAT(type_spec->type_count, Ge(1u));
216 ASSERT_THAT(type_spec->types[0], NotNull());
217
218 type_spec = package->GetTypeSpecByTypeIndex(1);
219 ASSERT_THAT(type_spec, NotNull());
220 ASSERT_THAT(type_spec->type_count, Ge(1u));
221 ASSERT_THAT(type_spec->types[0], NotNull());
Adam Lesinskic6aada92017-01-13 15:34:14 -0800222}
223
Adam Lesinski970bd8d2017-09-25 13:21:55 -0700224class MockLoadedIdmap : public LoadedIdmap {
225 public:
226 MockLoadedIdmap() : LoadedIdmap() {
227 local_header_.magic = kIdmapMagic;
228 local_header_.version = kIdmapCurrentVersion;
229 local_header_.target_package_id = 0x08;
230 local_header_.type_count = 1;
231 header_ = &local_header_;
232
233 entry_header = util::unique_cptr<IdmapEntry_header>(
234 (IdmapEntry_header*)::malloc(sizeof(IdmapEntry_header) + sizeof(uint32_t)));
235 entry_header->target_type_id = 0x03;
236 entry_header->overlay_type_id = 0x02;
237 entry_header->entry_id_offset = 1;
238 entry_header->entry_count = 1;
239 entry_header->entries[0] = 0x00000000u;
240 type_map_[entry_header->overlay_type_id] = entry_header.get();
241 }
242
243 private:
244 Idmap_header local_header_;
245 util::unique_cptr<IdmapEntry_header> entry_header;
246};
247
248TEST(LoadedArscTest, LoadOverlay) {
Adam Lesinskibebfcc42018-02-12 14:27:46 -0800249 std::string contents;
Adam Lesinski970bd8d2017-09-25 13:21:55 -0700250 ASSERT_TRUE(ReadFileFromZipToString(GetTestDataPath() + "/overlay/overlay.apk", "resources.arsc",
Adam Lesinskibebfcc42018-02-12 14:27:46 -0800251 &contents));
Adam Lesinski970bd8d2017-09-25 13:21:55 -0700252
253 MockLoadedIdmap loaded_idmap;
254
255 std::unique_ptr<const LoadedArsc> loaded_arsc =
Adam Lesinskibebfcc42018-02-12 14:27:46 -0800256 LoadedArsc::Load(StringPiece(contents), &loaded_idmap);
257 ASSERT_THAT(loaded_arsc, NotNull());
Adam Lesinski970bd8d2017-09-25 13:21:55 -0700258
Adam Lesinskibebfcc42018-02-12 14:27:46 -0800259 const LoadedPackage* package = loaded_arsc->GetPackageById(0x08u);
260 ASSERT_THAT(package, NotNull());
Adam Lesinski970bd8d2017-09-25 13:21:55 -0700261
Adam Lesinskibebfcc42018-02-12 14:27:46 -0800262 const TypeSpec* type_spec = package->GetTypeSpecByTypeIndex(0x03u - 1);
263 ASSERT_THAT(type_spec, NotNull());
264 ASSERT_THAT(type_spec->type_count, Ge(1u));
265 ASSERT_THAT(type_spec->types[0], NotNull());
266
267 // The entry being overlaid doesn't exist at the original entry index.
268 ASSERT_THAT(LoadedPackage::GetEntry(type_spec->types[0], 0x0001u), IsNull());
269
270 // Since this is an overlay, the actual entry ID must be mapped.
271 ASSERT_THAT(type_spec->idmap_entries, NotNull());
272 uint16_t target_entry_id = 0u;
273 ASSERT_TRUE(LoadedIdmap::Lookup(type_spec->idmap_entries, 0x0001u, &target_entry_id));
274 ASSERT_THAT(target_entry_id, Eq(0x0u));
275 ASSERT_THAT(LoadedPackage::GetEntry(type_spec->types[0], 0x0000), NotNull());
Adam Lesinski970bd8d2017-09-25 13:21:55 -0700276}
277
Ryan Mitchell75e20dd2018-11-06 16:39:36 -0800278TEST(LoadedArscTest, LoadOverlayable) {
279 std::string contents;
280 ASSERT_TRUE(ReadFileFromZipToString(GetTestDataPath() + "/overlayable/overlayable.apk",
281 "resources.arsc", &contents));
282
283 std::unique_ptr<const LoadedArsc> loaded_arsc =
284 LoadedArsc::Load(StringPiece(contents), nullptr /*loaded_idmap*/, false /*system*/,
285 false /*load_as_shared_library*/);
286
287 ASSERT_THAT(loaded_arsc, NotNull());
288 const LoadedPackage* package = loaded_arsc->GetPackageById(
289 get_package_id(overlayable::R::string::not_overlayable));
290
291 const OverlayableInfo* info = package->GetOverlayableInfo(
292 overlayable::R::string::not_overlayable);
293 ASSERT_THAT(info, IsNull());
294
295 info = package->GetOverlayableInfo(overlayable::R::string::overlayable1);
296 ASSERT_THAT(info, NotNull());
297 EXPECT_THAT(info->policy_flags, Eq(ResTable_overlayable_policy_header::POLICY_PUBLIC));
298
299 info = package->GetOverlayableInfo(overlayable::R::string::overlayable2);
300 ASSERT_THAT(info, NotNull());
301 EXPECT_THAT(info->policy_flags,
302 Eq(ResTable_overlayable_policy_header::POLICY_SYSTEM_PARTITION
303 | ResTable_overlayable_policy_header::POLICY_PRODUCT_PARTITION));
304
305 info = package->GetOverlayableInfo(overlayable::R::string::overlayable3);
306 ASSERT_THAT(info, NotNull());
307 EXPECT_THAT(info->policy_flags,
308 Eq(ResTable_overlayable_policy_header::POLICY_VENDOR_PARTITION
309 | ResTable_overlayable_policy_header::POLICY_PRODUCT_SERVICES_PARTITION
310 | ResTable_overlayable_policy_header::POLICY_PRODUCT_PARTITION));
311
312 info = package->GetOverlayableInfo(overlayable::R::string::overlayable4);
313 ASSERT_THAT(info, NotNull());
314 EXPECT_THAT(info->policy_flags, Eq(ResTable_overlayable_policy_header::POLICY_PUBLIC));
315}
Adam Lesinski7ad11102016-10-28 16:39:15 -0700316
MÃ¥rten Kongstad3f1f4fc2018-03-02 09:34:18 +0100317TEST(LoadedArscTest, ResourceIdentifierIterator) {
318 std::string contents;
319 ASSERT_TRUE(
320 ReadFileFromZipToString(GetTestDataPath() + "/basic/basic.apk", "resources.arsc", &contents));
321
322 std::unique_ptr<const LoadedArsc> loaded_arsc = LoadedArsc::Load(StringPiece(contents));
323 ASSERT_NE(nullptr, loaded_arsc);
324
325 const std::vector<std::unique_ptr<const LoadedPackage>>& packages = loaded_arsc->GetPackages();
326 ASSERT_EQ(1u, packages.size());
327 EXPECT_EQ(std::string("com.android.basic"), packages[0]->GetPackageName());
328
329 const auto& loaded_package = packages[0];
330 auto iter = loaded_package->begin();
331 auto end = loaded_package->end();
332
333 ASSERT_NE(end, iter);
334 ASSERT_EQ(0x7f010000u, *iter++);
335 ASSERT_EQ(0x7f010001u, *iter++);
336 ASSERT_EQ(0x7f020000u, *iter++);
337 ASSERT_EQ(0x7f020001u, *iter++);
338 ASSERT_EQ(0x7f030000u, *iter++);
339 ASSERT_EQ(0x7f030001u, *iter++);
340 ASSERT_EQ(0x7f030002u, *iter++); // note: string without default, excluded by aapt2 dump
341 ASSERT_EQ(0x7f040000u, *iter++);
342 ASSERT_EQ(0x7f040001u, *iter++);
343 ASSERT_EQ(0x7f040002u, *iter++);
344 ASSERT_EQ(0x7f040003u, *iter++);
345 ASSERT_EQ(0x7f040004u, *iter++);
346 ASSERT_EQ(0x7f040005u, *iter++);
347 ASSERT_EQ(0x7f040006u, *iter++);
348 ASSERT_EQ(0x7f040007u, *iter++);
349 ASSERT_EQ(0x7f040008u, *iter++);
350 ASSERT_EQ(0x7f040009u, *iter++);
351 ASSERT_EQ(0x7f04000au, *iter++);
352 ASSERT_EQ(0x7f04000bu, *iter++);
353 ASSERT_EQ(0x7f04000cu, *iter++);
354 ASSERT_EQ(0x7f04000du, *iter++);
355 ASSERT_EQ(0x7f050000u, *iter++);
356 ASSERT_EQ(0x7f050001u, *iter++);
357 ASSERT_EQ(0x7f060000u, *iter++);
358 ASSERT_EQ(0x7f070000u, *iter++);
359 ASSERT_EQ(0x7f070001u, *iter++);
360 ASSERT_EQ(0x7f070002u, *iter++);
361 ASSERT_EQ(0x7f070003u, *iter++);
362 ASSERT_EQ(end, iter);
363}
364
Ryan Mitchell75e20dd2018-11-06 16:39:36 -0800365// structs with size fields (like Res_value, ResTable_entry) should be
366// backwards and forwards compatible (aka checking the size field against
367// sizeof(Res_value) might not be backwards compatible.
Aurimas Liutikas6c2a8b482018-12-14 10:12:27 -0800368// TEST(LoadedArscTest, LoadingShouldBeForwardsAndBackwardsCompatible) { ASSERT_TRUE(false); }
Ryan Mitchell75e20dd2018-11-06 16:39:36 -0800369
Adam Lesinski7ad11102016-10-28 16:39:15 -0700370} // namespace android