blob: 2b72d146f83ab48d238b4d595c3d6ae242366908 [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 Lesinski7ad11102016-10-28 16:39:15 -070019#include "TestHelpers.h"
20#include "data/basic/R.h"
Adam Lesinskida431a22016-12-29 16:08:16 -050021#include "data/libclient/R.h"
Adam Lesinski7ad11102016-10-28 16:39:15 -070022#include "data/styles/R.h"
23
24namespace app = com::android::app;
25namespace basic = com::android::basic;
Adam Lesinskida431a22016-12-29 16:08:16 -050026namespace libclient = com::android::libclient;
Adam Lesinski7ad11102016-10-28 16:39:15 -070027
28namespace android {
29
30TEST(LoadedArscTest, LoadSinglePackageArsc) {
Adam Lesinski7ad11102016-10-28 16:39:15 -070031 std::string contents;
32 ASSERT_TRUE(ReadFileFromZipToString(GetTestDataPath() + "/styles/styles.apk", "resources.arsc",
33 &contents));
34
Adam Lesinski970bd8d2017-09-25 13:21:55 -070035 std::unique_ptr<const LoadedArsc> loaded_arsc = LoadedArsc::Load(StringPiece(contents));
Adam Lesinski7ad11102016-10-28 16:39:15 -070036 ASSERT_NE(nullptr, loaded_arsc);
37
Adam Lesinskida431a22016-12-29 16:08:16 -050038 const std::vector<std::unique_ptr<const LoadedPackage>>& packages = loaded_arsc->GetPackages();
39 ASSERT_EQ(1u, packages.size());
40 EXPECT_EQ(std::string("com.android.app"), packages[0]->GetPackageName());
41 EXPECT_EQ(0x7f, packages[0]->GetPackageId());
42
Adam Lesinski7ad11102016-10-28 16:39:15 -070043 ResTable_config config;
44 memset(&config, 0, sizeof(config));
45 config.sdkVersion = 24;
46
Adam Lesinskida431a22016-12-29 16:08:16 -050047 LoadedArscEntry entry;
Adam Lesinski7ad11102016-10-28 16:39:15 -070048 ResTable_config selected_config;
49 uint32_t flags;
50
51 ASSERT_TRUE(
52 loaded_arsc->FindEntry(app::R::string::string_one, config, &entry, &selected_config, &flags));
53 ASSERT_NE(nullptr, entry.entry);
54}
55
56TEST(LoadedArscTest, FindDefaultEntry) {
Adam Lesinski7ad11102016-10-28 16:39:15 -070057 std::string contents;
58 ASSERT_TRUE(
59 ReadFileFromZipToString(GetTestDataPath() + "/basic/basic.apk", "resources.arsc", &contents));
60
Adam Lesinski970bd8d2017-09-25 13:21:55 -070061 std::unique_ptr<const LoadedArsc> loaded_arsc = LoadedArsc::Load(StringPiece(contents));
Adam Lesinski7ad11102016-10-28 16:39:15 -070062 ASSERT_NE(nullptr, loaded_arsc);
63
64 ResTable_config desired_config;
65 memset(&desired_config, 0, sizeof(desired_config));
66 desired_config.language[0] = 'd';
67 desired_config.language[1] = 'e';
68
Adam Lesinskida431a22016-12-29 16:08:16 -050069 LoadedArscEntry entry;
Adam Lesinski7ad11102016-10-28 16:39:15 -070070 ResTable_config selected_config;
71 uint32_t flags;
72
73 ASSERT_TRUE(loaded_arsc->FindEntry(basic::R::string::test1, desired_config, &entry,
74 &selected_config, &flags));
75 ASSERT_NE(nullptr, entry.entry);
76}
77
Adam Lesinskida431a22016-12-29 16:08:16 -050078TEST(LoadedArscTest, LoadSharedLibrary) {
79 std::string contents;
80 ASSERT_TRUE(ReadFileFromZipToString(GetTestDataPath() + "/lib_one/lib_one.apk", "resources.arsc",
81 &contents));
82
Adam Lesinski970bd8d2017-09-25 13:21:55 -070083 std::unique_ptr<const LoadedArsc> loaded_arsc = LoadedArsc::Load(StringPiece(contents));
Adam Lesinskida431a22016-12-29 16:08:16 -050084 ASSERT_NE(nullptr, loaded_arsc);
85
86 const auto& packages = loaded_arsc->GetPackages();
87 ASSERT_EQ(1u, packages.size());
88
89 EXPECT_TRUE(packages[0]->IsDynamic());
90 EXPECT_EQ(std::string("com.android.lib_one"), packages[0]->GetPackageName());
91 EXPECT_EQ(0, packages[0]->GetPackageId());
92
93 const auto& dynamic_pkg_map = packages[0]->GetDynamicPackageMap();
94
95 // The library has no dependencies.
96 ASSERT_TRUE(dynamic_pkg_map.empty());
97}
98
99TEST(LoadedArscTest, LoadAppLinkedAgainstSharedLibrary) {
100 std::string contents;
101 ASSERT_TRUE(ReadFileFromZipToString(GetTestDataPath() + "/libclient/libclient.apk",
102 "resources.arsc", &contents));
103
Adam Lesinski970bd8d2017-09-25 13:21:55 -0700104 std::unique_ptr<const LoadedArsc> loaded_arsc = LoadedArsc::Load(StringPiece(contents));
Adam Lesinskida431a22016-12-29 16:08:16 -0500105 ASSERT_NE(nullptr, loaded_arsc);
106
107 const auto& packages = loaded_arsc->GetPackages();
108 ASSERT_EQ(1u, packages.size());
109
110 EXPECT_FALSE(packages[0]->IsDynamic());
111 EXPECT_EQ(std::string("com.android.libclient"), packages[0]->GetPackageName());
112 EXPECT_EQ(0x7f, packages[0]->GetPackageId());
113
114 const auto& dynamic_pkg_map = packages[0]->GetDynamicPackageMap();
115
116 // The library has two dependencies.
117 ASSERT_EQ(2u, dynamic_pkg_map.size());
118
119 EXPECT_EQ(std::string("com.android.lib_one"), dynamic_pkg_map[0].package_name);
120 EXPECT_EQ(0x02, dynamic_pkg_map[0].package_id);
121
122 EXPECT_EQ(std::string("com.android.lib_two"), dynamic_pkg_map[1].package_name);
123 EXPECT_EQ(0x03, dynamic_pkg_map[1].package_id);
124}
125
126TEST(LoadedArscTest, LoadAppAsSharedLibrary) {
127 std::string contents;
128 ASSERT_TRUE(ReadFileFromZipToString(GetTestDataPath() + "/appaslib/appaslib.apk",
129 "resources.arsc", &contents));
130
Adam Lesinski970bd8d2017-09-25 13:21:55 -0700131 std::unique_ptr<const LoadedArsc> loaded_arsc =
132 LoadedArsc::Load(StringPiece(contents), nullptr /*loaded_idmap*/, false /*system*/,
133 true /*load_as_shared_library*/);
Adam Lesinskida431a22016-12-29 16:08:16 -0500134 ASSERT_NE(nullptr, loaded_arsc);
135
136 const auto& packages = loaded_arsc->GetPackages();
137 ASSERT_EQ(1u, packages.size());
138
139 EXPECT_TRUE(packages[0]->IsDynamic());
140 EXPECT_EQ(0x7f, packages[0]->GetPackageId());
141}
142
Adam Lesinskic6aada92017-01-13 15:34:14 -0800143TEST(LoadedArscTest, LoadFeatureSplit) {
144 std::string contents;
145 ASSERT_TRUE(ReadFileFromZipToString(GetTestDataPath() + "/feature/feature.apk", "resources.arsc",
146 &contents));
Adam Lesinski970bd8d2017-09-25 13:21:55 -0700147 std::unique_ptr<const LoadedArsc> loaded_arsc = LoadedArsc::Load(StringPiece(contents));
Adam Lesinskic6aada92017-01-13 15:34:14 -0800148 ASSERT_NE(nullptr, loaded_arsc);
149
150 ResTable_config desired_config;
151 memset(&desired_config, 0, sizeof(desired_config));
152
153 LoadedArscEntry entry;
154 ResTable_config selected_config;
155 uint32_t flags;
156
157 ASSERT_TRUE(loaded_arsc->FindEntry(basic::R::string::test3, desired_config, &entry,
158 &selected_config, &flags));
159
160 size_t len;
161 const char16_t* type_name16 = entry.type_string_ref.string16(&len);
162 ASSERT_NE(nullptr, type_name16);
163 ASSERT_NE(0u, len);
164
165 size_t utf8_len = utf16_to_utf8_length(type_name16, len);
166 std::string type_name;
167 type_name.resize(utf8_len);
168 utf16_to_utf8(type_name16, len, &*type_name.begin(), utf8_len + 1);
169
170 EXPECT_EQ(std::string("string"), type_name);
171}
172
Adam Lesinski970bd8d2017-09-25 13:21:55 -0700173class MockLoadedIdmap : public LoadedIdmap {
174 public:
175 MockLoadedIdmap() : LoadedIdmap() {
176 local_header_.magic = kIdmapMagic;
177 local_header_.version = kIdmapCurrentVersion;
178 local_header_.target_package_id = 0x08;
179 local_header_.type_count = 1;
180 header_ = &local_header_;
181
182 entry_header = util::unique_cptr<IdmapEntry_header>(
183 (IdmapEntry_header*)::malloc(sizeof(IdmapEntry_header) + sizeof(uint32_t)));
184 entry_header->target_type_id = 0x03;
185 entry_header->overlay_type_id = 0x02;
186 entry_header->entry_id_offset = 1;
187 entry_header->entry_count = 1;
188 entry_header->entries[0] = 0x00000000u;
189 type_map_[entry_header->overlay_type_id] = entry_header.get();
190 }
191
192 private:
193 Idmap_header local_header_;
194 util::unique_cptr<IdmapEntry_header> entry_header;
195};
196
197TEST(LoadedArscTest, LoadOverlay) {
198 std::string contents, overlay_contents;
199 ASSERT_TRUE(
200 ReadFileFromZipToString(GetTestDataPath() + "/basic/basic.apk", "resources.arsc", &contents));
201 ASSERT_TRUE(ReadFileFromZipToString(GetTestDataPath() + "/overlay/overlay.apk", "resources.arsc",
202 &overlay_contents));
203
204 MockLoadedIdmap loaded_idmap;
205
206 std::unique_ptr<const LoadedArsc> loaded_arsc =
207 LoadedArsc::Load(StringPiece(overlay_contents), &loaded_idmap);
208 ASSERT_NE(nullptr, loaded_arsc);
209
210 ResTable_config desired_config;
211 memset(&desired_config, 0, sizeof(desired_config));
212
213 LoadedArscEntry entry;
214 ResTable_config selected_config;
215 uint32_t flags;
216
217 ASSERT_TRUE(
218 loaded_arsc->FindEntry(0x08030001u, desired_config, &entry, &selected_config, &flags));
219}
220
Adam Lesinski7ad11102016-10-28 16:39:15 -0700221// structs with size fields (like Res_value, ResTable_entry) should be
222// backwards and forwards compatible (aka checking the size field against
223// sizeof(Res_value) might not be backwards compatible.
224TEST(LoadedArscTest, LoadingShouldBeForwardsAndBackwardsCompatible) { ASSERT_TRUE(false); }
225
226} // namespace android