blob: c90ec197b5efbe893321ea4322b9853f84c68c59 [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
Winson62ac8b52019-12-04 08:36:48 -080044using PolicyFlags = android::ResTable_overlayable_policy_header::PolicyFlags;
45
Adam Lesinski7ad11102016-10-28 16:39:15 -070046namespace android {
47
48TEST(LoadedArscTest, LoadSinglePackageArsc) {
Adam Lesinski7ad11102016-10-28 16:39:15 -070049 std::string contents;
50 ASSERT_TRUE(ReadFileFromZipToString(GetTestDataPath() + "/styles/styles.apk", "resources.arsc",
51 &contents));
52
Ryan Mitchelldb21f09a2020-11-16 23:08:18 +000053 auto loaded_arsc = LoadedArsc::Load(reinterpret_cast<const void*>(contents.data()),
54 contents.length());
Adam Lesinskibebfcc42018-02-12 14:27:46 -080055 ASSERT_THAT(loaded_arsc, NotNull());
Adam Lesinski7ad11102016-10-28 16:39:15 -070056
Adam Lesinskibebfcc42018-02-12 14:27:46 -080057 const LoadedPackage* package =
58 loaded_arsc->GetPackageById(get_package_id(app::R::string::string_one));
59 ASSERT_THAT(package, NotNull());
60 EXPECT_THAT(package->GetPackageName(), StrEq("com.android.app"));
61 EXPECT_THAT(package->GetPackageId(), Eq(0x7f));
Adam Lesinskida431a22016-12-29 16:08:16 -050062
Adam Lesinskibebfcc42018-02-12 14:27:46 -080063 const uint8_t type_index = get_type_id(app::R::string::string_one) - 1;
64 const uint16_t entry_index = get_entry_id(app::R::string::string_one);
Adam Lesinski7ad11102016-10-28 16:39:15 -070065
Adam Lesinskibebfcc42018-02-12 14:27:46 -080066 const TypeSpec* type_spec = package->GetTypeSpecByTypeIndex(type_index);
67 ASSERT_THAT(type_spec, NotNull());
Ryan Mitchell14e8ade2021-01-11 16:01:35 -080068 ASSERT_THAT(type_spec->type_entries.size(), Ge(1u));
Adam Lesinski7ad11102016-10-28 16:39:15 -070069
Ryan Mitchell14e8ade2021-01-11 16:01:35 -080070 auto type = type_spec->type_entries[0];
71 ASSERT_TRUE(LoadedPackage::GetEntry(type.type, entry_index).has_value());
Adam Lesinski7ad11102016-10-28 16:39:15 -070072}
73
Adam Lesinskida431a22016-12-29 16:08:16 -050074TEST(LoadedArscTest, LoadSharedLibrary) {
75 std::string contents;
76 ASSERT_TRUE(ReadFileFromZipToString(GetTestDataPath() + "/lib_one/lib_one.apk", "resources.arsc",
77 &contents));
78
Ryan Mitchelldb21f09a2020-11-16 23:08:18 +000079 std::unique_ptr<const LoadedArsc> loaded_arsc = LoadedArsc::Load(contents.data(),
80 contents.length());
Adam Lesinskibebfcc42018-02-12 14:27:46 -080081 ASSERT_THAT(loaded_arsc, NotNull());
Adam Lesinskida431a22016-12-29 16:08:16 -050082
83 const auto& packages = loaded_arsc->GetPackages();
Adam Lesinskibebfcc42018-02-12 14:27:46 -080084 ASSERT_THAT(packages, SizeIs(1u));
Adam Lesinskida431a22016-12-29 16:08:16 -050085 EXPECT_TRUE(packages[0]->IsDynamic());
Adam Lesinskibebfcc42018-02-12 14:27:46 -080086 EXPECT_THAT(packages[0]->GetPackageName(), StrEq("com.android.lib_one"));
87 EXPECT_THAT(packages[0]->GetPackageId(), Eq(0));
Adam Lesinskida431a22016-12-29 16:08:16 -050088
89 const auto& dynamic_pkg_map = packages[0]->GetDynamicPackageMap();
90
91 // The library has no dependencies.
92 ASSERT_TRUE(dynamic_pkg_map.empty());
93}
94
95TEST(LoadedArscTest, LoadAppLinkedAgainstSharedLibrary) {
96 std::string contents;
97 ASSERT_TRUE(ReadFileFromZipToString(GetTestDataPath() + "/libclient/libclient.apk",
98 "resources.arsc", &contents));
99
Ryan Mitchelldb21f09a2020-11-16 23:08:18 +0000100 std::unique_ptr<const LoadedArsc> loaded_arsc = LoadedArsc::Load(contents.data(),
101 contents.length());
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_FALSE(packages[0]->IsDynamic());
Adam Lesinskibebfcc42018-02-12 14:27:46 -0800107 EXPECT_THAT(packages[0]->GetPackageName(), StrEq("com.android.libclient"));
108 EXPECT_THAT(packages[0]->GetPackageId(), Eq(0x7f));
Adam Lesinskida431a22016-12-29 16:08:16 -0500109
110 const auto& dynamic_pkg_map = packages[0]->GetDynamicPackageMap();
111
112 // The library has two dependencies.
Adam Lesinskibebfcc42018-02-12 14:27:46 -0800113 ASSERT_THAT(dynamic_pkg_map, SizeIs(2u));
114 EXPECT_THAT(dynamic_pkg_map[0].package_name, StrEq("com.android.lib_one"));
115 EXPECT_THAT(dynamic_pkg_map[0].package_id, Eq(0x02));
Adam Lesinskida431a22016-12-29 16:08:16 -0500116
Adam Lesinskibebfcc42018-02-12 14:27:46 -0800117 EXPECT_THAT(dynamic_pkg_map[1].package_name, StrEq("com.android.lib_two"));
118 EXPECT_THAT(dynamic_pkg_map[1].package_id, Eq(0x03));
Adam Lesinskida431a22016-12-29 16:08:16 -0500119}
120
121TEST(LoadedArscTest, LoadAppAsSharedLibrary) {
122 std::string contents;
123 ASSERT_TRUE(ReadFileFromZipToString(GetTestDataPath() + "/appaslib/appaslib.apk",
124 "resources.arsc", &contents));
125
Ryan Mitchelldb21f09a2020-11-16 23:08:18 +0000126 std::unique_ptr<const LoadedArsc> loaded_arsc = LoadedArsc::Load(contents.data(),
127 contents.length(),
128 nullptr /* loaded_idmap */,
129 PROPERTY_DYNAMIC);
Adam Lesinskibebfcc42018-02-12 14:27:46 -0800130 ASSERT_THAT(loaded_arsc, NotNull());
Adam Lesinskida431a22016-12-29 16:08:16 -0500131
132 const auto& packages = loaded_arsc->GetPackages();
Adam Lesinskibebfcc42018-02-12 14:27:46 -0800133 ASSERT_THAT(packages, SizeIs(1u));
Adam Lesinskida431a22016-12-29 16:08:16 -0500134 EXPECT_TRUE(packages[0]->IsDynamic());
Adam Lesinskibebfcc42018-02-12 14:27:46 -0800135 EXPECT_THAT(packages[0]->GetPackageId(), Eq(0x7f));
Adam Lesinskida431a22016-12-29 16:08:16 -0500136}
137
Adam Lesinskic6aada92017-01-13 15:34:14 -0800138TEST(LoadedArscTest, LoadFeatureSplit) {
139 std::string contents;
140 ASSERT_TRUE(ReadFileFromZipToString(GetTestDataPath() + "/feature/feature.apk", "resources.arsc",
141 &contents));
Ryan Mitchelldb21f09a2020-11-16 23:08:18 +0000142 std::unique_ptr<const LoadedArsc> loaded_arsc = LoadedArsc::Load(contents.data(),
143 contents.length());
Adam Lesinskibebfcc42018-02-12 14:27:46 -0800144 ASSERT_THAT(loaded_arsc, NotNull());
Adam Lesinskic6aada92017-01-13 15:34:14 -0800145
Adam Lesinskibebfcc42018-02-12 14:27:46 -0800146 const LoadedPackage* package =
147 loaded_arsc->GetPackageById(get_package_id(basic::R::string::test3));
148 ASSERT_THAT(package, NotNull());
Adam Lesinskic6aada92017-01-13 15:34:14 -0800149
Adam Lesinskibebfcc42018-02-12 14:27:46 -0800150 uint8_t type_index = get_type_id(basic::R::string::test3) - 1;
151 uint8_t entry_index = get_entry_id(basic::R::string::test3);
152
153 const TypeSpec* type_spec = package->GetTypeSpecByTypeIndex(type_index);
154 ASSERT_THAT(type_spec, NotNull());
Ryan Mitchell14e8ade2021-01-11 16:01:35 -0800155 ASSERT_THAT(type_spec->type_entries.size(), Ge(1u));
Adam Lesinskic6aada92017-01-13 15:34:14 -0800156
Ryan Mitchelldb21f09a2020-11-16 23:08:18 +0000157 auto type_name16 = package->GetTypeStringPool()->stringAt(type_spec->type_spec->id - 1);
158 ASSERT_TRUE(type_name16.has_value());
159 EXPECT_THAT(util::Utf16ToUtf8(*type_name16), StrEq("string"));
Adam Lesinskic6aada92017-01-13 15:34:14 -0800160
Ryan Mitchell14e8ade2021-01-11 16:01:35 -0800161 ASSERT_TRUE(LoadedPackage::GetEntry(type_spec->type_entries[0].type, entry_index).has_value());
Adam Lesinskibebfcc42018-02-12 14:27:46 -0800162}
163
164// AAPT(2) generates resource tables with chunks in a certain order. The rule is that
165// a RES_TABLE_TYPE_TYPE with id `i` must always be preceded by a RES_TABLE_TYPE_SPEC_TYPE with
166// id `i`. The RES_TABLE_TYPE_SPEC_TYPE does not need to be directly preceding, however.
167//
168// AAPT(2) generates something like:
169// RES_TABLE_TYPE_SPEC_TYPE id=1
170// RES_TABLE_TYPE_TYPE id=1
171// RES_TABLE_TYPE_SPEC_TYPE id=2
172// RES_TABLE_TYPE_TYPE id=2
173//
174// But the following is valid too:
175// RES_TABLE_TYPE_SPEC_TYPE id=1
176// RES_TABLE_TYPE_SPEC_TYPE id=2
177// RES_TABLE_TYPE_TYPE id=1
178// RES_TABLE_TYPE_TYPE id=2
179//
180TEST(LoadedArscTest, LoadOutOfOrderTypeSpecs) {
181 std::string contents;
182 ASSERT_TRUE(
183 ReadFileFromZipToString(GetTestDataPath() + "/out_of_order_types/out_of_order_types.apk",
184 "resources.arsc", &contents));
185
Ryan Mitchelldb21f09a2020-11-16 23:08:18 +0000186 std::unique_ptr<const LoadedArsc> loaded_arsc = LoadedArsc::Load(contents.data(),
187 contents.length());
Adam Lesinskibebfcc42018-02-12 14:27:46 -0800188 ASSERT_THAT(loaded_arsc, NotNull());
189
190 ASSERT_THAT(loaded_arsc->GetPackages(), SizeIs(1u));
191 const auto& package = loaded_arsc->GetPackages()[0];
192 ASSERT_THAT(package, NotNull());
193
194 const TypeSpec* type_spec = package->GetTypeSpecByTypeIndex(0);
195 ASSERT_THAT(type_spec, NotNull());
Ryan Mitchell14e8ade2021-01-11 16:01:35 -0800196 ASSERT_THAT(type_spec->type_entries.size(), Ge(1u));
Adam Lesinskibebfcc42018-02-12 14:27:46 -0800197
198 type_spec = package->GetTypeSpecByTypeIndex(1);
199 ASSERT_THAT(type_spec, NotNull());
Ryan Mitchell14e8ade2021-01-11 16:01:35 -0800200 ASSERT_THAT(type_spec->type_entries.size(), Ge(1u));
Adam Lesinskic6aada92017-01-13 15:34:14 -0800201}
202
Ryan Mitchell75e20dd2018-11-06 16:39:36 -0800203TEST(LoadedArscTest, LoadOverlayable) {
204 std::string contents;
205 ASSERT_TRUE(ReadFileFromZipToString(GetTestDataPath() + "/overlayable/overlayable.apk",
206 "resources.arsc", &contents));
207
Ryan Mitchelldb21f09a2020-11-16 23:08:18 +0000208 std::unique_ptr<const LoadedArsc> loaded_arsc = LoadedArsc::Load(contents.data(),
209 contents.length());
Ryan Mitchell75e20dd2018-11-06 16:39:36 -0800210
211 ASSERT_THAT(loaded_arsc, NotNull());
212 const LoadedPackage* package = loaded_arsc->GetPackageById(
213 get_package_id(overlayable::R::string::not_overlayable));
214
215 const OverlayableInfo* info = package->GetOverlayableInfo(
216 overlayable::R::string::not_overlayable);
217 ASSERT_THAT(info, IsNull());
218
219 info = package->GetOverlayableInfo(overlayable::R::string::overlayable1);
220 ASSERT_THAT(info, NotNull());
Ryan Mitchellef5673a2018-12-12 18:45:34 -0800221 EXPECT_THAT(info->name, Eq("OverlayableResources1"));
222 EXPECT_THAT(info->actor, Eq("overlay://theme"));
Winson62ac8b52019-12-04 08:36:48 -0800223 EXPECT_THAT(info->policy_flags, Eq(PolicyFlags::PUBLIC));
Ryan Mitchell75e20dd2018-11-06 16:39:36 -0800224
225 info = package->GetOverlayableInfo(overlayable::R::string::overlayable2);
226 ASSERT_THAT(info, NotNull());
Ryan Mitchellef5673a2018-12-12 18:45:34 -0800227 EXPECT_THAT(info->name, Eq("OverlayableResources1"));
228 EXPECT_THAT(info->actor, Eq("overlay://theme"));
Ryan Mitchell75e20dd2018-11-06 16:39:36 -0800229 EXPECT_THAT(info->policy_flags,
Winson62ac8b52019-12-04 08:36:48 -0800230 Eq(PolicyFlags::SYSTEM_PARTITION
231 | PolicyFlags::PRODUCT_PARTITION));
Ryan Mitchell75e20dd2018-11-06 16:39:36 -0800232
233 info = package->GetOverlayableInfo(overlayable::R::string::overlayable3);
234 ASSERT_THAT(info, NotNull());
Ryan Mitchellef5673a2018-12-12 18:45:34 -0800235 EXPECT_THAT(info->name, Eq("OverlayableResources2"));
236 EXPECT_THAT(info->actor, Eq("overlay://com.android.overlayable"));
Ryan Mitchell75e20dd2018-11-06 16:39:36 -0800237 EXPECT_THAT(info->policy_flags,
Winson62ac8b52019-12-04 08:36:48 -0800238 Eq(PolicyFlags::VENDOR_PARTITION
239 | PolicyFlags::PRODUCT_PARTITION));
Ryan Mitchell75e20dd2018-11-06 16:39:36 -0800240
241 info = package->GetOverlayableInfo(overlayable::R::string::overlayable4);
Ryan Mitchellef5673a2018-12-12 18:45:34 -0800242 EXPECT_THAT(info->name, Eq("OverlayableResources1"));
243 EXPECT_THAT(info->actor, Eq("overlay://theme"));
Ryan Mitchell75e20dd2018-11-06 16:39:36 -0800244 ASSERT_THAT(info, NotNull());
Winson62ac8b52019-12-04 08:36:48 -0800245 EXPECT_THAT(info->policy_flags, Eq(PolicyFlags::PUBLIC));
Ryan Mitchell75e20dd2018-11-06 16:39:36 -0800246}
Adam Lesinski7ad11102016-10-28 16:39:15 -0700247
MÃ¥rten Kongstad3f1f4fc2018-03-02 09:34:18 +0100248TEST(LoadedArscTest, ResourceIdentifierIterator) {
249 std::string contents;
250 ASSERT_TRUE(
251 ReadFileFromZipToString(GetTestDataPath() + "/basic/basic.apk", "resources.arsc", &contents));
252
Ryan Mitchelldb21f09a2020-11-16 23:08:18 +0000253 std::unique_ptr<const LoadedArsc> loaded_arsc = LoadedArsc::Load(contents.data(),
254 contents.length());
MÃ¥rten Kongstad3f1f4fc2018-03-02 09:34:18 +0100255 ASSERT_NE(nullptr, loaded_arsc);
256
257 const std::vector<std::unique_ptr<const LoadedPackage>>& packages = loaded_arsc->GetPackages();
258 ASSERT_EQ(1u, packages.size());
MÃ¥rten Kongstadc92c4dd2019-02-05 01:29:59 +0100259 ASSERT_EQ(std::string("com.android.basic"), packages[0]->GetPackageName());
MÃ¥rten Kongstad3f1f4fc2018-03-02 09:34:18 +0100260
261 const auto& loaded_package = packages[0];
262 auto iter = loaded_package->begin();
263 auto end = loaded_package->end();
264
265 ASSERT_NE(end, iter);
266 ASSERT_EQ(0x7f010000u, *iter++);
267 ASSERT_EQ(0x7f010001u, *iter++);
268 ASSERT_EQ(0x7f020000u, *iter++);
269 ASSERT_EQ(0x7f020001u, *iter++);
270 ASSERT_EQ(0x7f030000u, *iter++);
271 ASSERT_EQ(0x7f030001u, *iter++);
272 ASSERT_EQ(0x7f030002u, *iter++); // note: string without default, excluded by aapt2 dump
273 ASSERT_EQ(0x7f040000u, *iter++);
274 ASSERT_EQ(0x7f040001u, *iter++);
275 ASSERT_EQ(0x7f040002u, *iter++);
276 ASSERT_EQ(0x7f040003u, *iter++);
277 ASSERT_EQ(0x7f040004u, *iter++);
278 ASSERT_EQ(0x7f040005u, *iter++);
279 ASSERT_EQ(0x7f040006u, *iter++);
280 ASSERT_EQ(0x7f040007u, *iter++);
281 ASSERT_EQ(0x7f040008u, *iter++);
282 ASSERT_EQ(0x7f040009u, *iter++);
283 ASSERT_EQ(0x7f04000au, *iter++);
284 ASSERT_EQ(0x7f04000bu, *iter++);
285 ASSERT_EQ(0x7f04000cu, *iter++);
286 ASSERT_EQ(0x7f04000du, *iter++);
287 ASSERT_EQ(0x7f050000u, *iter++);
288 ASSERT_EQ(0x7f050001u, *iter++);
289 ASSERT_EQ(0x7f060000u, *iter++);
290 ASSERT_EQ(0x7f070000u, *iter++);
291 ASSERT_EQ(0x7f070001u, *iter++);
292 ASSERT_EQ(0x7f070002u, *iter++);
293 ASSERT_EQ(0x7f070003u, *iter++);
294 ASSERT_EQ(end, iter);
295}
296
MÃ¥rten Kongstadc92c4dd2019-02-05 01:29:59 +0100297TEST(LoadedArscTest, GetOverlayableMap) {
298 std::string contents;
299 ASSERT_TRUE(ReadFileFromZipToString(GetTestDataPath() + "/overlayable/overlayable.apk",
300 "resources.arsc", &contents));
301
Ryan Mitchelldb21f09a2020-11-16 23:08:18 +0000302 std::unique_ptr<const LoadedArsc> loaded_arsc = LoadedArsc::Load(contents.data(),
303 contents.length());
MÃ¥rten Kongstadc92c4dd2019-02-05 01:29:59 +0100304 ASSERT_NE(nullptr, loaded_arsc);
305
306 const std::vector<std::unique_ptr<const LoadedPackage>>& packages = loaded_arsc->GetPackages();
307 ASSERT_EQ(1u, packages.size());
308 ASSERT_EQ(std::string("com.android.overlayable"), packages[0]->GetPackageName());
309
310 const auto map = packages[0]->GetOverlayableMap();
Ryan Mitchell8a891d82019-07-01 09:48:23 -0700311 ASSERT_EQ(3, map.size());
MÃ¥rten Kongstadc92c4dd2019-02-05 01:29:59 +0100312 ASSERT_EQ(map.at("OverlayableResources1"), "overlay://theme");
313 ASSERT_EQ(map.at("OverlayableResources2"), "overlay://com.android.overlayable");
Ryan Mitchell8a891d82019-07-01 09:48:23 -0700314 ASSERT_EQ(map.at("OverlayableResources3"), "");
MÃ¥rten Kongstadc92c4dd2019-02-05 01:29:59 +0100315}
316
Winson9947f1e2019-08-16 10:20:39 -0700317TEST(LoadedArscTest, LoadCustomLoader) {
Ryan Mitchell1a48fa62021-01-10 08:36:36 -0800318 auto asset = AssetsProvider::CreateAssetFromFile(GetTestDataPath() + "/loader/resources.arsc");
319 ASSERT_THAT(asset, NotNull());
Winson9947f1e2019-08-16 10:20:39 -0700320
Winson9947f1e2019-08-16 10:20:39 -0700321 const StringPiece data(
322 reinterpret_cast<const char*>(asset->getBuffer(true /*wordAligned*/)),
323 asset->getLength());
324
325 std::unique_ptr<const LoadedArsc> loaded_arsc =
Ryan Mitchelldb21f09a2020-11-16 23:08:18 +0000326 LoadedArsc::Load(data.data(), data.length(), nullptr, PROPERTY_LOADER);
Winson9947f1e2019-08-16 10:20:39 -0700327 ASSERT_THAT(loaded_arsc, NotNull());
328
329 const LoadedPackage* package =
Ryan Mitchell8a891d82019-07-01 09:48:23 -0700330 loaded_arsc->GetPackageById(get_package_id(overlayable::R::string::overlayable11));
Winson9947f1e2019-08-16 10:20:39 -0700331 ASSERT_THAT(package, NotNull());
Ryan Mitchell8a891d82019-07-01 09:48:23 -0700332 EXPECT_THAT(package->GetPackageName(), StrEq("com.android.loader"));
333 EXPECT_THAT(package->GetPackageId(), Eq(0x7f));
Winson9947f1e2019-08-16 10:20:39 -0700334
Ryan Mitchell8a891d82019-07-01 09:48:23 -0700335 const uint8_t type_index = get_type_id(overlayable::R::string::overlayable11) - 1;
336 const uint16_t entry_index = get_entry_id(overlayable::R::string::overlayable11);
Winson9947f1e2019-08-16 10:20:39 -0700337
338 const TypeSpec* type_spec = package->GetTypeSpecByTypeIndex(type_index);
339 ASSERT_THAT(type_spec, NotNull());
Ryan Mitchell14e8ade2021-01-11 16:01:35 -0800340 ASSERT_THAT(type_spec->type_entries.size(), Ge(1u));
Winson9947f1e2019-08-16 10:20:39 -0700341
Ryan Mitchell14e8ade2021-01-11 16:01:35 -0800342 auto type = type_spec->type_entries[0];
343 ASSERT_TRUE(LoadedPackage::GetEntry(type.type, entry_index).has_value());
Winson9947f1e2019-08-16 10:20:39 -0700344}
345
Ryan Mitchell75e20dd2018-11-06 16:39:36 -0800346// structs with size fields (like Res_value, ResTable_entry) should be
347// backwards and forwards compatible (aka checking the size field against
348// sizeof(Res_value) might not be backwards compatible.
Aurimas Liutikas6c2a8b482018-12-14 10:12:27 -0800349// TEST(LoadedArscTest, LoadingShouldBeForwardsAndBackwardsCompatible) { ASSERT_TRUE(false); }
Ryan Mitchell75e20dd2018-11-06 16:39:36 -0800350
Brandon Liu742b11e2022-11-03 23:23:28 +0000351class LoadedArscParameterizedTest :
352 public testing::TestWithParam<std::string> {
353};
354
355TEST_P(LoadedArscParameterizedTest, LoadSparseEntryApp) {
356 std::string contents;
357 ASSERT_TRUE(ReadFileFromZipToString(GetParam(), "resources.arsc", &contents));
358
359 std::unique_ptr<const LoadedArsc> loaded_arsc = LoadedArsc::Load(contents.data(),
360 contents.length());
361 ASSERT_THAT(loaded_arsc, NotNull());
362
363 const LoadedPackage* package =
364 loaded_arsc->GetPackageById(get_package_id(sparse::R::integer::foo_9));
365 ASSERT_THAT(package, NotNull());
366
367 const uint8_t type_index = get_type_id(sparse::R::integer::foo_9) - 1;
368 const uint16_t entry_index = get_entry_id(sparse::R::integer::foo_9);
369
370 const TypeSpec* type_spec = package->GetTypeSpecByTypeIndex(type_index);
371 ASSERT_THAT(type_spec, NotNull());
372 ASSERT_THAT(type_spec->type_entries.size(), Ge(1u));
373
374 auto type = type_spec->type_entries[0];
375 ASSERT_TRUE(LoadedPackage::GetEntry(type.type, entry_index).has_value());
376}
377
378TEST_P(LoadedArscParameterizedTest, FindSparseEntryApp) {
379 std::string contents;
380 ASSERT_TRUE(ReadFileFromZipToString(GetParam(), "resources.arsc", &contents));
381
382 std::unique_ptr<const LoadedArsc> loaded_arsc = LoadedArsc::Load(contents.data(),
383 contents.length());
384 ASSERT_THAT(loaded_arsc, NotNull());
385
386 const LoadedPackage* package =
387 loaded_arsc->GetPackageById(get_package_id(sparse::R::string::only_land));
388 ASSERT_THAT(package, NotNull());
389
390 const uint8_t type_index = get_type_id(sparse::R::string::only_land) - 1;
391
392 const TypeSpec* type_spec = package->GetTypeSpecByTypeIndex(type_index);
393 ASSERT_THAT(type_spec, NotNull());
394 ASSERT_THAT(type_spec->type_entries.size(), Ge(1u));
395
396 // Type Entry with default orientation is not sparse encoded because the ratio of
397 // populated entries to total entries is above threshold.
398 // Only find out default locale because Soong build system will introduce pseudo
399 // locales for the apk generated at runtime.
400 auto type_entry_default = std::find_if(
401 type_spec->type_entries.begin(), type_spec->type_entries.end(),
402 [] (const TypeSpec::TypeEntry& x) { return x.config.orientation == 0 &&
403 x.config.locale == 0; });
404 ASSERT_NE(type_entry_default, type_spec->type_entries.end());
405 ASSERT_EQ(type_entry_default->type->flags & ResTable_type::FLAG_SPARSE, 0);
406
407 // Type Entry with land orientation is sparse encoded as expected.
408 // Only find out default locale because Soong build system will introduce pseudo
409 // locales for the apk generated at runtime.
410 auto type_entry_land = std::find_if(
411 type_spec->type_entries.begin(), type_spec->type_entries.end(),
412 [](const TypeSpec::TypeEntry& x) { return x.config.orientation ==
413 ResTable_config::ORIENTATION_LAND &&
414 x.config.locale == 0; });
415 ASSERT_NE(type_entry_land, type_spec->type_entries.end());
416 ASSERT_NE(type_entry_land->type->flags & ResTable_type::FLAG_SPARSE, 0);
417
418 // Test fetching a resource with only sparsely encoded configs by name.
419 auto id = package->FindEntryByName(u"string", u"only_land");
420 ASSERT_EQ(id.value(), fix_package_id(sparse::R::string::only_land, 0));
421}
422
423INSTANTIATE_TEST_SUITE_P(
424 FrameWorkResourcesLoadedArscTests,
425 LoadedArscParameterizedTest,
426 ::testing::Values(
427 base::GetExecutableDirectory() + "/tests/data/sparse/sparse.apk",
428 base::GetExecutableDirectory() + "/FrameworkResourcesSparseTestApp.apk"
429 ));
430
Adam Lesinski7ad11102016-10-28 16:39:15 -0700431} // namespace android