1 package org.robolectric.res.android; 2 3 import static org.robolectric.res.android.ApkAssetsCookie.K_INVALID_COOKIE; 4 import static org.robolectric.res.android.ApkAssetsCookie.kInvalidCookie; 5 import static org.robolectric.res.android.Errors.NO_ERROR; 6 import static org.robolectric.res.android.ResourceUtils.ExtractResourceName; 7 import static org.robolectric.res.android.ResourceUtils.fix_package_id; 8 import static org.robolectric.res.android.ResourceUtils.get_entry_id; 9 import static org.robolectric.res.android.ResourceUtils.get_package_id; 10 import static org.robolectric.res.android.ResourceUtils.get_type_id; 11 import static org.robolectric.res.android.ResourceUtils.is_internal_resid; 12 import static org.robolectric.res.android.ResourceUtils.is_valid_resid; 13 import static org.robolectric.res.android.Util.ATRACE_CALL; 14 import static org.robolectric.res.android.Util.dtohl; 15 import static org.robolectric.res.android.Util.dtohs; 16 import static org.robolectric.res.android.Util.isTruthy; 17 18 import java.util.ArrayList; 19 import java.util.Arrays; 20 import java.util.Collections; 21 import java.util.HashMap; 22 import java.util.HashSet; 23 import java.util.List; 24 import java.util.Map; 25 import java.util.Objects; 26 import java.util.Set; 27 import org.robolectric.res.Fs; 28 import org.robolectric.res.FsFile; 29 import org.robolectric.res.android.CppApkAssets.ForEachFileCallback; 30 import org.robolectric.res.android.AssetDir.FileInfo; 31 import org.robolectric.res.android.CppAssetManager.FileType; 32 import org.robolectric.res.android.CppAssetManager2.ResolvedBag.Entry; 33 import org.robolectric.res.android.Idmap.LoadedIdmap; 34 import org.robolectric.res.android.LoadedArsc.DynamicPackageEntry; 35 import org.robolectric.res.android.LoadedArsc.LoadedPackage; 36 import org.robolectric.res.android.LoadedArsc.TypeSpec; 37 import org.robolectric.res.android.ResourceTypes.ResTable_entry; 38 import org.robolectric.res.android.ResourceTypes.ResTable_map; 39 import org.robolectric.res.android.ResourceTypes.ResTable_map_entry; 40 import org.robolectric.res.android.ResourceTypes.ResTable_type; 41 import org.robolectric.res.android.ResourceTypes.Res_value; 42 43 // transliterated from https://android.googlesource.com/platform/frameworks/base/+/android-9.0.0_r12/libs/androidfw/include/androidfw/AssetManager2.h 44 // and https://android.googlesource.com/platform/frameworks/base/+/android-9.0.0_r12/libs/androidfw/AssetManager2.cpp 45 public class CppAssetManager2 { 46 // #define ATRACE_TAG ATRACE_TAG_RESOURCES 47 // 48 // #include "androidfw/AssetManager2.h" 49 50 //#include <array> 51 //#include <limits> 52 //#include <set> 53 //#include <unordered_map> 54 // 55 //#include "androidfw/ApkAssets.h" 56 //#include "androidfw/Asset.h" 57 //#include "androidfw/AssetManager.h" 58 //#include "androidfw/ResourceTypes.h" 59 //#include "androidfw/Util.h" 60 // 61 //namespace android { 62 // 63 //class Theme; 64 // 65 //using ApkAssetsCookie = int32_t; 66 // 67 //enum : ApkAssetsCookie { 68 //}; 69 70 // Holds a bag that has been merged with its parent, if one exists. 71 public static class ResolvedBag { 72 // A single key-value entry in a bag. 73 public static class Entry { 74 // The key, as described in ResTable_map.name. 75 public int key; 76 77 public Res_value value = new Res_value(); 78 79 // BEGIN-INTERNAL 80 // The resource ID of the origin style associated with the given entry 81 public int style; 82 // END-INTERNAL 83 84 // Which ApkAssets this entry came from. 85 public ApkAssetsCookie cookie; 86 87 ResStringPool key_pool; 88 ResStringPool type_pool; 89 90 public ResolvedBag.Entry copy() { 91 Entry entry = new Entry(); 92 entry.key = key; 93 entry.value = value.copy(); 94 entry.cookie = cookie == null ? null : ApkAssetsCookie.forInt(cookie.intValue()); 95 entry.key_pool = key_pool; 96 entry.type_pool = type_pool; 97 return entry; 98 } 99 100 @Override 101 public String toString() { 102 return "Entry{" + 103 "key=" + key + 104 ", value=" + value + 105 '}'; 106 } 107 }; 108 109 // Denotes the configuration axis that this bag varies with. 110 // If a configuration changes with respect to one of these axis, 111 // the bag should be reloaded. 112 public int type_spec_flags; 113 114 // The number of entries in this bag. Access them by indexing into `entries`. 115 public int entry_count; 116 117 // The array of entries for this bag. An empty array is a neat trick to force alignment 118 // of the Entry structs that follow this structure and avoids a bunch of casts. 119 public Entry[] entries; 120 }; 121 122 // AssetManager2 is the main entry point for accessing assets and resources. 123 // AssetManager2 provides caching of resources retrieved via the underlying ApkAssets. 124 // class AssetManager2 : public .AAssetManager { 125 // public: 126 public static class ResourceName { 127 public String package_ = null; 128 // int package_len = 0; 129 130 public String type = null; 131 // public String type16 = null; 132 // int type_len = 0; 133 134 public String entry = null; 135 // public String entry16 = null; 136 // int entry_len = 0; 137 }; 138 139 public CppAssetManager2() { 140 } 141 142 143 public final List<CppApkAssets> GetApkAssets() { return apk_assets_; } 144 145 final ResTable_config GetConfiguration() { return configuration_; } 146 147 // private: 148 // DISALLOW_COPY_AND_ASSIGN(AssetManager2); 149 150 // The ordered list of ApkAssets to search. These are not owned by the AssetManager, and must 151 // have a longer lifetime. 152 private List<CppApkAssets> apk_assets_; 153 154 // A collection of configurations and their associated ResTable_type that match the current 155 // AssetManager configuration. 156 static class FilteredConfigGroup { 157 final List<ResTable_config> configurations = new ArrayList<>(); 158 final List<ResTable_type> types = new ArrayList<>(); 159 } 160 161 // Represents an single package. 162 static class ConfiguredPackage { 163 // A pointer to the immutable, loaded package info. 164 LoadedPackage loaded_package_; 165 166 // A mutable AssetManager-specific list of configurations that match the AssetManager's 167 // current configuration. This is used as an optimization to avoid checking every single 168 // candidate configuration when looking up resources. 169 ByteBucketArray<FilteredConfigGroup> filtered_configs_; 170 171 public ConfiguredPackage(LoadedPackage package_) { 172 this.loaded_package_ = package_; 173 } 174 } 175 176 // Represents a logical package, which can be made up of many individual packages. Each package 177 // in a PackageGroup shares the same package name and package ID. 178 static class PackageGroup { 179 // The set of packages that make-up this group. 180 final List<ConfiguredPackage> packages_ = new ArrayList<>(); 181 182 // The cookies associated with each package in the group. They share the same order as 183 // packages_. 184 final List<ApkAssetsCookie> cookies_ = new ArrayList<>(); 185 186 // A library reference table that contains build-package ID to runtime-package ID mappings. 187 DynamicRefTable dynamic_ref_table; 188 } 189 190 // DynamicRefTables for shared library package resolution. 191 // These are ordered according to apk_assets_. The mappings may change depending on what is 192 // in apk_assets_, therefore they must be stored in the AssetManager and not in the 193 // immutable ApkAssets class. 194 final private List<PackageGroup> package_groups_ = new ArrayList<>(); 195 196 // An array mapping package ID to index into package_groups. This keeps the lookup fast 197 // without taking too much memory. 198 // private std.array<byte, std.numeric_limits<byte>.max() + 1> package_ids_; 199 final private byte[] package_ids_ = new byte[256]; 200 201 // The current configuration set for this AssetManager. When this changes, cached resources 202 // may need to be purged. 203 private ResTable_config configuration_ = new ResTable_config(); 204 205 // Cached set of bags. These are cached because they can inherit keys from parent bags, 206 // which involves some calculation. 207 // private std.unordered_map<int, util.unique_cptr<ResolvedBag>> cached_bags_; 208 final private Map<Integer, ResolvedBag> cached_bags_ = new HashMap<>(); 209 // }; 210 211 //final ResolvedBag.Entry* begin(final ResolvedBag* bag) { return bag.entries; } 212 // 213 //final ResolvedBag.Entry* end(final ResolvedBag* bag) { 214 // return bag.entries + bag.entry_count; 215 //} 216 // 217 //} // namespace android 218 // 219 //#endif /* ANDROIDFW_ASSETMANAGER2_H_ */ 220 221 222 // 223 // #include <set> 224 // 225 // #include "android-base/logging.h" 226 // #include "android-base/stringprintf.h" 227 // #include "utils/ByteOrder.h" 228 // #include "utils/Trace.h" 229 // 230 // #ifdef _WIN32 231 // #ifdef ERROR 232 // #undef ERROR 233 // #endif 234 // #endif 235 // 236 // #include "androidfw/ResourceUtils.h" 237 // 238 // namespace android { 239 static class FindEntryResult { 240 // A pointer to the resource table entry for this resource. 241 // If the size of the entry is > sizeof(ResTable_entry), it can be cast to 242 // a ResTable_map_entry and processed as a bag/map. 243 ResTable_entry entry; 244 245 // The configuration for which the resulting entry was defined. This is already swapped to host 246 // endianness. 247 ResTable_config config; 248 249 // The bitmask of configuration axis with which the resource value varies. 250 int type_flags; 251 252 // The dynamic package ID map for the package from which this resource came from. 253 DynamicRefTable dynamic_ref_table; 254 255 // The string pool reference to the type's name. This uses a different string pool than 256 // the global string pool, but this is hidden from the caller. 257 StringPoolRef type_string_ref; 258 259 // The string pool reference to the entry's name. This uses a different string pool than 260 // the global string pool, but this is hidden from the caller. 261 StringPoolRef entry_string_ref; 262 } 263 264 // AssetManager2() { memset(&configuration_, 0, sizeof(configuration_)); } 265 266 // Sets/resets the underlying ApkAssets for this AssetManager. The ApkAssets 267 // are not owned by the AssetManager, and must have a longer lifetime. 268 // 269 // Only pass invalidate_caches=false when it is known that the structure 270 // change in ApkAssets is due to a safe addition of resources with completely 271 // new resource IDs. 272 // boolean SetApkAssets(final List<ApkAssets> apk_assets, boolean invalidate_caches = true); 273 public boolean SetApkAssets(final List<CppApkAssets> apk_assets, boolean invalidate_caches) { 274 apk_assets_ = apk_assets; 275 BuildDynamicRefTable(); 276 RebuildFilterList(); 277 if (invalidate_caches) { 278 // InvalidateCaches(static_cast<int>(-1)); 279 InvalidateCaches(-1); 280 } 281 return true; 282 } 283 284 // Assigns package IDs to all shared library ApkAssets. 285 // Should be called whenever the ApkAssets are changed. 286 // void BuildDynamicRefTable(); 287 void BuildDynamicRefTable() { 288 package_groups_.clear(); 289 // package_ids_.fill(0xff); 290 for (int i = 0; i < package_ids_.length; i++) { 291 package_ids_[i] = (byte) 0xff; 292 } 293 294 // 0x01 is reserved for the android package. 295 int next_package_id = 0x02; 296 final int apk_assets_count = apk_assets_.size(); 297 for (int i = 0; i < apk_assets_count; i++) { 298 final LoadedArsc loaded_arsc = apk_assets_.get(i).GetLoadedArsc(); 299 // for (final std.unique_ptr<final LoadedPackage>& package_ : 300 for (final LoadedPackage package_ : 301 loaded_arsc.GetPackages()) { 302 // Get the package ID or assign one if a shared library. 303 int package_id; 304 if (package_.IsDynamic()) { 305 package_id = next_package_id++; 306 } else { 307 package_id = package_.GetPackageId(); 308 } 309 310 // Add the mapping for package ID to index if not present. 311 byte idx = package_ids_[package_id]; 312 if (idx == (byte) 0xff) { 313 // package_ids_[package_id] = idx = static_cast<byte>(package_groups_.size()); 314 package_ids_[package_id] = idx = (byte) package_groups_.size(); 315 // DynamicRefTable& ref_table = package_groups_.back().dynamic_ref_table; 316 // ref_table.mAssignedPackageId = package_id; 317 // ref_table.mAppAsLib = package->IsDynamic() && package->GetPackageId() == 0x7f; 318 DynamicRefTable ref_table = new DynamicRefTable((byte) package_id, 319 package_.IsDynamic() && package_.GetPackageId() == 0x7f); 320 PackageGroup newPackageGroup = new PackageGroup(); 321 newPackageGroup.dynamic_ref_table = ref_table; 322 323 package_groups_.add(newPackageGroup); 324 } 325 PackageGroup package_group = package_groups_.get(idx); 326 327 // Add the package and to the set of packages with the same ID. 328 // package_group->packages_.push_back(ConfiguredPackage{package.get(), {}}); 329 // package_group.cookies_.push_back(static_cast<ApkAssetsCookie>(i)); 330 package_group.packages_.add(new ConfiguredPackage(package_)); 331 package_group.cookies_.add(ApkAssetsCookie.forInt(i)); 332 333 // Add the package name . build time ID mappings. 334 for (final DynamicPackageEntry entry : package_.GetDynamicPackageMap()) { 335 // String package_name(entry.package_name.c_str(), entry.package_name.size()); 336 package_group.dynamic_ref_table.mEntries.put( 337 entry.package_name, (byte) entry.package_id); 338 } 339 } 340 } 341 342 // Now assign the runtime IDs so that we have a build-time to runtime ID map. 343 for (PackageGroup iter : package_groups_) { 344 String package_name = iter.packages_.get(0).loaded_package_.GetPackageName(); 345 for (PackageGroup iter2 : package_groups_) { 346 iter2.dynamic_ref_table.addMapping(package_name, 347 iter.dynamic_ref_table.mAssignedPackageId); 348 } 349 } 350 } 351 352 // void AssetManager2::DumpToLog() const { 353 // base::ScopedLogSeverity _log(base::INFO); 354 // 355 // LOG(INFO) << base::StringPrintf("AssetManager2(this=%p)", this); 356 // 357 // std::string list; 358 // for (const auto& apk_assets : apk_assets_) { 359 // base::StringAppendF(&list, "%s,", apk_assets->GetPath().c_str()); 360 // } 361 // LOG(INFO) << "ApkAssets: " << list; 362 // 363 // list = ""; 364 // for (size_t i = 0; i < package_ids_.size(); i++) { 365 // if (package_ids_[i] != 0xff) { 366 // base::StringAppendF(&list, "%02x -> %d, ", (int)i, package_ids_[i]); 367 // } 368 // } 369 // LOG(INFO) << "Package ID map: " << list; 370 // 371 // for (const auto& package_group: package_groups_) { 372 // list = ""; 373 // for (const auto& package : package_group.packages_) { 374 // const LoadedPackage* loaded_package = package.loaded_package_; 375 // base::StringAppendF(&list, "%s(%02x%s), ", loaded_package->GetPackageName().c_str(), 376 // loaded_package->GetPackageId(), 377 // (loaded_package->IsDynamic() ? " dynamic" : "")); 378 // } 379 // LOG(INFO) << base::StringPrintf("PG (%02x): ", 380 // package_group.dynamic_ref_table.mAssignedPackageId) 381 // << list; 382 // } 383 // } 384 385 // Returns the string pool for the given asset cookie. 386 // Use the string pool returned here with a valid Res_value object of type Res_value.TYPE_STRING. 387 // final ResStringPool GetStringPoolForCookie(ApkAssetsCookie cookie) const; 388 final ResStringPool GetStringPoolForCookie(ApkAssetsCookie cookie) { 389 if (cookie.intValue() < 0 || cookie.intValue() >= apk_assets_.size()) { 390 return null; 391 } 392 return apk_assets_.get(cookie.intValue()).GetLoadedArsc().GetStringPool(); 393 } 394 395 // Returns the DynamicRefTable for the given package ID. 396 // This may be nullptr if the APK represented by `cookie` has no resource table. 397 // final DynamicRefTable GetDynamicRefTableForPackage(int package_id) const; 398 final DynamicRefTable GetDynamicRefTableForPackage(int package_id) { 399 if (package_id >= package_ids_.length) { 400 return null; 401 } 402 403 final int idx = package_ids_[package_id]; 404 if (idx == 0xff) { 405 return null; 406 } 407 return package_groups_.get(idx).dynamic_ref_table; 408 } 409 410 // Returns the DynamicRefTable for the ApkAssets represented by the cookie. 411 // final DynamicRefTable GetDynamicRefTableForCookie(ApkAssetsCookie cookie) const; 412 public final DynamicRefTable GetDynamicRefTableForCookie(ApkAssetsCookie cookie) { 413 for (final PackageGroup package_group : package_groups_) { 414 for (final ApkAssetsCookie package_cookie : package_group.cookies_) { 415 if (package_cookie == cookie) { 416 return package_group.dynamic_ref_table; 417 } 418 } 419 } 420 return null; 421 } 422 423 // Sets/resets the configuration for this AssetManager. This will cause all 424 // caches that are related to the configuration change to be invalidated. 425 // void SetConfiguration(final ResTable_config& configuration); 426 public void SetConfiguration(final ResTable_config configuration) { 427 final int diff = configuration_.diff(configuration); 428 configuration_ = configuration; 429 430 if (isTruthy(diff)) { 431 RebuildFilterList(); 432 // InvalidateCaches(static_cast<int>(diff)); 433 InvalidateCaches(diff); 434 } 435 } 436 437 // Returns all configurations for which there are resources defined. This includes resource 438 // configurations in all the ApkAssets set for this AssetManager. 439 // If `exclude_system` is set to true, resource configurations from system APKs 440 // ('android' package, other libraries) will be excluded from the list. 441 // If `exclude_mipmap` is set to true, resource configurations defined for resource type 'mipmap' 442 // will be excluded from the list. 443 // Set<ResTable_config> GetResourceConfigurations(boolean exclude_system = false, 444 // boolean exclude_mipmap = false); 445 public Set<ResTable_config> GetResourceConfigurations(boolean exclude_system, 446 boolean exclude_mipmap) { 447 // ATRACE_NAME("AssetManager::GetResourceConfigurations"); 448 Set<ResTable_config> configurations = new HashSet<>(); 449 for (final PackageGroup package_group : package_groups_) { 450 for (final ConfiguredPackage package_ : package_group.packages_) { 451 if (exclude_system && package_.loaded_package_.IsSystem()) { 452 continue; 453 } 454 package_.loaded_package_.CollectConfigurations(exclude_mipmap, configurations); 455 } 456 } 457 return configurations; 458 } 459 460 // Returns all the locales for which there are resources defined. This includes resource 461 // locales in all the ApkAssets set for this AssetManager. 462 // If `exclude_system` is set to true, resource locales from system APKs 463 // ('android' package, other libraries) will be excluded from the list. 464 // If `merge_equivalent_languages` is set to true, resource locales will be canonicalized 465 // and de-duped in the resulting list. 466 // Set<String> GetResourceLocales(boolean exclude_system = false, 467 // boolean merge_equivalent_languages = false); 468 public Set<String> GetResourceLocales(boolean exclude_system, 469 boolean merge_equivalent_languages) { 470 ATRACE_CALL(); 471 Set<String> locales = new HashSet<>(); 472 for (final PackageGroup package_group : package_groups_) { 473 for (final ConfiguredPackage package_ : package_group.packages_) { 474 if (exclude_system && package_.loaded_package_.IsSystem()) { 475 continue; 476 } 477 package_.loaded_package_.CollectLocales(merge_equivalent_languages, locales); 478 } 479 } 480 return locales; 481 } 482 483 // Searches the set of APKs loaded by this AssetManager and opens the first one found located 484 // in the assets/ directory. 485 // `mode` controls how the file is opened. 486 // 487 // NOTE: The loaded APKs are searched in reverse order. 488 // Asset Open(final String filename, Asset.AccessMode mode); 489 public Asset Open(final String filename, Asset.AccessMode mode) { 490 final String new_path = "assets/" + filename; 491 return OpenNonAsset(new_path, mode); 492 } 493 494 // Opens a file within the assets/ directory of the APK specified by `cookie`. 495 // `mode` controls how the file is opened. 496 // Asset Open(final String filename, ApkAssetsCookie cookie, 497 // Asset.AccessMode mode); 498 Asset Open(final String filename, ApkAssetsCookie cookie, 499 Asset.AccessMode mode) { 500 final String new_path = "assets/" + filename; 501 return OpenNonAsset(new_path, cookie, mode); 502 } 503 504 // Opens the directory specified by `dirname`. The result is an AssetDir that is the combination 505 // of all directories matching `dirname` under the assets/ directory of every ApkAssets loaded. 506 // The entries are sorted by their ASCII name. 507 // AssetDir OpenDir(final String dirname); 508 public AssetDir OpenDir(final String dirname) { 509 ATRACE_CALL(); 510 511 String full_path = "assets/" + dirname; 512 // std.unique_ptr<SortedVector<AssetDir.FileInfo>> files = 513 // util.make_unique<SortedVector<AssetDir.FileInfo>>(); 514 SortedVector<FileInfo> files = new SortedVector<>(); 515 516 // Start from the back. 517 for (CppApkAssets apk_assets : apk_assets_) { 518 // auto func = [&](final String& name, FileType type) { 519 ForEachFileCallback func = (final String name, FileType type) -> { 520 AssetDir.FileInfo info = new FileInfo(); 521 info.setFileName(new String8(name)); 522 info.setFileType(type); 523 info.setSourceName(new String8(apk_assets.GetPath())); 524 files.add(info); 525 }; 526 527 if (!apk_assets.ForEachFile(full_path, func)) { 528 return new AssetDir(); 529 } 530 } 531 532 // std.unique_ptr<AssetDir> asset_dir = util.make_unique<AssetDir>(); 533 AssetDir asset_dir = new AssetDir(); 534 asset_dir.setFileList(files); 535 return asset_dir; 536 } 537 538 // Searches the set of APKs loaded by this AssetManager and opens the first one found. 539 // `mode` controls how the file is opened. 540 // `out_cookie` is populated with the cookie of the APK this file was found in. 541 // 542 // NOTE: The loaded APKs are searched in reverse order. 543 // Asset OpenNonAsset(final String filename, Asset.AccessMode mode, 544 // ApkAssetsCookie* out_cookie = null); 545 // Search in reverse because that's how we used to do it and we need to preserve behaviour. 546 // This is unfortunate, because ClassLoaders delegate to the parent first, so the order 547 // is inconsistent for split APKs. 548 public Asset OpenNonAsset(final String filename, 549 Asset.AccessMode mode, 550 Ref<ApkAssetsCookie> out_cookie) { 551 ATRACE_CALL(); 552 for (int i = apk_assets_.size() - 1; i >= 0; i--) { 553 Asset asset = apk_assets_.get(i).Open(filename, mode); 554 if (isTruthy(asset)) { 555 if (out_cookie != null) { 556 out_cookie.set(ApkAssetsCookie.forInt(i)); 557 } 558 return asset; 559 } 560 } 561 562 if (out_cookie != null) { 563 out_cookie.set(K_INVALID_COOKIE); 564 } 565 return null; 566 } 567 568 public Asset OpenNonAsset(final String filename, Asset.AccessMode mode) { 569 return OpenNonAsset(filename, mode, null); 570 } 571 572 // Opens a file in the APK specified by `cookie`. `mode` controls how the file is opened. 573 // This is typically used to open a specific AndroidManifest.xml, or a binary XML file 574 // referenced by a resource lookup with GetResource(). 575 // Asset OpenNonAsset(final String filename, ApkAssetsCookie cookie, 576 // Asset.AccessMode mode); 577 public Asset OpenNonAsset(final String filename, 578 ApkAssetsCookie cookie, Asset.AccessMode mode) { 579 ATRACE_CALL(); 580 if (cookie.intValue() < 0 || cookie.intValue() >= apk_assets_.size()) { 581 return null; 582 } 583 return apk_assets_.get(cookie.intValue()).Open(filename, mode); 584 } 585 586 // template <typename Func> 587 public interface PackageFunc { 588 void apply(String package_name, byte package_id); 589 } 590 591 public void ForEachPackage(PackageFunc func) { 592 for (PackageGroup package_group : package_groups_) { 593 func.apply(package_group.packages_.get(0).loaded_package_.GetPackageName(), 594 package_group.dynamic_ref_table.mAssignedPackageId); 595 } 596 } 597 598 // Finds the best entry for `resid` from the set of ApkAssets. The entry can be a simple 599 // Res_value, or a complex map/bag type. If successful, it is available in `out_entry`. 600 // Returns kInvalidCookie on failure. Otherwise, the return value is the cookie associated with 601 // the ApkAssets in which the entry was found. 602 // 603 // `density_override` overrides the density of the current configuration when doing a search. 604 // 605 // When `stop_at_first_match` is true, the first match found is selected and the search 606 // terminates. This is useful for methods that just look up the name of a resource and don't 607 // care about the value. In this case, the value of `FindEntryResult::type_flags` is incomplete 608 // and should not be used. 609 // 610 // NOTE: FindEntry takes care of ensuring that structs within FindEntryResult have been properly 611 // bounds-checked. Callers of FindEntry are free to trust the data if this method succeeds. 612 // ApkAssetsCookie FindEntry(int resid, short density_override, boolean stop_at_first_match, 613 // LoadedArscEntry* out_entry, ResTable_config out_selected_config, 614 // int* out_flags); 615 private ApkAssetsCookie FindEntry(int resid, short density_override, 616 boolean stop_at_first_match, 617 final Ref<FindEntryResult> out_entry) { 618 ATRACE_CALL(); 619 620 // Might use this if density_override != 0. 621 ResTable_config density_override_config; 622 623 // Select our configuration or generate a density override configuration. 624 ResTable_config desired_config = configuration_; 625 if (density_override != 0 && density_override != configuration_.density) { 626 density_override_config = configuration_; 627 density_override_config.density = density_override; 628 desired_config = density_override_config; 629 } 630 631 if (!is_valid_resid(resid)) { 632 System.err.println(String.format("Invalid ID 0x%08x.", resid)); 633 return K_INVALID_COOKIE; 634 } 635 636 final int package_id = get_package_id(resid); 637 final int type_idx = (byte) (get_type_id(resid) - 1); 638 final int entry_idx = get_entry_id(resid); 639 640 final byte package_idx = package_ids_[package_id]; 641 if (package_idx == (byte) 0xff) { 642 System.err.println(String.format("No package ID %02x found for ID 0x%08x.", package_id, resid)); 643 return K_INVALID_COOKIE; 644 } 645 646 final PackageGroup package_group = package_groups_.get(package_idx); 647 final int package_count = package_group.packages_.size(); 648 649 ApkAssetsCookie best_cookie = K_INVALID_COOKIE; 650 LoadedPackage best_package = null; 651 ResTable_type best_type = null; 652 ResTable_config best_config = null; 653 ResTable_config best_config_copy; 654 int best_offset = 0; 655 int type_flags = 0; 656 657 // If desired_config is the same as the set configuration, then we can use our filtered list 658 // and we don't need to match the configurations, since they already matched. 659 boolean use_fast_path = desired_config == configuration_; 660 661 for (int pi = 0; pi < package_count; pi++) { 662 ConfiguredPackage loaded_package_impl = package_group.packages_.get(pi); 663 LoadedPackage loaded_package = loaded_package_impl.loaded_package_; 664 ApkAssetsCookie cookie = package_group.cookies_.get(pi); 665 666 // If the type IDs are offset in this package, we need to take that into account when searching 667 // for a type. 668 TypeSpec type_spec = loaded_package.GetTypeSpecByTypeIndex(type_idx); 669 if (Util.UNLIKELY(type_spec == null)) { 670 continue; 671 } 672 673 int local_entry_idx = entry_idx; 674 675 // If there is an IDMAP supplied with this package, translate the entry ID. 676 if (type_spec.idmap_entries != null) { 677 if (!LoadedIdmap 678 .Lookup(type_spec.idmap_entries, local_entry_idx, new Ref<>(local_entry_idx))) { 679 // There is no mapping, so the resource is not meant to be in this overlay package. 680 continue; 681 } 682 } 683 684 type_flags |= type_spec.GetFlagsForEntryIndex(local_entry_idx); 685 686 // If the package is an overlay, then even configurations that are the same MUST be chosen. 687 boolean package_is_overlay = loaded_package.IsOverlay(); 688 689 FilteredConfigGroup filtered_group = loaded_package_impl.filtered_configs_.get(type_idx); 690 if (use_fast_path) { 691 List<ResTable_config> candidate_configs = filtered_group.configurations; 692 int type_count = candidate_configs.size(); 693 for (int i = 0; i < type_count; i++) { 694 ResTable_config this_config = candidate_configs.get(i); 695 696 // We can skip calling ResTable_config.match() because we know that all candidate 697 // configurations that do NOT match have been filtered-out. 698 if ((best_config == null || this_config.isBetterThan(best_config, desired_config)) || 699 (package_is_overlay && this_config.compare(best_config) == 0)) { 700 // The configuration matches and is better than the previous selection. 701 // Find the entry value if it exists for this configuration. 702 ResTable_type type_chunk = filtered_group.types.get(i); 703 int offset = LoadedPackage.GetEntryOffset(type_chunk, local_entry_idx); 704 if (offset == ResTable_type.NO_ENTRY) { 705 continue; 706 } 707 708 best_cookie = cookie; 709 best_package = loaded_package; 710 best_type = type_chunk; 711 best_config = this_config; 712 best_offset = offset; 713 } 714 } 715 } else { 716 // This is the slower path, which doesn't use the filtered list of configurations. 717 // Here we must read the ResTable_config from the mmapped APK, convert it to host endianness 718 // and fill in any new fields that did not exist when the APK was compiled. 719 // Furthermore when selecting configurations we can't just record the pointer to the 720 // ResTable_config, we must copy it. 721 // auto iter_end = type_spec.types + type_spec.type_count; 722 // for (auto iter = type_spec.types; iter != iter_end; ++iter) { 723 for (ResTable_type type : type_spec.types) { 724 ResTable_config this_config = ResTable_config.fromDtoH(type.config); 725 726 if (this_config.match(desired_config)) { 727 if ((best_config == null || this_config.isBetterThan(best_config, desired_config)) || 728 (package_is_overlay && this_config.compare(best_config) == 0)) { 729 // The configuration matches and is better than the previous selection. 730 // Find the entry value if it exists for this configuration. 731 int offset = LoadedPackage.GetEntryOffset(type, local_entry_idx); 732 if (offset == ResTable_type.NO_ENTRY) { 733 continue; 734 } 735 736 best_cookie = cookie; 737 best_package = loaded_package; 738 best_type = type; 739 best_config_copy = this_config; 740 best_config = best_config_copy; 741 best_offset = offset; 742 } 743 } 744 } 745 } 746 } 747 748 if (Util.UNLIKELY(best_cookie.intValue() == kInvalidCookie)) { 749 return K_INVALID_COOKIE; 750 } 751 752 ResTable_entry best_entry = LoadedPackage.GetEntryFromOffset(best_type, best_offset); 753 if (Util.UNLIKELY(best_entry == null)) { 754 return K_INVALID_COOKIE; 755 } 756 757 FindEntryResult out_entry_ = new FindEntryResult(); 758 out_entry_.entry = best_entry; 759 out_entry_.config = best_config; 760 out_entry_.type_flags = type_flags; 761 out_entry_.type_string_ref = new StringPoolRef(best_package.GetTypeStringPool(), best_type.id - 1); 762 out_entry_.entry_string_ref = 763 new StringPoolRef(best_package.GetKeyStringPool(), best_entry.key.index); 764 out_entry_.dynamic_ref_table = package_group.dynamic_ref_table; 765 out_entry.set(out_entry_); 766 return best_cookie; 767 } 768 769 // Populates the `out_name` parameter with resource name information. 770 // Utf8 strings are preferred, and only if they are unavailable are 771 // the Utf16 variants populated. 772 // Returns false if the resource was not found or the name was missing/corrupt. 773 // boolean GetResourceName(int resid, ResourceName* out_name); 774 public boolean GetResourceName(int resid, ResourceName out_name) { 775 final Ref<FindEntryResult> entryRef = new Ref<>(null); 776 ApkAssetsCookie cookie = 777 FindEntry(resid, (short) 0 /* density_override */, true /* stop_at_first_match */, entryRef); 778 if (cookie.intValue() == kInvalidCookie) { 779 return false; 780 } 781 782 final LoadedPackage package_ = 783 apk_assets_.get(cookie.intValue()).GetLoadedArsc().GetPackageById(get_package_id(resid)); 784 if (package_ == null) { 785 return false; 786 } 787 788 out_name.package_ = package_.GetPackageName(); 789 // out_name.package_len = out_name.package_.length(); 790 791 FindEntryResult entry = entryRef.get(); 792 out_name.type = entry.type_string_ref.string(); 793 // out_name.type_len = out_name.type == null ? 0 : out_name.type.length(); 794 // out_name.type16 = null; 795 if (out_name.type == null) { 796 // out_name.type16 = entry.type_string_ref.string(); 797 // out_name.type_len = out_name.type16 == null ? 0 : out_name.type16.length(); 798 // if (out_name.type16 == null) { 799 return false; 800 // } 801 } 802 803 out_name.entry = entry.entry_string_ref.string(); 804 // out_name.entry_len = out_name.entry == null ? 0 : out_name.entry.length(); 805 // out_name.entry16 = null; 806 if (out_name.entry == null) { 807 // out_name.entry16 = entry.entry_string_ref.string(); 808 // out_name.entry_len = out_name.entry16 == null ? 0 : out_name.entry16.length(); 809 // if (out_name.entry16 == null) { 810 return false; 811 // } 812 } 813 return true; 814 } 815 816 // Populates `out_flags` with the bitmask of configuration axis that this resource varies with. 817 // See ResTable_config for the list of configuration axis. 818 // Returns false if the resource was not found. 819 // boolean GetResourceFlags(int resid, int* out_flags); 820 boolean GetResourceFlags(int resid, Ref<Integer> out_flags) { 821 final Ref<FindEntryResult> entry = new Ref<>(null); 822 ApkAssetsCookie cookie = FindEntry(resid, (short) 0 /* density_override */, 823 false /* stop_at_first_match */, entry); 824 if (cookie.intValue() != kInvalidCookie) { 825 out_flags.set(entry.get().type_flags); 826 // this makes no sense, not a boolean: 827 // return cookie; 828 } 829 // this makes no sense, not a boolean: 830 // return kInvalidCookie; 831 832 return cookie.intValue() != kInvalidCookie; 833 } 834 835 836 // Retrieves the best matching resource with ID `resid`. The resource value is filled into 837 // `out_value` and the configuration for the selected value is populated in `out_selected_config`. 838 // `out_flags` holds the same flags as retrieved with GetResourceFlags(). 839 // If `density_override` is non-zero, the configuration to match against is overridden with that 840 // density. 841 // 842 // Returns a valid cookie if the resource was found. If the resource was not found, or if the 843 // resource was a map/bag type, then kInvalidCookie is returned. If `may_be_bag` is false, 844 // this function logs if the resource was a map/bag type before returning kInvalidCookie. 845 // ApkAssetsCookie GetResource(int resid, boolean may_be_bag, short density_override, 846 // Res_value out_value, ResTable_config out_selected_config, 847 // int* out_flags); 848 public ApkAssetsCookie GetResource(int resid, boolean may_be_bag, 849 short density_override, Ref<Res_value> out_value, 850 final Ref<ResTable_config> out_selected_config, 851 final Ref<Integer> out_flags) { 852 final Ref<FindEntryResult> entry = new Ref<>(null); 853 ApkAssetsCookie cookie = 854 FindEntry(resid, density_override, false /* stop_at_first_match */, entry); 855 if (cookie.intValue() == kInvalidCookie) { 856 return K_INVALID_COOKIE; 857 } 858 859 if (isTruthy(dtohl(entry.get().entry.flags) & ResTable_entry.FLAG_COMPLEX)) { 860 if (!may_be_bag) { 861 System.err.println(String.format("Resource %08x is a complex map type.", resid)); 862 return K_INVALID_COOKIE; 863 } 864 865 // Create a reference since we can't represent this complex type as a Res_value. 866 out_value.set(new Res_value((byte) Res_value.TYPE_REFERENCE, resid)); 867 out_selected_config.set(entry.get().config); 868 out_flags.set(entry.get().type_flags); 869 return cookie; 870 } 871 872 // final Res_value device_value = reinterpret_cast<final Res_value>( 873 // reinterpret_cast<final byte*>(entry.entry) + dtohs(entry.entry.size)); 874 // out_value.copyFrom_dtoh(*device_value); 875 Res_value device_value = entry.get().entry.getResValue(); 876 out_value.set(device_value.copy()); 877 878 // Convert the package ID to the runtime assigned package ID. 879 entry.get().dynamic_ref_table.lookupResourceValue(out_value); 880 881 out_selected_config.set(entry.get().config); 882 out_flags.set(entry.get().type_flags); 883 return cookie; 884 } 885 886 // Resolves the resource reference in `in_out_value` if the data type is 887 // Res_value::TYPE_REFERENCE. 888 // `cookie` is the ApkAssetsCookie of the reference in `in_out_value`. 889 // `in_out_value` is the reference to resolve. The result is placed back into this object. 890 // `in_out_flags` is the type spec flags returned from calls to GetResource() or 891 // GetResourceFlags(). Configuration flags of the values pointed to by the reference 892 // are OR'd together with `in_out_flags`. 893 // `in_out_config` is populated with the configuration for which the resolved value was defined. 894 // `out_last_reference` is populated with the last reference ID before resolving to an actual 895 // value. This is only initialized if the passed in `in_out_value` is a reference. 896 // Returns the cookie of the APK the resolved resource was defined in, or kInvalidCookie if 897 // it was not found. 898 // ApkAssetsCookie ResolveReference(ApkAssetsCookie cookie, Res_value in_out_value, 899 // ResTable_config in_out_selected_config, int* in_out_flags, 900 // int* out_last_reference); 901 public ApkAssetsCookie ResolveReference(ApkAssetsCookie cookie, Ref<Res_value> in_out_value, 902 final Ref<ResTable_config> in_out_selected_config, 903 final Ref<Integer> in_out_flags, 904 final Ref<Integer> out_last_reference) { 905 final int kMaxIterations = 20; 906 907 for (int iteration = 0; in_out_value.get().dataType == Res_value.TYPE_REFERENCE && 908 in_out_value.get().data != 0 && iteration < kMaxIterations; 909 iteration++) { 910 out_last_reference.set(in_out_value.get().data); 911 final Ref<Integer> new_flags = new Ref<>(0); 912 cookie = GetResource(in_out_value.get().data, true /*may_be_bag*/, (short) 0 /*density_override*/, 913 in_out_value, in_out_selected_config, new_flags); 914 if (cookie.intValue() == kInvalidCookie) { 915 return K_INVALID_COOKIE; 916 } 917 if (in_out_flags != null) { 918 in_out_flags.set(in_out_flags.get() | new_flags.get()); 919 } 920 if (out_last_reference.get() == in_out_value.get().data) { 921 // This reference can't be resolved, so exit now and let the caller deal with it. 922 return cookie; 923 } 924 } 925 return cookie; 926 } 927 928 // AssetManager2::GetBag(resid) wraps this function to track which resource ids have already 929 // been seen while traversing bag parents. 930 // final ResolvedBag* GetBag(int resid); 931 public final ResolvedBag GetBag(int resid) { 932 List<Integer> found_resids = new ArrayList<>(); 933 return GetBag(resid, found_resids); 934 } 935 936 // Retrieves the best matching bag/map resource with ID `resid`. 937 // This method will resolve all parent references for this bag and merge keys with the child. 938 // To iterate over the keys, use the following idiom: 939 // 940 // final ResolvedBag* bag = asset_manager.GetBag(id); 941 // if (bag != null) { 942 // for (auto iter = begin(bag); iter != end(bag); ++iter) { 943 // ... 944 // } 945 // } 946 ResolvedBag GetBag(int resid, List<Integer> child_resids) { 947 // ATRACE_NAME("AssetManager::GetBag"); 948 949 ResolvedBag cached_iter = cached_bags_.get(resid); 950 if (cached_iter != null) { 951 return cached_iter; 952 } 953 954 final Ref<FindEntryResult> entryRef = new Ref<>(null); 955 ApkAssetsCookie cookie = 956 FindEntry(resid, (short) 0 /* density_override */, false /* stop_at_first_match */, entryRef); 957 if (cookie.intValue() == kInvalidCookie) { 958 return null; 959 } 960 961 FindEntryResult entry = entryRef.get(); 962 963 // Check that the size of the entry header is at least as big as 964 // the desired ResTable_map_entry. Also verify that the entry 965 // was intended to be a map. 966 if (dtohs(entry.entry.size) < ResTable_map_entry.BASE_SIZEOF || 967 (dtohs(entry.entry.flags) & ResourceTypes.ResTable_entry.FLAG_COMPLEX) == 0) { 968 // Not a bag, nothing to do. 969 return null; 970 } 971 972 // final ResTable_map_entry map = reinterpret_cast<final ResTable_map_entry*>(entry.entry); 973 // final ResTable_map map_entry = 974 // reinterpret_cast<final ResTable_map*>(reinterpret_cast<final byte*>(map) + map.size); 975 // final ResTable_map map_entry_end = map_entry + dtohl(map.count); 976 final ResTable_map_entry map = new ResTable_map_entry(entry.entry.myBuf(), entry.entry.myOffset()); 977 int curOffset = map.myOffset() + map.size; 978 ResTable_map map_entry = null; // = new ResTable_map(map.myBuf(), curOffset); 979 final int map_entry_end = 980 curOffset + dtohl(map.count) * ResTable_map.SIZEOF; 981 if (curOffset < map_entry_end) { 982 map_entry = new ResTable_map(map.myBuf(), curOffset); 983 } 984 985 // Keep track of ids that have already been seen to prevent infinite loops caused by circular 986 // dependencies between bags 987 child_resids.add(resid); 988 989 final Ref<Integer> parent_resid = new Ref<>(dtohl(map.parent.ident)); 990 if (parent_resid.get() == 0 || child_resids.contains(parent_resid.get())) { 991 // There is no parent or that a circular dependency exist, meaning there is nothing to 992 // inherit and we can do a simple copy of the entries in the map. 993 final int entry_count = (map_entry_end - curOffset) / ResTable_map.SIZEOF; 994 // util.unique_cptr<ResolvedBag> new_bag{reinterpret_cast<ResolvedBag*>( 995 // malloc(sizeof(ResolvedBag) + (entry_count * sizeof(ResolvedBag.Entry))))}; 996 ResolvedBag new_bag = new ResolvedBag(); 997 ResolvedBag.Entry[] new_entry = new_bag.entries = new Entry[entry_count]; 998 int i = 0; 999 for (; curOffset < map_entry_end; 1000 map_entry = new ResTable_map(map_entry.myBuf(), curOffset)) { 1001 final Ref<Integer> new_key = new Ref<>(dtohl(map_entry.name.ident)); 1002 if (!is_internal_resid(new_key.get())) { 1003 // Attributes, arrays, etc don't have a resource id as the name. They specify 1004 // other data, which would be wrong to change via a lookup. 1005 if (entry.dynamic_ref_table.lookupResourceId(new_key) != NO_ERROR) { 1006 System.err.println(String.format("Failed to resolve key 0x%08x in bag 0x%08x.", new_key.get(), 1007 resid)); 1008 return null; 1009 } 1010 } 1011 Entry new_entry_ = new_entry[i] = new Entry(); 1012 new_entry_.cookie = cookie; 1013 new_entry_.key = new_key.get(); 1014 new_entry_.key_pool = null; 1015 new_entry_.type_pool = null; 1016 // BEGIN-INTERNAL 1017 new_entry_.style = resid; 1018 // END-INTERNAL 1019 new_entry_.value = map_entry.value.copy(); 1020 final Ref<Res_value> valueRef = new Ref<>(new_entry_.value); 1021 int err = entry.dynamic_ref_table.lookupResourceValue(valueRef); 1022 new_entry_.value = valueRef.get(); 1023 if (err != NO_ERROR) { 1024 System.err.println(String.format( 1025 "Failed to resolve value t=0x%02x d=0x%08x for key 0x%08x.", new_entry_.value.dataType, 1026 new_entry_.value.data, new_key.get())); 1027 return null; 1028 } 1029 // ++new_entry; 1030 ++i; 1031 1032 final int size = dtohs(map_entry.value.size); 1033 // curOffset += size + sizeof(*map)-sizeof(map->value); 1034 curOffset += size + ResTable_map.SIZEOF-Res_value.SIZEOF; 1035 1036 } 1037 new_bag.type_spec_flags = entry.type_flags; 1038 new_bag.entry_count = entry_count; 1039 ResolvedBag result = new_bag; 1040 cached_bags_.put(resid, new_bag); 1041 return result; 1042 } 1043 1044 // In case the parent is a dynamic reference, resolve it. 1045 entry.dynamic_ref_table.lookupResourceId(parent_resid); 1046 1047 // Get the parent and do a merge of the keys. 1048 final ResolvedBag parent_bag = GetBag(parent_resid.get(), child_resids); 1049 if (parent_bag == null) { 1050 // Failed to get the parent that should exist. 1051 System.err.println(String.format("Failed to find parent 0x%08x of bag 0x%08x.", parent_resid.get(), 1052 resid)); 1053 return null; 1054 } 1055 1056 // Create the max possible entries we can make. Once we construct the bag, 1057 // we will realloc to fit to size. 1058 final int max_count = parent_bag.entry_count + dtohl(map.count); 1059 // util::unique_cptr<ResolvedBag> new_bag{reinterpret_cast<ResolvedBag*>( 1060 // malloc(sizeof(ResolvedBag) + (max_count * sizeof(ResolvedBag::Entry))))}; 1061 ResolvedBag new_bag = new ResolvedBag(); 1062 new_bag.entries = new Entry[max_count]; 1063 final ResolvedBag.Entry[] new_entry = new_bag.entries; 1064 int newEntryIndex = 0; 1065 1066 // const ResolvedBag::Entry* parent_entry = parent_bag->entries; 1067 int parentEntryIndex = 0; 1068 // final ResolvedBag.Entry parent_entry_end = parent_entry + parent_bag.entry_count; 1069 final int parentEntryCount = parent_bag.entry_count; 1070 1071 // The keys are expected to be in sorted order. Merge the two bags. 1072 while (map_entry != null && map_entry.myOffset() != map_entry_end && parentEntryIndex != parentEntryCount) { 1073 final Ref<Integer> child_keyRef = new Ref<>(dtohl(map_entry.name.ident)); 1074 if (!is_internal_resid(child_keyRef.get())) { 1075 if (entry.dynamic_ref_table.lookupResourceId(child_keyRef) != NO_ERROR) { 1076 System.err.println(String.format("Failed to resolve key 0x%08x in bag 0x%08x.", child_keyRef.get(), 1077 resid)); 1078 return null; 1079 } 1080 } 1081 int child_key = child_keyRef.get(); 1082 1083 Entry parent_entry = parent_bag.entries[parentEntryIndex]; 1084 if (parent_entry == null) { 1085 parent_entry = new Entry(); 1086 } 1087 1088 if (child_key <= parent_entry.key) { 1089 // Use the child key if it comes before the parent 1090 // or is equal to the parent (overrides). 1091 Entry new_entry_ = new_entry[newEntryIndex] = new Entry(); 1092 new_entry_.cookie = cookie; 1093 new_entry_.key = child_key; 1094 new_entry_.key_pool = null; 1095 new_entry_.type_pool = null; 1096 new_entry_.value = map_entry.value.copy(); 1097 // BEGIN-INTERNAL 1098 new_entry_.style = resid; 1099 // END-INTERNAL 1100 final Ref<Res_value> valueRef = new Ref<>(new_entry_.value); 1101 int err = entry.dynamic_ref_table.lookupResourceValue(valueRef); 1102 new_entry_.value = valueRef.get(); 1103 if (err != NO_ERROR) { 1104 System.err.println(String.format( 1105 "Failed to resolve value t=0x%02x d=0x%08x for key 0x%08x.", new_entry_.value.dataType, 1106 new_entry_.value.data, child_key)); 1107 return null; 1108 } 1109 1110 // ++map_entry; 1111 map_entry = new ResTable_map(map_entry.myBuf(), map_entry.myOffset() + map_entry.value.size + ResTable_map.SIZEOF - Res_value.SIZEOF); 1112 } else { 1113 // Take the parent entry as-is. 1114 // memcpy(new_entry, parent_entry, sizeof(*new_entry)); 1115 new_entry[newEntryIndex] = parent_entry.copy(); 1116 } 1117 1118 if (child_key >= parent_entry.key) { 1119 // Move to the next parent entry if we used it or it was overridden. 1120 // ++parent_entry; 1121 ++parentEntryIndex; 1122 // parent_entry = parent_bag.entries[parentEntryIndex]; 1123 } 1124 // Increment to the next entry to fill. 1125 // ++new_entry; 1126 ++newEntryIndex; 1127 } 1128 1129 // Finish the child entries if they exist. 1130 while (map_entry != null && map_entry.myOffset() != map_entry_end) { 1131 final Ref<Integer> new_key = new Ref<>(map_entry.name.ident); 1132 if (!is_internal_resid(new_key.get())) { 1133 if (entry.dynamic_ref_table.lookupResourceId(new_key) != NO_ERROR) { 1134 System.err.println(String.format("Failed to resolve key 0x%08x in bag 0x%08x.", new_key.get(), 1135 resid)); 1136 return null; 1137 } 1138 } 1139 Entry new_entry_ = new_entry[newEntryIndex] = new Entry(); 1140 new_entry_.cookie = cookie; 1141 new_entry_.key = new_key.get(); 1142 new_entry_.key_pool = null; 1143 new_entry_.type_pool = null; 1144 new_entry_.value = map_entry.value.copy(); 1145 // BEGIN-INTERNAL 1146 new_entry_.style = resid; 1147 // END-INTERNAL 1148 final Ref<Res_value> valueRef = new Ref<>(new_entry_.value); 1149 int err = entry.dynamic_ref_table.lookupResourceValue(valueRef); 1150 new_entry_.value = valueRef.get(); 1151 if (err != NO_ERROR) { 1152 System.err.println(String.format( 1153 "Failed to resolve value t=0x%02x d=0x%08x for key 0x%08x.", 1154 new_entry_.value.dataType, 1155 new_entry_.value.data, new_key.get())); 1156 return null; 1157 } 1158 // ++map_entry; 1159 map_entry = new ResTable_map(map_entry.myBuf(), map_entry.myOffset() + map_entry.value.size + ResTable_map.SIZEOF - Res_value.SIZEOF); 1160 // ++new_entry; 1161 ++newEntryIndex; 1162 } 1163 1164 // Finish the parent entries if they exist. 1165 while (parentEntryIndex != parent_bag.entry_count) { 1166 // Take the rest of the parent entries as-is. 1167 // final int num_entries_to_copy = parent_entry_end - parent_entry; 1168 // final int num_entries_to_copy = parent_bag.entry_count - parentEntryIndex; 1169 // memcpy(new_entry, parent_entry, num_entries_to_copy * sizeof(*new_entry)); 1170 Entry parentEntry = parent_bag.entries[parentEntryIndex]; 1171 new_entry[newEntryIndex] = parentEntry == null ? new Entry() : parentEntry.copy(); 1172 // new_entry += num_entries_to_copy; 1173 ++newEntryIndex; 1174 ++parentEntryIndex; 1175 } 1176 1177 // Resize the resulting array to fit. 1178 // final int actual_count = new_entry - new_bag.entries; 1179 final int actual_count = newEntryIndex; 1180 if (actual_count != max_count) { 1181 // new_bag.reset(reinterpret_cast<ResolvedBag*>(realloc( 1182 // new_bag.release(), sizeof(ResolvedBag) + (actual_count * sizeof(ResolvedBag::Entry))))); 1183 Entry[] resizedEntries = new Entry[actual_count]; 1184 System.arraycopy(new_bag.entries, 0, resizedEntries, 0, actual_count); 1185 new_bag.entries = resizedEntries; 1186 } 1187 1188 // Combine flags from the parent and our own bag. 1189 new_bag.type_spec_flags = entry.type_flags | parent_bag.type_spec_flags; 1190 new_bag.entry_count = actual_count; 1191 ResolvedBag result2 = new_bag; 1192 // cached_bags_[resid] = std::move(new_bag); 1193 cached_bags_.put(resid, new_bag); 1194 return result2; 1195 } 1196 1197 String GetResourceName(int resid) { 1198 ResourceName out_name = new ResourceName(); 1199 if (GetResourceName(resid, out_name)) { 1200 return out_name.package_ + ":" + out_name.type + "@" + out_name.entry; 1201 } else { 1202 return null; 1203 } 1204 } 1205 1206 static boolean Utf8ToUtf16(final String str, Ref<String> out) { 1207 throw new UnsupportedOperationException(); 1208 // ssize_t len = 1209 // utf8_to_utf16_length(reinterpret_cast<final byte*>(str.data()), str.size(), false); 1210 // if (len < 0) { 1211 // return false; 1212 // } 1213 // out.resize(static_cast<int>(len)); 1214 // utf8_to_utf16(reinterpret_cast<final byte*>(str.data()), str.size(), &*out.begin(), 1215 // static_cast<int>(len + 1)); 1216 // return true; 1217 } 1218 1219 // Finds the resource ID assigned to `resource_name`. 1220 // `resource_name` must be of the form '[package:][type/]entry'. 1221 // If no package is specified in `resource_name`, then `fallback_package` is used as the package. 1222 // If no type is specified in `resource_name`, then `fallback_type` is used as the type. 1223 // Returns 0x0 if no resource by that name was found. 1224 // int GetResourceId(final String resource_name, final String fallback_type = {}, 1225 // final String fallback_package = {}); 1226 @SuppressWarnings("NewApi") 1227 public int GetResourceId(final String resource_name, 1228 final String fallback_type, 1229 final String fallback_package) { 1230 final Ref<String> package_name = new Ref<>(null), 1231 type = new Ref<>(null), 1232 entry = new Ref<>(null); 1233 if (!ExtractResourceName(resource_name, package_name, type, entry)) { 1234 return 0; 1235 } 1236 1237 if (entry.get().isEmpty()) { 1238 return 0; 1239 } 1240 1241 if (package_name.get().isEmpty()) { 1242 package_name.set(fallback_package); 1243 } 1244 1245 if (type.get().isEmpty()) { 1246 type.set(fallback_type); 1247 } 1248 1249 String type16 = type.get(); 1250 // if (!Utf8ToUtf16(type, &type16)) { 1251 // return 0; 1252 // } 1253 1254 String entry16 = entry.get(); 1255 // if (!Utf8ToUtf16(entry, &entry16)) { 1256 // return 0; 1257 // } 1258 1259 final String kAttr16 = "attr"; 1260 final String kAttrPrivate16 = "^attr-private"; 1261 1262 for (final PackageGroup package_group : package_groups_) { 1263 for (final ConfiguredPackage package_impl : package_group.packages_) { 1264 LoadedPackage package_= package_impl.loaded_package_; 1265 if (!Objects.equals(package_name.get(), package_.GetPackageName())) { 1266 // All packages in the same group are expected to have the same package name. 1267 break; 1268 } 1269 1270 int resid = package_.FindEntryByName(type16, entry16); 1271 if (resid == 0 && Objects.equals(kAttr16, type16)) { 1272 // Private attributes in libraries (such as the framework) are sometimes encoded 1273 // under the type '^attr-private' in order to leave the ID space of public 'attr' 1274 // free for future additions. Check '^attr-private' for the same name. 1275 resid = package_.FindEntryByName(kAttrPrivate16, entry16); 1276 } 1277 1278 if (resid != 0) { 1279 return fix_package_id(resid, package_group.dynamic_ref_table.mAssignedPackageId); 1280 } 1281 } 1282 } 1283 return 0; 1284 } 1285 1286 // Triggers the re-construction of lists of types that match the set configuration. 1287 // This should always be called when mutating the AssetManager's configuration or ApkAssets set. 1288 void RebuildFilterList() { 1289 for (PackageGroup group : package_groups_) { 1290 for (ConfiguredPackage impl : group.packages_) { 1291 // // Destroy it. 1292 // impl.filtered_configs_.~ByteBucketArray(); 1293 // 1294 // // Re-create it. 1295 // new (impl.filtered_configs_) ByteBucketArray<FilteredConfigGroup>(); 1296 impl.filtered_configs_ = 1297 new ByteBucketArray<FilteredConfigGroup>(new FilteredConfigGroup()) { 1298 @Override 1299 FilteredConfigGroup newInstance() { 1300 return new FilteredConfigGroup(); 1301 } 1302 }; 1303 1304 // Create the filters here. 1305 impl.loaded_package_.ForEachTypeSpec((TypeSpec spec, byte type_index) -> { 1306 FilteredConfigGroup configGroup = impl.filtered_configs_.editItemAt(type_index); 1307 // const auto iter_end = spec->types + spec->type_count; 1308 // for (auto iter = spec->types; iter != iter_end; ++iter) { 1309 for (ResTable_type iter : spec.types) { 1310 ResTable_config this_config = ResTable_config.fromDtoH(iter.config); 1311 if (this_config.match(configuration_)) { 1312 configGroup.configurations.add(this_config); 1313 configGroup.types.add(iter); 1314 } 1315 } 1316 }); 1317 } 1318 } 1319 } 1320 1321 // Purge all resources that are cached and vary by the configuration axis denoted by the 1322 // bitmask `diff`. 1323 // void InvalidateCaches(int diff); 1324 private void InvalidateCaches(int diff) { 1325 if (diff == 0xffffffff) { 1326 // Everything must go. 1327 cached_bags_.clear(); 1328 return; 1329 } 1330 1331 // Be more conservative with what gets purged. Only if the bag has other possible 1332 // variations with respect to what changed (diff) should we remove it. 1333 // for (auto iter = cached_bags_.cbegin(); iter != cached_bags_.cend();) { 1334 for (Integer key : new ArrayList<>(cached_bags_.keySet())) { 1335 // if (diff & iter.second.type_spec_flags) { 1336 if (isTruthy(diff & cached_bags_.get(key).type_spec_flags)) { 1337 // iter = cached_bags_.erase(iter); 1338 cached_bags_.remove(key); 1339 } 1340 } 1341 } 1342 1343 // Creates a new Theme from this AssetManager. 1344 // std.unique_ptr<Theme> NewTheme(); 1345 public Theme NewTheme() { 1346 return new Theme(this); 1347 } 1348 1349 public static class Theme { 1350 // friend class AssetManager2; 1351 // 1352 // public: 1353 // 1354 // 1355 // 1356 // final AssetManager2* GetAssetManager() { return asset_manager_; } 1357 // 1358 public CppAssetManager2 GetAssetManager() { return asset_manager_; } 1359 // 1360 // // Returns a bit mask of configuration changes that will impact this 1361 // // theme (and thus require completely reloading it). 1362 public int GetChangingConfigurations() { return type_spec_flags_; } 1363 1364 // private: 1365 // private DISALLOW_COPY_AND_ASSIGN(Theme); 1366 1367 // Called by AssetManager2. 1368 // private explicit Theme(AssetManager2* asset_manager) : asset_manager_(asset_manager) {} 1369 1370 private final CppAssetManager2 asset_manager_; 1371 private int type_spec_flags_ = 0; 1372 // std.array<std.unique_ptr<Package>, kPackageCount> packages_; 1373 private Package[] packages_ = new Package[kPackageCount]; 1374 1375 public Theme(CppAssetManager2 cppAssetManager2) { 1376 asset_manager_ = cppAssetManager2; 1377 } 1378 1379 private static class ThemeEntry { 1380 static final int SIZEOF = 8 + Res_value.SIZEOF; 1381 1382 ApkAssetsCookie cookie; 1383 int type_spec_flags; 1384 Res_value value; 1385 } 1386 1387 private static class ThemeType { 1388 static final int SIZEOF_WITHOUT_ENTRIES = 8; 1389 1390 int entry_count; 1391 ThemeEntry entries[]; 1392 } 1393 1394 // static final int kPackageCount = std.numeric_limits<byte>.max() + 1; 1395 static final int kPackageCount = 256; 1396 // static final int kTypeCount = std.numeric_limits<byte>.max() + 1; 1397 static final int kTypeCount = 256; 1398 1399 private static class Package { 1400 // Each element of Type will be a dynamically sized object 1401 // allocated to have the entries stored contiguously with the Type. 1402 // std::array<util::unique_cptr<ThemeType>, kTypeCount> types; 1403 ThemeType[] types = new ThemeType[kTypeCount]; 1404 } 1405 1406 // Applies the style identified by `resid` to this theme. This can be called 1407 // multiple times with different styles. By default, any theme attributes that 1408 // are already defined before this call are not overridden. If `force` is set 1409 // to true, this behavior is changed and all theme attributes from the style at 1410 // `resid` are applied. 1411 // Returns false if the style failed to apply. 1412 // boolean ApplyStyle(int resid, boolean force = false); 1413 public boolean ApplyStyle(int resid, boolean force) { 1414 // ATRACE_NAME("Theme::ApplyStyle"); 1415 1416 final ResolvedBag bag = asset_manager_.GetBag(resid); 1417 if (bag == null) { 1418 return false; 1419 } 1420 1421 // Merge the flags from this style. 1422 type_spec_flags_ |= bag.type_spec_flags; 1423 1424 int last_type_idx = -1; 1425 int last_package_idx = -1; 1426 Package last_package = null; 1427 ThemeType last_type = null; 1428 1429 // Iterate backwards, because each bag is sorted in ascending key ID order, meaning we will only 1430 // need to perform one resize per type. 1431 // using reverse_bag_iterator = std::reverse_iterator<const ResolvedBag::Entry*>; 1432 // const auto bag_iter_end = reverse_bag_iterator(begin(bag)); 1433 // for (auto bag_iter = reverse_bag_iterator(end(bag)); bag_iter != bag_iter_end; ++bag_iter) { 1434 List<Entry> bagEntries = new ArrayList<>(Arrays.asList(bag.entries)); 1435 Collections.reverse(bagEntries); 1436 for (ResolvedBag.Entry bag_iter : bagEntries) { 1437 // final int attr_resid = bag_iter.key; 1438 final int attr_resid = bag_iter == null ? 0 : bag_iter.key; 1439 1440 // If the resource ID passed in is not a style, the key can be some other identifier that is not 1441 // a resource ID. We should fail fast instead of operating with strange resource IDs. 1442 if (!is_valid_resid(attr_resid)) { 1443 return false; 1444 } 1445 1446 // We don't use the 0-based index for the type so that we can avoid doing ID validation 1447 // upon lookup. Instead, we keep space for the type ID 0 in our data structures. Since 1448 // the construction of this type is guarded with a resource ID check, it will never be 1449 // populated, and querying type ID 0 will always fail. 1450 int package_idx = get_package_id(attr_resid); 1451 int type_idx = get_type_id(attr_resid); 1452 int entry_idx = get_entry_id(attr_resid); 1453 1454 if (last_package_idx != package_idx) { 1455 Package package_ = packages_[package_idx]; 1456 if (package_ == null) { 1457 package_ = packages_[package_idx] = new Package(); 1458 } 1459 last_package_idx = package_idx; 1460 last_package = package_; 1461 last_type_idx = -1; 1462 } 1463 1464 if (last_type_idx != type_idx) { 1465 ThemeType type = last_package.types[type_idx]; 1466 if (type == null) { 1467 // Allocate enough memory to contain this entry_idx. Since we're iterating in reverse over 1468 // a sorted list of attributes, this shouldn't be resized again during this method call. 1469 // type.reset(reinterpret_cast<ThemeType*>( 1470 // calloc(sizeof(ThemeType) + (entry_idx + 1) * sizeof(ThemeEntry), 1))); 1471 type = last_package.types[type_idx] = new ThemeType(); 1472 type.entries = new ThemeEntry[entry_idx + 1]; 1473 type.entry_count = entry_idx + 1; 1474 } else if (entry_idx >= type.entry_count) { 1475 // Reallocate the memory to contain this entry_idx. Since we're iterating in reverse over 1476 // a sorted list of attributes, this shouldn't be resized again during this method call. 1477 int new_count = entry_idx + 1; 1478 // type.reset(reinterpret_cast<ThemeType*>( 1479 // realloc(type.release(), sizeof(ThemeType) + (new_count * sizeof(ThemeEntry))))); 1480 ThemeEntry[] oldEntries = type.entries; 1481 type.entries = new ThemeEntry[new_count]; 1482 System.arraycopy(oldEntries, 0, type.entries, 0, oldEntries.length); 1483 1484 // Clear out the newly allocated space (which isn't zeroed). 1485 // memset(type.entries + type.entry_count, 0, 1486 // (new_count - type.entry_count) * sizeof(ThemeEntry)); 1487 type.entry_count = new_count; 1488 } 1489 last_type_idx = type_idx; 1490 last_type = type; 1491 } 1492 1493 ThemeEntry entry = last_type.entries[entry_idx]; 1494 if (entry == null) { 1495 entry = last_type.entries[entry_idx] = new ThemeEntry(); 1496 entry.value = new Res_value(); 1497 } 1498 if (force || (entry.value.dataType == Res_value.TYPE_NULL && 1499 entry.value.data != Res_value.DATA_NULL_EMPTY)) { 1500 entry.cookie = bag_iter.cookie; 1501 entry.type_spec_flags |= bag.type_spec_flags; 1502 entry.value = bag_iter.value; 1503 } 1504 } 1505 return true; 1506 } 1507 1508 // Retrieve a value in the theme. If the theme defines this value, returns an asset cookie 1509 // indicating which ApkAssets it came from and populates `out_value` with the value. 1510 // `out_flags` is populated with a bitmask of the configuration axis with which the resource 1511 // varies. 1512 // 1513 // If the attribute is not found, returns kInvalidCookie. 1514 // 1515 // NOTE: This function does not do reference traversal. If you want to follow references to other 1516 // resources to get the "real" value to use, you need to call ResolveReference() after this 1517 // function. 1518 // ApkAssetsCookie GetAttribute(int resid, Res_value* out_value, 1519 // int* out_flags) const; 1520 public ApkAssetsCookie GetAttribute(int resid, Ref<Res_value> out_value, 1521 final Ref<Integer> out_flags) { 1522 int cnt = 20; 1523 1524 int type_spec_flags = 0; 1525 1526 do { 1527 int package_idx = get_package_id(resid); 1528 Package package_ = packages_[package_idx]; 1529 if (package_ != null) { 1530 // The themes are constructed with a 1-based type ID, so no need to decrement here. 1531 int type_idx = get_type_id(resid); 1532 ThemeType type = package_.types[type_idx]; 1533 if (type != null) { 1534 int entry_idx = get_entry_id(resid); 1535 if (entry_idx < type.entry_count) { 1536 ThemeEntry entry = type.entries[entry_idx]; 1537 if (entry == null) { 1538 entry = new ThemeEntry(); 1539 entry.value = new Res_value(); 1540 } 1541 type_spec_flags |= entry.type_spec_flags; 1542 1543 if (entry.value.dataType == Res_value.TYPE_ATTRIBUTE) { 1544 if (cnt > 0) { 1545 cnt--; 1546 resid = entry.value.data; 1547 continue; 1548 } 1549 return K_INVALID_COOKIE; 1550 } 1551 1552 // @null is different than @empty. 1553 if (entry.value.dataType == Res_value.TYPE_NULL && 1554 entry.value.data != Res_value.DATA_NULL_EMPTY) { 1555 return K_INVALID_COOKIE; 1556 } 1557 1558 out_value.set(entry.value); 1559 out_flags.set(type_spec_flags); 1560 return entry.cookie; 1561 } 1562 } 1563 } 1564 break; 1565 } while (true); 1566 return K_INVALID_COOKIE; 1567 } 1568 1569 // This is like ResolveReference(), but also takes 1570 // care of resolving attribute references to the theme. 1571 // ApkAssetsCookie ResolveAttributeReference(ApkAssetsCookie cookie, Res_value* in_out_value, 1572 // ResTable_config in_out_selected_config = null, 1573 // int* in_out_type_spec_flags = null, 1574 // int* out_last_ref = null); 1575 ApkAssetsCookie ResolveAttributeReference(ApkAssetsCookie cookie, Ref<Res_value> in_out_value, 1576 final Ref<ResTable_config> in_out_selected_config, 1577 final Ref<Integer> in_out_type_spec_flags, 1578 final Ref<Integer> out_last_ref) { 1579 if (in_out_value.get().dataType == Res_value.TYPE_ATTRIBUTE) { 1580 final Ref<Integer> new_flags = new Ref<>(0); 1581 cookie = GetAttribute(in_out_value.get().data, in_out_value, new_flags); 1582 if (cookie.intValue() == kInvalidCookie) { 1583 return K_INVALID_COOKIE; 1584 } 1585 1586 if (in_out_type_spec_flags != null) { 1587 // *in_out_type_spec_flags |= new_flags; 1588 in_out_type_spec_flags.set(in_out_type_spec_flags.get() | new_flags.get()); 1589 } 1590 } 1591 return asset_manager_.ResolveReference(cookie, in_out_value, in_out_selected_config, 1592 in_out_type_spec_flags, out_last_ref); 1593 } 1594 1595 // void Clear(); 1596 public void Clear() { 1597 type_spec_flags_ = 0; 1598 for (int i = 0; i < packages_.length; i++) { 1599 // package_.reset(); 1600 packages_[i] = null; 1601 } 1602 } 1603 1604 // Sets this Theme to be a copy of `o` if `o` has the same AssetManager as this Theme. 1605 // Returns false if the AssetManagers of the Themes were not compatible. 1606 // boolean SetTo(final Theme& o); 1607 public boolean SetTo(final Theme o) { 1608 if (this == o) { 1609 return true; 1610 } 1611 1612 type_spec_flags_ = o.type_spec_flags_; 1613 1614 boolean copy_only_system = asset_manager_ != o.asset_manager_; 1615 1616 // for (int p = 0; p < packages_.size(); p++) { 1617 // final Package package_ = o.packages_[p].get(); 1618 for (int p = 0; p < packages_.length; p++) { 1619 Package package_ = o.packages_[p]; 1620 if (package_ == null || (copy_only_system && p != 0x01)) { 1621 // The other theme doesn't have this package, clear ours. 1622 packages_[p] = new Package(); 1623 continue; 1624 } 1625 1626 if (packages_[p] == null) { 1627 // The other theme has this package, but we don't. Make one. 1628 packages_[p] = new Package(); 1629 } 1630 1631 // for (int t = 0; t < package_.types.size(); t++) { 1632 // final Type type = package_.types[t].get(); 1633 for (int t = 0; t < package_.types.length; t++) { 1634 ThemeType type = package_.types[t]; 1635 if (type == null) { 1636 // The other theme doesn't have this type, clear ours. 1637 // packages_[p].types[t].reset(); 1638 continue; 1639 } 1640 1641 // Create a new type and update it to theirs. 1642 // const size_t type_alloc_size = sizeof(ThemeType) + (type->entry_count * sizeof(ThemeEntry)); 1643 // void* copied_data = malloc(type_alloc_size); 1644 ThemeType copied_data = new ThemeType(); 1645 copied_data.entry_count = type.entry_count; 1646 // memcpy(copied_data, type, type_alloc_size); 1647 ThemeEntry[] newEntries = copied_data.entries = new ThemeEntry[type.entry_count]; 1648 for (int i = 0; i < type.entry_count; i++) { 1649 ThemeEntry entry = type.entries[i]; 1650 ThemeEntry newEntry = new ThemeEntry(); 1651 if (entry != null) { 1652 newEntry.cookie = entry.cookie; 1653 newEntry.type_spec_flags = entry.type_spec_flags; 1654 newEntry.value = entry.value.copy(); 1655 } else { 1656 newEntry.value = Res_value.NULL_VALUE; 1657 } 1658 newEntries[i] = newEntry; 1659 } 1660 1661 packages_[p].types[t] = copied_data; 1662 // packages_[p].types[t].reset(reinterpret_cast<Type*>(copied_data)); 1663 } 1664 } 1665 return true; 1666 } 1667 1668 // 1669 } // namespace android 1670 1671 public List<AssetPath> getAssetPaths() { 1672 ArrayList<AssetPath> assetPaths = new ArrayList<>(apk_assets_.size()); 1673 for (CppApkAssets apkAssets : apk_assets_) { 1674 FsFile fsFile = Fs.newFile(apkAssets.GetPath()); 1675 assetPaths.add(new AssetPath(fsFile, apkAssets.GetLoadedArsc().IsSystem())); 1676 } 1677 return assetPaths; 1678 } 1679 1680 } 1681