blob: fd57a92c216b88e18b38922f2e9a4bd18ef956f5 [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"
Winson9947f1e2019-08-16 10:20:39 -070028#include "data/system/R.h"
Adam Lesinski7ad11102016-10-28 16:39:15 -070029
30namespace app = com::android::app;
31namespace basic = com::android::basic;
Adam Lesinskida431a22016-12-29 16:08:16 -050032namespace libclient = com::android::libclient;
Ryan Mitchell75e20dd2018-11-06 16:39:36 -080033namespace overlayable = com::android::overlayable;
Adam Lesinski73f6f9d2017-11-14 10:18:05 -080034namespace sparse = com::android::sparse;
Adam Lesinski7ad11102016-10-28 16:39:15 -070035
Adam Lesinskibebfcc42018-02-12 14:27:46 -080036using ::android::base::ReadFileToString;
37using ::testing::Eq;
38using ::testing::Ge;
39using ::testing::IsNull;
40using ::testing::NotNull;
41using ::testing::SizeIs;
42using ::testing::StrEq;
43
Adam Lesinski7ad11102016-10-28 16:39:15 -070044namespace android {
45
46TEST(LoadedArscTest, LoadSinglePackageArsc) {
Adam Lesinski7ad11102016-10-28 16:39:15 -070047 std::string contents;
48 ASSERT_TRUE(ReadFileFromZipToString(GetTestDataPath() + "/styles/styles.apk", "resources.arsc",
49 &contents));
50
Adam Lesinski970bd8d2017-09-25 13:21:55 -070051 std::unique_ptr<const LoadedArsc> loaded_arsc = LoadedArsc::Load(StringPiece(contents));
Adam Lesinskibebfcc42018-02-12 14:27:46 -080052 ASSERT_THAT(loaded_arsc, NotNull());
Adam Lesinski7ad11102016-10-28 16:39:15 -070053
Adam Lesinskibebfcc42018-02-12 14:27:46 -080054 const LoadedPackage* package =
55 loaded_arsc->GetPackageById(get_package_id(app::R::string::string_one));
56 ASSERT_THAT(package, NotNull());
57 EXPECT_THAT(package->GetPackageName(), StrEq("com.android.app"));
58 EXPECT_THAT(package->GetPackageId(), Eq(0x7f));
Adam Lesinskida431a22016-12-29 16:08:16 -050059
Adam Lesinskibebfcc42018-02-12 14:27:46 -080060 const uint8_t type_index = get_type_id(app::R::string::string_one) - 1;
61 const uint16_t entry_index = get_entry_id(app::R::string::string_one);
Adam Lesinski7ad11102016-10-28 16:39:15 -070062
Adam Lesinskibebfcc42018-02-12 14:27:46 -080063 const TypeSpec* type_spec = package->GetTypeSpecByTypeIndex(type_index);
64 ASSERT_THAT(type_spec, NotNull());
65 ASSERT_THAT(type_spec->type_count, Ge(1u));
Adam Lesinski7ad11102016-10-28 16:39:15 -070066
Adam Lesinskibebfcc42018-02-12 14:27:46 -080067 const ResTable_type* type = type_spec->types[0];
68 ASSERT_THAT(type, NotNull());
69 ASSERT_THAT(LoadedPackage::GetEntry(type, entry_index), NotNull());
Adam Lesinski7ad11102016-10-28 16:39:15 -070070}
71
Adam Lesinski73f6f9d2017-11-14 10:18:05 -080072TEST(LoadedArscTest, LoadSparseEntryApp) {
73 std::string contents;
74 ASSERT_TRUE(ReadFileFromZipToString(GetTestDataPath() + "/sparse/sparse.apk", "resources.arsc",
75 &contents));
76
77 std::unique_ptr<const LoadedArsc> loaded_arsc = LoadedArsc::Load(StringPiece(contents));
Adam Lesinskibebfcc42018-02-12 14:27:46 -080078 ASSERT_THAT(loaded_arsc, NotNull());
Adam Lesinski73f6f9d2017-11-14 10:18:05 -080079
Adam Lesinskibebfcc42018-02-12 14:27:46 -080080 const LoadedPackage* package =
81 loaded_arsc->GetPackageById(get_package_id(sparse::R::integer::foo_9));
82 ASSERT_THAT(package, NotNull());
Adam Lesinski73f6f9d2017-11-14 10:18:05 -080083
Adam Lesinskibebfcc42018-02-12 14:27:46 -080084 const uint8_t type_index = get_type_id(sparse::R::integer::foo_9) - 1;
85 const uint16_t entry_index = get_entry_id(sparse::R::integer::foo_9);
86
87 const TypeSpec* type_spec = package->GetTypeSpecByTypeIndex(type_index);
88 ASSERT_THAT(type_spec, NotNull());
89 ASSERT_THAT(type_spec->type_count, Ge(1u));
90
91 const ResTable_type* type = type_spec->types[0];
92 ASSERT_THAT(type, NotNull());
93 ASSERT_THAT(LoadedPackage::GetEntry(type, entry_index), NotNull());
Adam Lesinski73f6f9d2017-11-14 10:18:05 -080094}
95
Adam Lesinskida431a22016-12-29 16:08:16 -050096TEST(LoadedArscTest, LoadSharedLibrary) {
97 std::string contents;
98 ASSERT_TRUE(ReadFileFromZipToString(GetTestDataPath() + "/lib_one/lib_one.apk", "resources.arsc",
99 &contents));
100
Adam Lesinski970bd8d2017-09-25 13:21:55 -0700101 std::unique_ptr<const LoadedArsc> loaded_arsc = LoadedArsc::Load(StringPiece(contents));
Adam Lesinskibebfcc42018-02-12 14:27:46 -0800102 ASSERT_THAT(loaded_arsc, NotNull());
Adam Lesinskida431a22016-12-29 16:08:16 -0500103
104 const auto& packages = loaded_arsc->GetPackages();
Adam Lesinskibebfcc42018-02-12 14:27:46 -0800105 ASSERT_THAT(packages, SizeIs(1u));
Adam Lesinskida431a22016-12-29 16:08:16 -0500106 EXPECT_TRUE(packages[0]->IsDynamic());
Adam Lesinskibebfcc42018-02-12 14:27:46 -0800107 EXPECT_THAT(packages[0]->GetPackageName(), StrEq("com.android.lib_one"));
108 EXPECT_THAT(packages[0]->GetPackageId(), Eq(0));
Adam Lesinskida431a22016-12-29 16:08:16 -0500109
110 const auto& dynamic_pkg_map = packages[0]->GetDynamicPackageMap();
111
112 // The library has no dependencies.
113 ASSERT_TRUE(dynamic_pkg_map.empty());
114}
115
116TEST(LoadedArscTest, LoadAppLinkedAgainstSharedLibrary) {
117 std::string contents;
118 ASSERT_TRUE(ReadFileFromZipToString(GetTestDataPath() + "/libclient/libclient.apk",
119 "resources.arsc", &contents));
120
Adam Lesinski970bd8d2017-09-25 13:21:55 -0700121 std::unique_ptr<const LoadedArsc> loaded_arsc = LoadedArsc::Load(StringPiece(contents));
Adam Lesinskibebfcc42018-02-12 14:27:46 -0800122 ASSERT_THAT(loaded_arsc, NotNull());
Adam Lesinskida431a22016-12-29 16:08:16 -0500123
124 const auto& packages = loaded_arsc->GetPackages();
Adam Lesinskibebfcc42018-02-12 14:27:46 -0800125 ASSERT_THAT(packages, SizeIs(1u));
Adam Lesinskida431a22016-12-29 16:08:16 -0500126 EXPECT_FALSE(packages[0]->IsDynamic());
Adam Lesinskibebfcc42018-02-12 14:27:46 -0800127 EXPECT_THAT(packages[0]->GetPackageName(), StrEq("com.android.libclient"));
128 EXPECT_THAT(packages[0]->GetPackageId(), Eq(0x7f));
Adam Lesinskida431a22016-12-29 16:08:16 -0500129
130 const auto& dynamic_pkg_map = packages[0]->GetDynamicPackageMap();
131
132 // The library has two dependencies.
Adam Lesinskibebfcc42018-02-12 14:27:46 -0800133 ASSERT_THAT(dynamic_pkg_map, SizeIs(2u));
134 EXPECT_THAT(dynamic_pkg_map[0].package_name, StrEq("com.android.lib_one"));
135 EXPECT_THAT(dynamic_pkg_map[0].package_id, Eq(0x02));
Adam Lesinskida431a22016-12-29 16:08:16 -0500136
Adam Lesinskibebfcc42018-02-12 14:27:46 -0800137 EXPECT_THAT(dynamic_pkg_map[1].package_name, StrEq("com.android.lib_two"));
138 EXPECT_THAT(dynamic_pkg_map[1].package_id, Eq(0x03));
Adam Lesinskida431a22016-12-29 16:08:16 -0500139}
140
141TEST(LoadedArscTest, LoadAppAsSharedLibrary) {
142 std::string contents;
143 ASSERT_TRUE(ReadFileFromZipToString(GetTestDataPath() + "/appaslib/appaslib.apk",
144 "resources.arsc", &contents));
145
Adam Lesinski970bd8d2017-09-25 13:21:55 -0700146 std::unique_ptr<const LoadedArsc> loaded_arsc =
147 LoadedArsc::Load(StringPiece(contents), nullptr /*loaded_idmap*/, false /*system*/,
148 true /*load_as_shared_library*/);
Adam Lesinskibebfcc42018-02-12 14:27:46 -0800149 ASSERT_THAT(loaded_arsc, NotNull());
Adam Lesinskida431a22016-12-29 16:08:16 -0500150
151 const auto& packages = loaded_arsc->GetPackages();
Adam Lesinskibebfcc42018-02-12 14:27:46 -0800152 ASSERT_THAT(packages, SizeIs(1u));
Adam Lesinskida431a22016-12-29 16:08:16 -0500153 EXPECT_TRUE(packages[0]->IsDynamic());
Adam Lesinskibebfcc42018-02-12 14:27:46 -0800154 EXPECT_THAT(packages[0]->GetPackageId(), Eq(0x7f));
Adam Lesinskida431a22016-12-29 16:08:16 -0500155}
156
Adam Lesinskic6aada92017-01-13 15:34:14 -0800157TEST(LoadedArscTest, LoadFeatureSplit) {
158 std::string contents;
159 ASSERT_TRUE(ReadFileFromZipToString(GetTestDataPath() + "/feature/feature.apk", "resources.arsc",
160 &contents));
Adam Lesinski970bd8d2017-09-25 13:21:55 -0700161 std::unique_ptr<const LoadedArsc> loaded_arsc = LoadedArsc::Load(StringPiece(contents));
Adam Lesinskibebfcc42018-02-12 14:27:46 -0800162 ASSERT_THAT(loaded_arsc, NotNull());
Adam Lesinskic6aada92017-01-13 15:34:14 -0800163
Adam Lesinskibebfcc42018-02-12 14:27:46 -0800164 const LoadedPackage* package =
165 loaded_arsc->GetPackageById(get_package_id(basic::R::string::test3));
166 ASSERT_THAT(package, NotNull());
Adam Lesinskic6aada92017-01-13 15:34:14 -0800167
Adam Lesinskibebfcc42018-02-12 14:27:46 -0800168 uint8_t type_index = get_type_id(basic::R::string::test3) - 1;
169 uint8_t entry_index = get_entry_id(basic::R::string::test3);
170
171 const TypeSpec* type_spec = package->GetTypeSpecByTypeIndex(type_index);
172 ASSERT_THAT(type_spec, NotNull());
173 ASSERT_THAT(type_spec->type_count, Ge(1u));
174 ASSERT_THAT(type_spec->types[0], NotNull());
Adam Lesinskic6aada92017-01-13 15:34:14 -0800175
176 size_t len;
Adam Lesinskibebfcc42018-02-12 14:27:46 -0800177 const char16_t* type_name16 =
178 package->GetTypeStringPool()->stringAt(type_spec->type_spec->id - 1, &len);
179 ASSERT_THAT(type_name16, NotNull());
180 EXPECT_THAT(util::Utf16ToUtf8(StringPiece16(type_name16, len)), StrEq("string"));
Adam Lesinskic6aada92017-01-13 15:34:14 -0800181
Adam Lesinskibebfcc42018-02-12 14:27:46 -0800182 ASSERT_THAT(LoadedPackage::GetEntry(type_spec->types[0], entry_index), NotNull());
183}
184
185// AAPT(2) generates resource tables with chunks in a certain order. The rule is that
186// a RES_TABLE_TYPE_TYPE with id `i` must always be preceded by a RES_TABLE_TYPE_SPEC_TYPE with
187// id `i`. The RES_TABLE_TYPE_SPEC_TYPE does not need to be directly preceding, however.
188//
189// AAPT(2) generates something like:
190// RES_TABLE_TYPE_SPEC_TYPE id=1
191// RES_TABLE_TYPE_TYPE id=1
192// RES_TABLE_TYPE_SPEC_TYPE id=2
193// RES_TABLE_TYPE_TYPE id=2
194//
195// But the following is valid too:
196// RES_TABLE_TYPE_SPEC_TYPE id=1
197// RES_TABLE_TYPE_SPEC_TYPE id=2
198// RES_TABLE_TYPE_TYPE id=1
199// RES_TABLE_TYPE_TYPE id=2
200//
201TEST(LoadedArscTest, LoadOutOfOrderTypeSpecs) {
202 std::string contents;
203 ASSERT_TRUE(
204 ReadFileFromZipToString(GetTestDataPath() + "/out_of_order_types/out_of_order_types.apk",
205 "resources.arsc", &contents));
206
207 std::unique_ptr<const LoadedArsc> loaded_arsc = LoadedArsc::Load(StringPiece(contents));
208 ASSERT_THAT(loaded_arsc, NotNull());
209
210 ASSERT_THAT(loaded_arsc->GetPackages(), SizeIs(1u));
211 const auto& package = loaded_arsc->GetPackages()[0];
212 ASSERT_THAT(package, NotNull());
213
214 const TypeSpec* type_spec = package->GetTypeSpecByTypeIndex(0);
215 ASSERT_THAT(type_spec, NotNull());
216 ASSERT_THAT(type_spec->type_count, Ge(1u));
217 ASSERT_THAT(type_spec->types[0], NotNull());
218
219 type_spec = package->GetTypeSpecByTypeIndex(1);
220 ASSERT_THAT(type_spec, NotNull());
221 ASSERT_THAT(type_spec->type_count, Ge(1u));
222 ASSERT_THAT(type_spec->types[0], NotNull());
Adam Lesinskic6aada92017-01-13 15:34:14 -0800223}
224
Adam Lesinski970bd8d2017-09-25 13:21:55 -0700225class MockLoadedIdmap : public LoadedIdmap {
226 public:
227 MockLoadedIdmap() : LoadedIdmap() {
228 local_header_.magic = kIdmapMagic;
229 local_header_.version = kIdmapCurrentVersion;
230 local_header_.target_package_id = 0x08;
231 local_header_.type_count = 1;
232 header_ = &local_header_;
233
234 entry_header = util::unique_cptr<IdmapEntry_header>(
235 (IdmapEntry_header*)::malloc(sizeof(IdmapEntry_header) + sizeof(uint32_t)));
236 entry_header->target_type_id = 0x03;
237 entry_header->overlay_type_id = 0x02;
238 entry_header->entry_id_offset = 1;
239 entry_header->entry_count = 1;
240 entry_header->entries[0] = 0x00000000u;
241 type_map_[entry_header->overlay_type_id] = entry_header.get();
242 }
243
244 private:
245 Idmap_header local_header_;
246 util::unique_cptr<IdmapEntry_header> entry_header;
247};
248
249TEST(LoadedArscTest, LoadOverlay) {
Adam Lesinskibebfcc42018-02-12 14:27:46 -0800250 std::string contents;
Adam Lesinski970bd8d2017-09-25 13:21:55 -0700251 ASSERT_TRUE(ReadFileFromZipToString(GetTestDataPath() + "/overlay/overlay.apk", "resources.arsc",
Adam Lesinskibebfcc42018-02-12 14:27:46 -0800252 &contents));
Adam Lesinski970bd8d2017-09-25 13:21:55 -0700253
254 MockLoadedIdmap loaded_idmap;
255
256 std::unique_ptr<const LoadedArsc> loaded_arsc =
Adam Lesinskibebfcc42018-02-12 14:27:46 -0800257 LoadedArsc::Load(StringPiece(contents), &loaded_idmap);
258 ASSERT_THAT(loaded_arsc, NotNull());
Adam Lesinski970bd8d2017-09-25 13:21:55 -0700259
Adam Lesinskibebfcc42018-02-12 14:27:46 -0800260 const LoadedPackage* package = loaded_arsc->GetPackageById(0x08u);
261 ASSERT_THAT(package, NotNull());
Adam Lesinski970bd8d2017-09-25 13:21:55 -0700262
Adam Lesinskibebfcc42018-02-12 14:27:46 -0800263 const TypeSpec* type_spec = package->GetTypeSpecByTypeIndex(0x03u - 1);
264 ASSERT_THAT(type_spec, NotNull());
265 ASSERT_THAT(type_spec->type_count, Ge(1u));
266 ASSERT_THAT(type_spec->types[0], NotNull());
267
268 // The entry being overlaid doesn't exist at the original entry index.
269 ASSERT_THAT(LoadedPackage::GetEntry(type_spec->types[0], 0x0001u), IsNull());
270
271 // Since this is an overlay, the actual entry ID must be mapped.
272 ASSERT_THAT(type_spec->idmap_entries, NotNull());
273 uint16_t target_entry_id = 0u;
274 ASSERT_TRUE(LoadedIdmap::Lookup(type_spec->idmap_entries, 0x0001u, &target_entry_id));
275 ASSERT_THAT(target_entry_id, Eq(0x0u));
276 ASSERT_THAT(LoadedPackage::GetEntry(type_spec->types[0], 0x0000), NotNull());
Adam Lesinski970bd8d2017-09-25 13:21:55 -0700277}
278
Ryan Mitchell75e20dd2018-11-06 16:39:36 -0800279TEST(LoadedArscTest, LoadOverlayable) {
280 std::string contents;
281 ASSERT_TRUE(ReadFileFromZipToString(GetTestDataPath() + "/overlayable/overlayable.apk",
282 "resources.arsc", &contents));
283
284 std::unique_ptr<const LoadedArsc> loaded_arsc =
285 LoadedArsc::Load(StringPiece(contents), nullptr /*loaded_idmap*/, false /*system*/,
286 false /*load_as_shared_library*/);
287
288 ASSERT_THAT(loaded_arsc, NotNull());
289 const LoadedPackage* package = loaded_arsc->GetPackageById(
290 get_package_id(overlayable::R::string::not_overlayable));
291
292 const OverlayableInfo* info = package->GetOverlayableInfo(
293 overlayable::R::string::not_overlayable);
294 ASSERT_THAT(info, IsNull());
295
296 info = package->GetOverlayableInfo(overlayable::R::string::overlayable1);
297 ASSERT_THAT(info, NotNull());
Ryan Mitchellef5673a2018-12-12 18:45:34 -0800298 EXPECT_THAT(info->name, Eq("OverlayableResources1"));
299 EXPECT_THAT(info->actor, Eq("overlay://theme"));
Ryan Mitchell75e20dd2018-11-06 16:39:36 -0800300 EXPECT_THAT(info->policy_flags, Eq(ResTable_overlayable_policy_header::POLICY_PUBLIC));
301
302 info = package->GetOverlayableInfo(overlayable::R::string::overlayable2);
303 ASSERT_THAT(info, NotNull());
Ryan Mitchellef5673a2018-12-12 18:45:34 -0800304 EXPECT_THAT(info->name, Eq("OverlayableResources1"));
305 EXPECT_THAT(info->actor, Eq("overlay://theme"));
Ryan Mitchell75e20dd2018-11-06 16:39:36 -0800306 EXPECT_THAT(info->policy_flags,
307 Eq(ResTable_overlayable_policy_header::POLICY_SYSTEM_PARTITION
308 | ResTable_overlayable_policy_header::POLICY_PRODUCT_PARTITION));
309
310 info = package->GetOverlayableInfo(overlayable::R::string::overlayable3);
311 ASSERT_THAT(info, NotNull());
Ryan Mitchellef5673a2018-12-12 18:45:34 -0800312 EXPECT_THAT(info->name, Eq("OverlayableResources2"));
313 EXPECT_THAT(info->actor, Eq("overlay://com.android.overlayable"));
Ryan Mitchell75e20dd2018-11-06 16:39:36 -0800314 EXPECT_THAT(info->policy_flags,
315 Eq(ResTable_overlayable_policy_header::POLICY_VENDOR_PARTITION
Ryan Mitchell75e20dd2018-11-06 16:39:36 -0800316 | ResTable_overlayable_policy_header::POLICY_PRODUCT_PARTITION));
317
318 info = package->GetOverlayableInfo(overlayable::R::string::overlayable4);
Ryan Mitchellef5673a2018-12-12 18:45:34 -0800319 EXPECT_THAT(info->name, Eq("OverlayableResources1"));
320 EXPECT_THAT(info->actor, Eq("overlay://theme"));
Ryan Mitchell75e20dd2018-11-06 16:39:36 -0800321 ASSERT_THAT(info, NotNull());
322 EXPECT_THAT(info->policy_flags, Eq(ResTable_overlayable_policy_header::POLICY_PUBLIC));
323}
Adam Lesinski7ad11102016-10-28 16:39:15 -0700324
Mårten Kongstad3f1f4fc2018-03-02 09:34:18 +0100325TEST(LoadedArscTest, ResourceIdentifierIterator) {
326 std::string contents;
327 ASSERT_TRUE(
328 ReadFileFromZipToString(GetTestDataPath() + "/basic/basic.apk", "resources.arsc", &contents));
329
330 std::unique_ptr<const LoadedArsc> loaded_arsc = LoadedArsc::Load(StringPiece(contents));
331 ASSERT_NE(nullptr, loaded_arsc);
332
333 const std::vector<std::unique_ptr<const LoadedPackage>>& packages = loaded_arsc->GetPackages();
334 ASSERT_EQ(1u, packages.size());
Mårten Kongstadc92c4dd2019-02-05 01:29:59 +0100335 ASSERT_EQ(std::string("com.android.basic"), packages[0]->GetPackageName());
Mårten Kongstad3f1f4fc2018-03-02 09:34:18 +0100336
337 const auto& loaded_package = packages[0];
338 auto iter = loaded_package->begin();
339 auto end = loaded_package->end();
340
341 ASSERT_NE(end, iter);
342 ASSERT_EQ(0x7f010000u, *iter++);
343 ASSERT_EQ(0x7f010001u, *iter++);
344 ASSERT_EQ(0x7f020000u, *iter++);
345 ASSERT_EQ(0x7f020001u, *iter++);
346 ASSERT_EQ(0x7f030000u, *iter++);
347 ASSERT_EQ(0x7f030001u, *iter++);
348 ASSERT_EQ(0x7f030002u, *iter++); // note: string without default, excluded by aapt2 dump
349 ASSERT_EQ(0x7f040000u, *iter++);
350 ASSERT_EQ(0x7f040001u, *iter++);
351 ASSERT_EQ(0x7f040002u, *iter++);
352 ASSERT_EQ(0x7f040003u, *iter++);
353 ASSERT_EQ(0x7f040004u, *iter++);
354 ASSERT_EQ(0x7f040005u, *iter++);
355 ASSERT_EQ(0x7f040006u, *iter++);
356 ASSERT_EQ(0x7f040007u, *iter++);
357 ASSERT_EQ(0x7f040008u, *iter++);
358 ASSERT_EQ(0x7f040009u, *iter++);
359 ASSERT_EQ(0x7f04000au, *iter++);
360 ASSERT_EQ(0x7f04000bu, *iter++);
361 ASSERT_EQ(0x7f04000cu, *iter++);
362 ASSERT_EQ(0x7f04000du, *iter++);
363 ASSERT_EQ(0x7f050000u, *iter++);
364 ASSERT_EQ(0x7f050001u, *iter++);
365 ASSERT_EQ(0x7f060000u, *iter++);
366 ASSERT_EQ(0x7f070000u, *iter++);
367 ASSERT_EQ(0x7f070001u, *iter++);
368 ASSERT_EQ(0x7f070002u, *iter++);
369 ASSERT_EQ(0x7f070003u, *iter++);
370 ASSERT_EQ(end, iter);
371}
372
Mårten Kongstadc92c4dd2019-02-05 01:29:59 +0100373TEST(LoadedArscTest, GetOverlayableMap) {
374 std::string contents;
375 ASSERT_TRUE(ReadFileFromZipToString(GetTestDataPath() + "/overlayable/overlayable.apk",
376 "resources.arsc", &contents));
377
378 std::unique_ptr<const LoadedArsc> loaded_arsc = LoadedArsc::Load(StringPiece(contents));
379 ASSERT_NE(nullptr, loaded_arsc);
380
381 const std::vector<std::unique_ptr<const LoadedPackage>>& packages = loaded_arsc->GetPackages();
382 ASSERT_EQ(1u, packages.size());
383 ASSERT_EQ(std::string("com.android.overlayable"), packages[0]->GetPackageName());
384
385 const auto map = packages[0]->GetOverlayableMap();
386 ASSERT_EQ(2, map.size());
387 ASSERT_EQ(map.at("OverlayableResources1"), "overlay://theme");
388 ASSERT_EQ(map.at("OverlayableResources2"), "overlay://com.android.overlayable");
389}
390
Winson9947f1e2019-08-16 10:20:39 -0700391TEST(LoadedArscTest, LoadCustomLoader) {
392 std::string contents;
393
394 std::unique_ptr<Asset>
395 asset = ApkAssets::CreateAssetFromFile(GetTestDataPath() + "/loader/resources.arsc");
396
397 MockLoadedIdmap loaded_idmap;
398 const StringPiece data(
399 reinterpret_cast<const char*>(asset->getBuffer(true /*wordAligned*/)),
400 asset->getLength());
401
402 std::unique_ptr<const LoadedArsc> loaded_arsc =
403 LoadedArsc::Load(data, nullptr, false, false, true);
404 ASSERT_THAT(loaded_arsc, NotNull());
405
406 const LoadedPackage* package =
407 loaded_arsc->GetPackageById(get_package_id(android::R::string::cancel));
408 ASSERT_THAT(package, NotNull());
409 EXPECT_THAT(package->GetPackageName(), StrEq("android"));
410 EXPECT_THAT(package->GetPackageId(), Eq(0x01));
411
412 const uint8_t type_index = get_type_id(android::R::string::cancel) - 1;
413 const uint16_t entry_index = get_entry_id(android::R::string::cancel);
414
415 const TypeSpec* type_spec = package->GetTypeSpecByTypeIndex(type_index);
416 ASSERT_THAT(type_spec, NotNull());
417 ASSERT_THAT(type_spec->type_count, Ge(1u));
418
419 const ResTable_type* type = type_spec->types[0];
420 ASSERT_THAT(type, NotNull());
421 ASSERT_THAT(LoadedPackage::GetEntry(type, entry_index), NotNull());
422}
423
Ryan Mitchell75e20dd2018-11-06 16:39:36 -0800424// structs with size fields (like Res_value, ResTable_entry) should be
425// backwards and forwards compatible (aka checking the size field against
426// sizeof(Res_value) might not be backwards compatible.
Aurimas Liutikas6c2a8b482018-12-14 10:12:27 -0800427// TEST(LoadedArscTest, LoadingShouldBeForwardsAndBackwardsCompatible) { ASSERT_TRUE(false); }
Ryan Mitchell75e20dd2018-11-06 16:39:36 -0800428
Adam Lesinski7ad11102016-10-28 16:39:15 -0700429} // namespace android