1 /*
2  * Copyright (C) 2018 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 <android-base/file.h>
18 #include <androidfw/ResourceTypes.h>
19 #include <gtest/gtest.h>
20 
21 #include <cstdio>  // fclose
22 #include <fstream>
23 #include <memory>
24 #include <string>
25 
26 #include <fcntl.h>
27 #include "R.h"
28 #include "TestConstants.h"
29 #include "TestHelpers.h"
30 #include "idmap2/LogInfo.h"
31 #include "idmap2/ResourceMapping.h"
32 
33 using PolicyFlags = android::ResTable_overlayable_policy_header::PolicyFlags;
34 
35 namespace android::idmap2 {
36 
37 #define ASSERT_RESULT(r)                             \
38   do {                                               \
39     auto result = r;                                 \
40     ASSERT_TRUE(result) << result.GetErrorMessage(); \
41   } while (0)
42 
TestGetResourceMapping(const std::string & local_target_path,const std::string & local_overlay_path,const std::string & overlay_name,const PolicyBitmask & fulfilled_policies,bool enforce_overlayable)43 Result<ResourceMapping> TestGetResourceMapping(const std::string& local_target_path,
44                                                const std::string& local_overlay_path,
45                                                const std::string& overlay_name,
46                                                const PolicyBitmask& fulfilled_policies,
47                                                bool enforce_overlayable) {
48   const std::string target_path = (local_target_path[0] == '/')
49                                       ? local_target_path
50                                       : (GetTestDataPath() + "/" + local_target_path);
51   auto target = TargetResourceContainer::FromPath(target_path);
52   if (!target) {
53     return Error(target.GetError(), R"(Failed to load target "%s")", target_path.c_str());
54   }
55 
56   const std::string overlay_path = (local_overlay_path[0] == '/')
57                                        ? local_overlay_path
58                                        : (GetTestDataPath() + "/" + local_overlay_path);
59   auto overlay = OverlayResourceContainer::FromPath(overlay_path);
60   if (!overlay) {
61     return Error(overlay.GetError(), R"(Failed to load overlay "%s")", overlay_path.c_str());
62   }
63 
64   auto overlay_info = (*overlay)->FindOverlayInfo(overlay_name);
65   if (!overlay_info) {
66     return Error(overlay_info.GetError(), R"(Failed to find overlay name "%s")",
67                  overlay_name.c_str());
68   }
69 
70   LogInfo log_info;
71   return ResourceMapping::FromContainers(**target, **overlay, *overlay_info, fulfilled_policies,
72                                          enforce_overlayable, log_info);
73 }
74 
MappingExists(const ResourceMapping & mapping,ResourceId target_resource,ResourceId overlay_resource,bool rewrite)75 Result<Unit> MappingExists(const ResourceMapping& mapping, ResourceId target_resource,
76                            ResourceId overlay_resource, bool rewrite) {
77   auto target_map = mapping.GetTargetToOverlayMap();
78   auto entry_map = target_map.find(target_resource);
79   if (entry_map == target_map.end()) {
80     std::string keys;
81     for (const auto &pair : target_map) {
82       keys.append(fmt::format("0x{:x}", pair.first)).append(" ");
83     }
84     return Error(R"(Failed to find mapping for target resource "0x%02x": "%s")",
85         target_resource, keys.c_str());
86   }
87 
88   auto actual_overlay_resource = std::get_if<ResourceId>(&entry_map->second);
89   if (actual_overlay_resource == nullptr) {
90     return Error("Target resource is not mapped to an overlay resource id");
91   }
92 
93   if (*actual_overlay_resource != overlay_resource) {
94     return Error(R"(Expected id: "0x%02x" Actual id: "0x%02x")", overlay_resource,
95                  *actual_overlay_resource);
96   }
97 
98   auto overlay_map = mapping.GetOverlayToTargetMap();
99   auto overlay_iter = overlay_map.find(overlay_resource);
100   if ((overlay_iter != overlay_map.end()) != rewrite) {
101     return Error(R"(Expected rewriting: "%s")", rewrite ? "true" : "false");
102   }
103 
104   if (rewrite && overlay_iter->second != target_resource) {
105     return Error(R"(Expected rewrite id: "0x%02x" Actual id: "0x%02x")", target_resource,
106                  overlay_iter->second);
107   }
108 
109   return Result<Unit>({});
110 }
111 
MappingExists(const ResourceMapping & mapping,const ResourceId & target_resource,const uint8_t type,const uint32_t value)112 Result<Unit> MappingExists(const ResourceMapping& mapping, const ResourceId& target_resource,
113                            const uint8_t type, const uint32_t value) {
114   auto target_map = mapping.GetTargetToOverlayMap();
115   auto entry_map = target_map.find(target_resource);
116   if (entry_map == target_map.end()) {
117     std::string keys;
118     for (const auto &pair : target_map) {
119       keys.append(fmt::format("{:x}", pair.first)).append(" ");
120     }
121     return Error(R"(Failed to find mapping for target resource "0x%02x": "%s")",
122         target_resource, keys.c_str());
123   }
124 
125   auto config_map = std::get_if<ConfigMap>(&entry_map->second);
126   if (config_map == nullptr || config_map->empty()) {
127     return Error("Target resource is not mapped to an inline value");
128   }
129   auto actual_overlay_value = config_map->begin()->second;
130 
131   if (actual_overlay_value.data_type != type) {
132     return Error(R"(Expected type: "0x%02x" Actual type: "0x%02x")", type,
133                  actual_overlay_value.data_type);
134   }
135 
136   if (actual_overlay_value.data_value != value) {
137     return Error(R"(Expected value: "0x%08x" Actual value: "0x%08x")", type,
138                  actual_overlay_value.data_value);
139   }
140 
141   return Result<Unit>({});
142 }
143 
TEST(ResourceMappingTests,ResourcesFromApkAssetsLegacy)144 TEST(ResourceMappingTests, ResourcesFromApkAssetsLegacy) {
145   auto resources = TestGetResourceMapping("target/target.apk", "overlay/overlay-legacy.apk", "",
146                                           PolicyFlags::PUBLIC, /* enforce_overlayable */ false);
147 
148   ASSERT_TRUE(resources) << resources.GetErrorMessage();
149   auto& res = *resources;
150   ASSERT_EQ(res.GetTargetToOverlayMap().size(), 4U);
151   ASSERT_RESULT(
152       MappingExists(res, R::target::integer::int1, R::overlay::integer::int1, false /* rewrite */));
153   ASSERT_RESULT(
154       MappingExists(res, R::target::string::str1, R::overlay::string::str1, false /* rewrite */));
155   ASSERT_RESULT(
156       MappingExists(res, R::target::string::str3, R::overlay::string::str3, false /* rewrite */));
157   ASSERT_RESULT(
158       MappingExists(res, R::target::string::str4, R::overlay::string::str4, false /* rewrite */));
159 }
160 
TEST(ResourceMappingTests,ResourcesFromApkAssetsNonMatchingNames)161 TEST(ResourceMappingTests, ResourcesFromApkAssetsNonMatchingNames) {
162   auto resources = TestGetResourceMapping("target/target.apk", "overlay/overlay.apk", "SwapNames",
163                                           PolicyFlags::PUBLIC,
164                                           /* enforce_overlayable */ false);
165 
166   ASSERT_TRUE(resources) << resources.GetErrorMessage();
167   auto& res = *resources;
168   ASSERT_EQ(res.GetTargetToOverlayMap().size(), 3U);
169   ASSERT_RESULT(
170       MappingExists(res, R::target::string::str1, R::overlay::string::str4, true /* rewrite */));
171   ASSERT_RESULT(
172       MappingExists(res, R::target::string::str3, R::overlay::string::str1, true /* rewrite */));
173   ASSERT_RESULT(
174       MappingExists(res, R::target::string::str4, R::overlay::string::str3, true /* rewrite */));
175 }
176 
TEST(ResourceMappingTests,DoNotRewriteNonOverlayResourceId)177 TEST(ResourceMappingTests, DoNotRewriteNonOverlayResourceId) {
178   auto resources = TestGetResourceMapping("target/target.apk", "overlay/overlay.apk",
179                                           "DifferentPackages", PolicyFlags::PUBLIC,
180                                           /* enforce_overlayable */ false);
181 
182   ASSERT_TRUE(resources) << resources.GetErrorMessage();
183   auto& res = *resources;
184   ASSERT_EQ(res.GetTargetToOverlayMap().size(), 2U);
185   ASSERT_EQ(res.GetOverlayToTargetMap().size(), 1U);
186   ASSERT_RESULT(MappingExists(res, R::target::string::str1, 0x0104000a,
187                               false /* rewrite */));  // -> android:string/ok
188   ASSERT_RESULT(
189       MappingExists(res, R::target::string::str3, R::overlay::string::str3, true /* rewrite */));
190 }
191 
TEST(ResourceMappingTests,InlineResources)192 TEST(ResourceMappingTests, InlineResources) {
193   auto resources = TestGetResourceMapping("target/target.apk", "overlay/overlay.apk", "Inline",
194                                           PolicyFlags::PUBLIC, /* enforce_overlayable */ false);
195 
196   constexpr size_t overlay_string_pool_size = 10U;
197   ASSERT_TRUE(resources) << resources.GetErrorMessage();
198   auto& res = *resources;
199   ASSERT_EQ(res.GetTargetToOverlayMap().size(), 2U);
200   ASSERT_EQ(res.GetOverlayToTargetMap().size(), 0U);
201   ASSERT_RESULT(MappingExists(res, R::target::string::str1, Res_value::TYPE_STRING,
202                               overlay_string_pool_size + 0U));  // -> "Hello World"
203   ASSERT_RESULT(MappingExists(res, R::target::integer::int1, Res_value::TYPE_INT_DEC, 73U));
204 }
205 
TEST(ResourceMappingTests,FabricatedOverlay)206 TEST(ResourceMappingTests, FabricatedOverlay) {
207   auto path = GetTestDataPath() + "/overlay/res/drawable/android.png";
208   auto fd = android::base::unique_fd(::open(path.c_str(), O_RDONLY | O_CLOEXEC));
209   ASSERT_TRUE(fd > 0) << "errno " << errno << " for path " << path;
210   auto frro = FabricatedOverlay::Builder("com.example.overlay", "SandTheme", "test.target")
211                   .SetOverlayable("TestResources")
212                   .SetResourceValue("integer/int1", Res_value::TYPE_INT_DEC, 2U, "")
213                   .SetResourceValue("string/str1", Res_value::TYPE_REFERENCE, 0x7f010000, "")
214                   .SetResourceValue("string/str2", Res_value::TYPE_STRING, "foobar", "")
215                   .SetResourceValue("drawable/dr1", fd, "")
216                   .setFrroPath("/foo/bar/biz.frro")
217                   .Build();
218 
219   ASSERT_TRUE(frro);
220   TemporaryFile tf;
221   std::ofstream out(tf.path);
222   ASSERT_TRUE((*frro).ToBinaryStream(out));
223   out.close();
224 
225   auto resources = TestGetResourceMapping("target/target.apk", tf.path, "SandTheme",
226                                           PolicyFlags::PUBLIC, /* enforce_overlayable */ false);
227 
228   ASSERT_TRUE(resources) << resources.GetErrorMessage();
229   auto& res = *resources;
230   auto string_pool_data = res.GetStringPoolData();
231   auto string_pool = ResStringPool(string_pool_data.data(), string_pool_data.size(), false);
232 
233   std::u16string expected_uri = u"frro://foo/bar/biz.frro?offset=16&size=8341";
234   uint32_t uri_index
235       = string_pool.indexOfString(expected_uri.data(), expected_uri.length()).value_or(-1);
236 
237   ASSERT_EQ(res.GetTargetToOverlayMap().size(), 4U);
238   ASSERT_EQ(res.GetOverlayToTargetMap().size(), 0U);
239   ASSERT_RESULT(MappingExists(res, R::target::string::str1, Res_value::TYPE_REFERENCE, 0x7f010000));
240   ASSERT_RESULT(MappingExists(res, R::target::string::str2, Res_value::TYPE_STRING,
241                               (uint32_t) (string_pool.indexOfString(u"foobar", 6)).value_or(-1)));
242   ASSERT_RESULT(MappingExists(res, R::target::drawable::dr1, Res_value::TYPE_STRING, uri_index));
243   ASSERT_RESULT(MappingExists(res, R::target::integer::int1, Res_value::TYPE_INT_DEC, 2U));
244 }
245 
TEST(ResourceMappingTests,CreateIdmapFromApkAssetsPolicySystemPublic)246 TEST(ResourceMappingTests, CreateIdmapFromApkAssetsPolicySystemPublic) {
247   auto resources = TestGetResourceMapping("target/target.apk", "overlay/overlay.apk",
248                                           TestConstants::OVERLAY_NAME_ALL_POLICIES,
249                                           PolicyFlags::SYSTEM_PARTITION | PolicyFlags::PUBLIC,
250                                           /* enforce_overlayable */ true);
251 
252   ASSERT_TRUE(resources) << resources.GetErrorMessage();
253   auto& res = *resources;
254   ASSERT_EQ(res.GetTargetToOverlayMap().size(), 3U);
255   ASSERT_RESULT(MappingExists(res, R::target::string::policy_public,
256                               R::overlay::string::policy_public, true /* rewrite */));
257   ASSERT_RESULT(MappingExists(res, R::target::string::policy_system,
258                               R::overlay::string::policy_system, true /* rewrite */));
259   ASSERT_RESULT(MappingExists(res, R::target::string::policy_system_vendor,
260                               R::overlay::string::policy_system_vendor, true /* rewrite */));
261 }
262 
263 // Resources that are not declared as overlayable and resources that a protected by policies the
264 // overlay does not fulfill must not map to overlay resources.
TEST(ResourceMappingTests,CreateIdmapFromApkAssetsPolicySystemPublicInvalid)265 TEST(ResourceMappingTests, CreateIdmapFromApkAssetsPolicySystemPublicInvalid) {
266   auto resources = TestGetResourceMapping("target/target.apk", "overlay/overlay.apk",
267                                           TestConstants::OVERLAY_NAME_ALL_POLICIES,
268                                           PolicyFlags::SYSTEM_PARTITION | PolicyFlags::PUBLIC,
269                                           /* enforce_overlayable */ true);
270 
271   ASSERT_TRUE(resources) << resources.GetErrorMessage();
272   auto& res = *resources;
273   ASSERT_EQ(res.GetTargetToOverlayMap().size(), 3U);
274   ASSERT_RESULT(MappingExists(res, R::target::string::policy_public,
275                               R::overlay::string::policy_public, true /* rewrite */));
276   ASSERT_RESULT(MappingExists(res, R::target::string::policy_system,
277                               R::overlay::string::policy_system, true /* rewrite */));
278   ASSERT_RESULT(MappingExists(res, R::target::string::policy_system_vendor,
279                               R::overlay::string::policy_system_vendor, true /* rewrite */));
280 }
281 
282 // Resources that are not declared as overlayable and resources that a protected by policies the
283 // overlay does not fulfilled can map to overlay resources when overlayable enforcement is turned
284 // off.
TEST(ResourceMappingTests,ResourcesFromApkAssetsPolicySystemPublicInvalidIgnoreOverlayable)285 TEST(ResourceMappingTests, ResourcesFromApkAssetsPolicySystemPublicInvalidIgnoreOverlayable) {
286   auto resources = TestGetResourceMapping("target/target.apk", "overlay/overlay.apk",
287                                           TestConstants::OVERLAY_NAME_ALL_POLICIES,
288                                           PolicyFlags::SYSTEM_PARTITION | PolicyFlags::PUBLIC,
289                                           /* enforce_overlayable */ false);
290 
291   ASSERT_TRUE(resources) << resources.GetErrorMessage();
292   auto& res = *resources;
293   ASSERT_EQ(res.GetTargetToOverlayMap().size(), 11U);
294   ASSERT_RESULT(MappingExists(res, R::target::string::not_overlayable,
295                               R::overlay::string::not_overlayable, true /* rewrite */));
296   ASSERT_RESULT(
297       MappingExists(res, R::target::string::other, R::overlay::string::other, true /* rewrite */));
298   ASSERT_RESULT(MappingExists(res, R::target::string::policy_actor,
299                               R::overlay::string::policy_actor, true /* rewrite */));
300   ASSERT_RESULT(MappingExists(res, R::target::string::policy_odm, R::overlay::string::policy_odm,
301                               true /* rewrite */));
302   ASSERT_RESULT(MappingExists(res, R::target::string::policy_oem, R::overlay::string::policy_oem,
303                               true /* rewrite */));
304   ASSERT_RESULT(MappingExists(res, R::target::string::policy_product,
305                               R::overlay::string::policy_product, true /* rewrite */));
306   ASSERT_RESULT(MappingExists(res, R::target::string::policy_public,
307                               R::overlay::string::policy_public, true /* rewrite */));
308   ASSERT_RESULT(MappingExists(res, R::target::string::policy_config_signature,
309                               R::overlay::string::policy_config_signature, true /* rewrite */));
310   ASSERT_RESULT(MappingExists(res, R::target::string::policy_signature,
311                               R::overlay::string::policy_signature, true /* rewrite */));
312   ASSERT_RESULT(MappingExists(res, R::target::string::policy_system,
313                               R::overlay::string::policy_system, true /* rewrite */));
314   ASSERT_RESULT(MappingExists(res, R::target::string::policy_system_vendor,
315                               R::overlay::string::policy_system_vendor, true /* rewrite */));
316 }
317 
318 // Overlays that do not target an <overlayable> tag can overlay any resource if overlayable
319 // enforcement is disabled.
TEST(ResourceMappingTests,ResourcesFromApkAssetsNoDefinedOverlayableAndNoTargetName)320 TEST(ResourceMappingTests, ResourcesFromApkAssetsNoDefinedOverlayableAndNoTargetName) {
321   auto resources = TestGetResourceMapping("target/target.apk", "overlay/overlay-legacy.apk", "",
322                                           PolicyFlags::PUBLIC,
323                                           /* enforce_overlayable */ false);
324 
325   ASSERT_TRUE(resources) << resources.GetErrorMessage();
326   auto& res = *resources;
327   ASSERT_EQ(res.GetTargetToOverlayMap().size(), 4U);
328   ASSERT_RESULT(
329       MappingExists(res, R::target::integer::int1, R::overlay::integer::int1, false /* rewrite */));
330   ASSERT_RESULT(
331       MappingExists(res, R::target::string::str1, R::overlay::string::str1, false /* rewrite */));
332   ASSERT_RESULT(
333       MappingExists(res, R::target::string::str3, R::overlay::string::str3, false /* rewrite */));
334   ASSERT_RESULT(
335       MappingExists(res, R::target::string::str4, R::overlay::string::str4, false /* rewrite */));
336 }
337 
338 // Overlays that are neither pre-installed nor signed with the same signature as the target cannot
339 // overlay packages that have not defined overlayable resources.
TEST(ResourceMappingTests,ResourcesFromApkAssetsDefaultPoliciesPublicFail)340 TEST(ResourceMappingTests, ResourcesFromApkAssetsDefaultPoliciesPublicFail) {
341   auto resources = TestGetResourceMapping("target/target-no-overlayable.apk", "overlay/overlay.apk",
342                                           "NoTargetName", PolicyFlags::PUBLIC,
343                                           /* enforce_overlayable */ true);
344 
345   ASSERT_TRUE(resources) << resources.GetErrorMessage();
346   ASSERT_EQ(resources->GetTargetToOverlayMap().size(), 0U);
347 }
348 
349 // Overlays that are pre-installed or are signed with the same signature as the target  or are
350 // signed with the same signature as the reference package can overlay packages that have not
351 // defined overlayable resources.
TEST(ResourceMappingTests,ResourcesFromApkAssetsDefaultPolicies)352 TEST(ResourceMappingTests, ResourcesFromApkAssetsDefaultPolicies) {
353   auto CheckEntries = [&](const PolicyBitmask& fulfilled_policies) {
354     auto resources =
355         TestGetResourceMapping("target/target-no-overlayable.apk", "overlay/overlay.apk",
356                                TestConstants::OVERLAY_NAME_ALL_POLICIES, fulfilled_policies,
357                                /* enforce_overlayable */ true);
358 
359     ASSERT_TRUE(resources) << resources.GetErrorMessage();
360     auto& res = *resources;
361     ASSERT_EQ(resources->GetTargetToOverlayMap().size(), 11U);
362     ASSERT_RESULT(MappingExists(res, R::target::string::not_overlayable,
363                                 R::overlay::string::not_overlayable, true /* rewrite */));
364     ASSERT_RESULT(MappingExists(res, R::target::string::other, R::overlay::string::other,
365                                 true /* rewrite */));
366     ASSERT_RESULT(MappingExists(res, R::target::string::policy_actor,
367                                 R::overlay::string::policy_actor, true /* rewrite */));
368     ASSERT_RESULT(MappingExists(res, R::target::string::policy_odm, R::overlay::string::policy_odm,
369                                 true /* rewrite */));
370     ASSERT_RESULT(MappingExists(res, R::target::string::policy_oem, R::overlay::string::policy_oem,
371                                 true /* rewrite */));
372     ASSERT_RESULT(MappingExists(res, R::target::string::policy_product,
373                                 R::overlay::string::policy_product, true /* rewrite */));
374     ASSERT_RESULT(MappingExists(res, R::target::string::policy_public,
375                                 R::overlay::string::policy_public, true /* rewrite */));
376     ASSERT_RESULT(MappingExists(res, R::target::string::policy_config_signature,
377                                 R::overlay::string::policy_config_signature, true /* rewrite */));
378     ASSERT_RESULT(MappingExists(res, R::target::string::policy_signature,
379                                 R::overlay::string::policy_signature, true /* rewrite */));
380     ASSERT_RESULT(MappingExists(res, R::target::string::policy_system,
381                                 R::overlay::string::policy_system, true /* rewrite */));
382     ASSERT_RESULT(MappingExists(res, R::target::string::policy_system_vendor,
383                                 R::overlay::string::policy_system_vendor, true /* rewrite */));
384   };
385 
386   CheckEntries(PolicyFlags::SIGNATURE);
387   CheckEntries(PolicyFlags::CONFIG_SIGNATURE);
388   CheckEntries(PolicyFlags::PRODUCT_PARTITION);
389   CheckEntries(PolicyFlags::SYSTEM_PARTITION);
390   CheckEntries(PolicyFlags::VENDOR_PARTITION);
391   CheckEntries(PolicyFlags::ODM_PARTITION);
392   CheckEntries(PolicyFlags::OEM_PARTITION);
393 }
394 
395 }  // namespace android::idmap2
396