|  | /* | 
|  | * Copyright (C) 2018 The Android Open Source Project | 
|  | * | 
|  | * Licensed under the Apache License, Version 2.0 (the "License"); | 
|  | * you may not use this file except in compliance with the License. | 
|  | * You may obtain a copy of the License at | 
|  | * | 
|  | *      http://www.apache.org/licenses/LICENSE-2.0 | 
|  | * | 
|  | * Unless required by applicable law or agreed to in writing, software | 
|  | * distributed under the License is distributed on an "AS IS" BASIS, | 
|  | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
|  | * See the License for the specific language governing permissions and | 
|  | * limitations under the License. | 
|  | */ | 
|  |  | 
|  | #include "DumpManifest.h" | 
|  |  | 
|  | #include <algorithm> | 
|  | #include <array> | 
|  | #include <memory> | 
|  | #include <set> | 
|  | #include <string_view> | 
|  | #include <vector> | 
|  |  | 
|  | #include "LoadedApk.h" | 
|  | #include "SdkConstants.h" | 
|  | #include "ValueVisitor.h" | 
|  | #include "androidfw/ConfigDescription.h" | 
|  | #include "io/File.h" | 
|  | #include "io/FileStream.h" | 
|  | #include "process/IResourceTableConsumer.h" | 
|  | #include "xml/XmlDom.h" | 
|  |  | 
|  | using ::android::base::StringPrintf; | 
|  | using ::android::ConfigDescription; | 
|  |  | 
|  | namespace aapt { | 
|  |  | 
|  | /** | 
|  | * These are attribute resource constants for the platform, as found in android.R.attr. | 
|  | */ | 
|  | enum { | 
|  | LABEL_ATTR = 0x01010001, | 
|  | ICON_ATTR = 0x01010002, | 
|  | NAME_ATTR = 0x01010003, | 
|  | PERMISSION_ATTR = 0x01010006, | 
|  | EXPORTED_ATTR = 0x01010010, | 
|  | GRANT_URI_PERMISSIONS_ATTR = 0x0101001b, | 
|  | PRIORITY_ATTR = 0x0101001c, | 
|  | RESOURCE_ATTR = 0x01010025, | 
|  | DEBUGGABLE_ATTR = 0x0101000f, | 
|  | TARGET_PACKAGE_ATTR = 0x01010021, | 
|  | VALUE_ATTR = 0x01010024, | 
|  | VERSION_CODE_ATTR = 0x0101021b, | 
|  | VERSION_NAME_ATTR = 0x0101021c, | 
|  | SCREEN_ORIENTATION_ATTR = 0x0101001e, | 
|  | MIN_SDK_VERSION_ATTR = 0x0101020c, | 
|  | MAX_SDK_VERSION_ATTR = 0x01010271, | 
|  | REQ_TOUCH_SCREEN_ATTR = 0x01010227, | 
|  | REQ_KEYBOARD_TYPE_ATTR = 0x01010228, | 
|  | REQ_HARD_KEYBOARD_ATTR = 0x01010229, | 
|  | REQ_NAVIGATION_ATTR = 0x0101022a, | 
|  | REQ_FIVE_WAY_NAV_ATTR = 0x01010232, | 
|  | TARGET_SDK_VERSION_ATTR = 0x01010270, | 
|  | TEST_ONLY_ATTR = 0x01010272, | 
|  | ANY_DENSITY_ATTR = 0x0101026c, | 
|  | GL_ES_VERSION_ATTR = 0x01010281, | 
|  | SMALL_SCREEN_ATTR = 0x01010284, | 
|  | NORMAL_SCREEN_ATTR = 0x01010285, | 
|  | LARGE_SCREEN_ATTR = 0x01010286, | 
|  | XLARGE_SCREEN_ATTR = 0x010102bf, | 
|  | REQUIRED_ATTR = 0x0101028e, | 
|  | INSTALL_LOCATION_ATTR = 0x010102b7, | 
|  | SCREEN_SIZE_ATTR = 0x010102ca, | 
|  | SCREEN_DENSITY_ATTR = 0x010102cb, | 
|  | REQUIRES_SMALLEST_WIDTH_DP_ATTR = 0x01010364, | 
|  | COMPATIBLE_WIDTH_LIMIT_DP_ATTR = 0x01010365, | 
|  | LARGEST_WIDTH_LIMIT_DP_ATTR = 0x01010366, | 
|  | PUBLIC_KEY_ATTR = 0x010103a6, | 
|  | CATEGORY_ATTR = 0x010103e8, | 
|  | BANNER_ATTR = 0x10103f2, | 
|  | ISGAME_ATTR = 0x10103f4, | 
|  | VERSION_ATTR = 0x01010519, | 
|  | CERT_DIGEST_ATTR = 0x01010548, | 
|  | REQUIRED_FEATURE_ATTR = 0x01010554, | 
|  | REQUIRED_NOT_FEATURE_ATTR = 0x01010555, | 
|  | IS_STATIC_ATTR = 0x0101055a, | 
|  | REQUIRED_SYSTEM_PROPERTY_NAME_ATTR = 0x01010565, | 
|  | REQUIRED_SYSTEM_PROPERTY_VALUE_ATTR = 0x01010566, | 
|  | COMPILE_SDK_VERSION_ATTR = 0x01010572, | 
|  | COMPILE_SDK_VERSION_CODENAME_ATTR = 0x01010573, | 
|  | VERSION_MAJOR_ATTR = 0x01010577, | 
|  | PACKAGE_TYPE_ATTR = 0x01010587, | 
|  | USES_PERMISSION_FLAGS_ATTR = 0x01010644, | 
|  | }; | 
|  |  | 
|  | const std::string& kAndroidNamespace = "http://schemas.android.com/apk/res/android"; | 
|  | constexpr int kNeverForLocation = 0x00010000; | 
|  |  | 
|  | /** Retrieves the attribute of the element with the specified attribute resource id. */ | 
|  | static xml::Attribute* FindAttribute(xml::Element *el, uint32_t resd_id) { | 
|  | for (auto& a : el->attributes) { | 
|  | if (a.compiled_attribute && a.compiled_attribute.value().id) { | 
|  | if (a.compiled_attribute.value().id.value() == resd_id) { | 
|  | return std::move(&a); | 
|  | } | 
|  | } | 
|  | } | 
|  | return nullptr; | 
|  | } | 
|  |  | 
|  | /** Retrieves the attribute of the element that has the specified namespace and attribute name. */ | 
|  | static xml::Attribute* FindAttribute(xml::Element *el, const std::string &package, | 
|  | const std::string &name) { | 
|  | return el->FindAttribute(package, name); | 
|  | } | 
|  |  | 
|  | class Architectures { | 
|  | public: | 
|  | std::set<std::string> architectures; | 
|  | std::set<std::string> alt_architectures; | 
|  |  | 
|  | void Print(text::Printer* printer) { | 
|  | if (!architectures.empty()) { | 
|  | printer->Print("native-code:"); | 
|  | for (auto& arch : architectures) { | 
|  | printer->Print(StringPrintf(" '%s'", arch.data())); | 
|  | } | 
|  | printer->Print("\n"); | 
|  | } | 
|  | if (!alt_architectures.empty()) { | 
|  | printer->Print("alt-native-code:"); | 
|  | for (auto& arch : alt_architectures) { | 
|  | printer->Print(StringPrintf(" '%s'", arch.data())); | 
|  | } | 
|  | printer->Print("\n"); | 
|  | } | 
|  | } | 
|  |  | 
|  | void ToProto(pb::Badging* out_badging) { | 
|  | auto out_architectures = out_badging->mutable_architectures(); | 
|  | for (auto& arch : architectures) { | 
|  | out_architectures->add_architectures(arch); | 
|  | } | 
|  | for (auto& arch : alt_architectures) { | 
|  | out_architectures->add_alt_architectures(arch); | 
|  | } | 
|  | } | 
|  | }; | 
|  |  | 
|  | const static std::array<std::string_view, 14> printable_components{"app-widget", | 
|  | "device-admin", | 
|  | "ime", | 
|  | "wallpaper", | 
|  | "accessibility", | 
|  | "print-service", | 
|  | "payment", | 
|  | "search", | 
|  | "document-provider", | 
|  | "launcher", | 
|  | "notification-listener", | 
|  | "dream", | 
|  | "camera", | 
|  | "camera-secure"}; | 
|  |  | 
|  | class Components { | 
|  | public: | 
|  | std::set<std::string, std::less<>> discovered_components; | 
|  | bool other_activities = false; | 
|  | bool other_receivers = false; | 
|  | bool other_services = false; | 
|  |  | 
|  | void Print(text::Printer* printer) { | 
|  | for (auto& component : printable_components) { | 
|  | if (discovered_components.find(component) != discovered_components.end()) { | 
|  | printer->Print(StringPrintf("provides-component:'%s'\n", component.data())); | 
|  | } | 
|  | } | 
|  | // Print presence of main activity | 
|  | if (discovered_components.find("main") != discovered_components.end()) { | 
|  | printer->Print("main\n"); | 
|  | } | 
|  |  | 
|  | if (other_activities) { | 
|  | printer->Print("other-activities\n"); | 
|  | } | 
|  | if (other_receivers) { | 
|  | printer->Print("other-receivers\n"); | 
|  | } | 
|  | if (other_services) { | 
|  | printer->Print("other-services\n"); | 
|  | } | 
|  | } | 
|  |  | 
|  | void ToProto(pb::Badging* out_badging) { | 
|  | auto out_components = out_badging->mutable_components(); | 
|  | for (auto& component : printable_components) { | 
|  | auto discovered = discovered_components.find(component); | 
|  | if (discovered != discovered_components.end()) { | 
|  | out_components->add_provided_components(*discovered); | 
|  | } | 
|  | } | 
|  | out_components->set_main(discovered_components.find("main") != discovered_components.end()); | 
|  | out_components->set_other_activities(other_activities); | 
|  | out_components->set_other_receivers(other_receivers); | 
|  | out_components->set_other_services(other_services); | 
|  | } | 
|  | }; | 
|  |  | 
|  | class CommonFeatureGroup; | 
|  | class FeatureGroup; | 
|  | class SupportsScreen; | 
|  |  | 
|  | class ManifestExtractor { | 
|  | public: | 
|  |  | 
|  | explicit ManifestExtractor(LoadedApk* apk, DumpManifestOptions& options) | 
|  | : apk_(apk), options_(options) { } | 
|  |  | 
|  | class Element { | 
|  | public: | 
|  | Element() = default; | 
|  | virtual ~Element() = default; | 
|  |  | 
|  | static std::unique_ptr<Element> Inflate(ManifestExtractor* extractor, xml::Element* el); | 
|  |  | 
|  | /** Writes out the extracted contents of the element. */ | 
|  | virtual void Print(text::Printer* printer) { | 
|  | } | 
|  |  | 
|  | /** Saves extracted information into Badging proto. */ | 
|  | virtual void ToProto(pb::Badging* out_badging) { | 
|  | } | 
|  |  | 
|  | /** Adds an element to the list of children of the element. */ | 
|  | void AddChild(std::unique_ptr<Element>& child) { children_.push_back(std::move(child)); } | 
|  |  | 
|  | template <typename Predicate> | 
|  | void Filter(Predicate&& func) { | 
|  | children_.erase(std::remove_if(children_.begin(), children_.end(), | 
|  | [&](const auto& e) { return func(e.get()); }), | 
|  | children_.end()); | 
|  | } | 
|  |  | 
|  | /** Retrieves the list of children of the element. */ | 
|  | const std::vector<std::unique_ptr<Element>>& children() const { | 
|  | return children_; | 
|  | } | 
|  |  | 
|  | /** Retrieves the extracted xml element tag. */ | 
|  | const std::string tag() const { | 
|  | return tag_; | 
|  | } | 
|  |  | 
|  | protected: | 
|  | ManifestExtractor* extractor() const { | 
|  | return extractor_; | 
|  | } | 
|  |  | 
|  | /** Retrieves and stores the information extracted from the xml element. */ | 
|  | virtual void Extract(xml::Element* el) { } | 
|  |  | 
|  | /* | 
|  | * Retrieves a configuration value of the resource entry that best matches the specified | 
|  | * configuration. | 
|  | */ | 
|  | static Value* BestConfigValue(ResourceEntry* entry, | 
|  | const ConfigDescription& match) { | 
|  | if (!entry) { | 
|  | return nullptr; | 
|  | } | 
|  |  | 
|  | // Determine the config that best matches the desired config | 
|  | ResourceConfigValue* best_value = nullptr; | 
|  | for (auto& value : entry->values) { | 
|  | if (!value->config.match(match)) { | 
|  | continue; | 
|  | } | 
|  |  | 
|  | if (best_value != nullptr) { | 
|  | if (!value->config.isBetterThan(best_value->config, &match)) { | 
|  | if (value->config.compare(best_value->config) != 0) { | 
|  | continue; | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | best_value = value.get(); | 
|  | } | 
|  |  | 
|  | // The entry has no values | 
|  | if (!best_value) { | 
|  | return nullptr; | 
|  | } | 
|  |  | 
|  | return best_value->value.get(); | 
|  | } | 
|  |  | 
|  | /** Retrieves the resource assigned to the specified resource id if one exists. */ | 
|  | Value* FindValueById(const ResourceTable* table, const ResourceId& res_id, | 
|  | const ConfigDescription& config = DefaultConfig()) { | 
|  | if (table) { | 
|  | for (auto& package : table->packages) { | 
|  | for (auto& type : package->types) { | 
|  | for (auto& entry : type->entries) { | 
|  | if (entry->id && entry->id.value() == res_id.id) { | 
|  | if (auto value = BestConfigValue(entry.get(), config)) { | 
|  | return value; | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  | return nullptr; | 
|  | } | 
|  |  | 
|  | /** Attempts to resolve the reference to a non-reference value. */ | 
|  | Value* ResolveReference(Reference* ref, const ConfigDescription& config = DefaultConfig()) { | 
|  | const int kMaxIterations = 40; | 
|  | int i = 0; | 
|  | while (ref && ref->id && i++ < kMaxIterations) { | 
|  | auto table = extractor_->apk_->GetResourceTable(); | 
|  | if (auto value = FindValueById(table, ref->id.value(), config)) { | 
|  | if (ValueCast<Reference>(value)) { | 
|  | ref = ValueCast<Reference>(value); | 
|  | } else { | 
|  | return value; | 
|  | } | 
|  | } | 
|  | } | 
|  | return nullptr; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Retrieves the integer value of the attribute . If the value of the attribute is a reference, | 
|  | * this will attempt to resolve the reference to an integer value. | 
|  | **/ | 
|  | int32_t* GetAttributeInteger(xml::Attribute* attr, | 
|  | const ConfigDescription& config = DefaultConfig()) { | 
|  | if (attr != nullptr) { | 
|  | if (attr->compiled_value) { | 
|  | // Resolve references using the configuration | 
|  | Value* value = attr->compiled_value.get(); | 
|  | if (ValueCast<Reference>(value)) { | 
|  | value = ResolveReference(ValueCast<Reference>(value), config); | 
|  | } else { | 
|  | value = attr->compiled_value.get(); | 
|  | } | 
|  | // Retrieve the integer data if possible | 
|  | if (value != nullptr) { | 
|  | if (BinaryPrimitive* intValue = ValueCast<BinaryPrimitive>(value)) { | 
|  | return (int32_t*) &intValue->value.data; | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  | return nullptr; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * A version of GetAttributeInteger that returns a default integer if the attribute does not | 
|  | * exist or cannot be resolved to an integer value. | 
|  | **/ | 
|  | int32_t GetAttributeIntegerDefault(xml::Attribute* attr, int32_t def, | 
|  | const ConfigDescription& config = DefaultConfig()) { | 
|  | auto value = GetAttributeInteger(attr, config); | 
|  | if (value) { | 
|  | return *value; | 
|  | } | 
|  | return def; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Retrieves the string value of the attribute. If the value of the attribute is a reference, | 
|  | * this will attempt to resolve the reference to a string value. | 
|  | **/ | 
|  | const std::string* GetAttributeString(xml::Attribute* attr, | 
|  | const ConfigDescription& config = DefaultConfig()) { | 
|  | if (attr != nullptr) { | 
|  | if (attr->compiled_value) { | 
|  | // Resolve references using the configuration | 
|  | Value* value = attr->compiled_value.get(); | 
|  | if (ValueCast<Reference>(value)) { | 
|  | value = ResolveReference(ValueCast<Reference>(value), config); | 
|  | } else { | 
|  | value = attr->compiled_value.get(); | 
|  | } | 
|  |  | 
|  | // Retrieve the string data of the value if possible | 
|  | if (value != nullptr) { | 
|  | if (String* intValue = ValueCast<String>(value)) { | 
|  | return &(*intValue->value); | 
|  | } else if (RawString* rawValue = ValueCast<RawString>(value)) { | 
|  | return &(*rawValue->value); | 
|  | } else if (FileReference* strValue = ValueCast<FileReference>(value)) { | 
|  | return &(*strValue->path); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | if (!attr->value.empty()) { | 
|  | return &attr->value; | 
|  | } | 
|  | } | 
|  | return nullptr; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * A version of GetAttributeString that returns a default string if the attribute does not | 
|  | * exist or cannot be resolved to an string value. | 
|  | **/ | 
|  | std::string GetAttributeStringDefault(xml::Attribute* attr, std::string def, | 
|  | const ConfigDescription& config = DefaultConfig()) { | 
|  | auto value = GetAttributeString(attr, config); | 
|  | if (value) { | 
|  | return *value; | 
|  | } | 
|  | return def; | 
|  | } | 
|  |  | 
|  | private: | 
|  | ManifestExtractor* extractor_; | 
|  | std::vector<std::unique_ptr<Element>> children_; | 
|  | std::string tag_; | 
|  | }; | 
|  |  | 
|  | friend Element; | 
|  |  | 
|  | /** Creates a default configuration used to retrieve resources. */ | 
|  | static ConfigDescription DefaultConfig() { | 
|  | ConfigDescription config; | 
|  | config.orientation = android::ResTable_config::ORIENTATION_PORT; | 
|  | config.density = android::ResTable_config::DENSITY_MEDIUM; | 
|  | config.sdkVersion = SDK_CUR_DEVELOPMENT;  // Very high. | 
|  | config.screenWidthDp = 320; | 
|  | config.screenHeightDp = 480; | 
|  | config.smallestScreenWidthDp = 320; | 
|  | config.screenLayout |= android::ResTable_config::SCREENSIZE_NORMAL; | 
|  | return config; | 
|  | } | 
|  |  | 
|  | bool Extract(IDiagnostics* diag); | 
|  | bool Dump(text::Printer* printer); | 
|  | bool DumpProto(pb::Badging* out_badging); | 
|  |  | 
|  | /** Recursively visit the xml element tree and return a processed badging element tree. */ | 
|  | std::unique_ptr<Element> Visit(xml::Element* element); | 
|  |  | 
|  | /** Raises the target sdk value if the min target is greater than the current target. */ | 
|  | void RaiseTargetSdk(int32_t min_target) { | 
|  | if (min_target > target_sdk_) { | 
|  | target_sdk_ = min_target; | 
|  | } | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Retrieves the default feature group that features are added into when <uses-feature> | 
|  | * are not in a <feature-group> element. | 
|  | **/ | 
|  | CommonFeatureGroup* common_feature_group() { | 
|  | return commonFeatureGroup_.get(); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Retrieves a mapping of density values to Configurations for retrieving resources that would be | 
|  | * used for that density setting. | 
|  | **/ | 
|  | const std::map<uint16_t, ConfigDescription> densities() const { | 
|  | return densities_; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Retrieves a mapping of locale BCP 47 strings to Configurations for retrieving resources that | 
|  | * would be used for that locale setting. | 
|  | **/ | 
|  | const std::map<std::string, ConfigDescription> locales() const { | 
|  | return locales_; | 
|  | } | 
|  |  | 
|  | /** Retrieves the current stack of parent during data extraction. */ | 
|  | const std::vector<Element*> parent_stack() const { | 
|  | return parent_stack_; | 
|  | } | 
|  |  | 
|  | int32_t target_sdk() const { | 
|  | return target_sdk_; | 
|  | } | 
|  |  | 
|  | LoadedApk* const apk_; | 
|  | DumpManifestOptions& options_; | 
|  |  | 
|  | private: | 
|  | std::unique_ptr<xml::XmlResource> doc_; | 
|  | std::unique_ptr<CommonFeatureGroup> commonFeatureGroup_ = util::make_unique<CommonFeatureGroup>(); | 
|  | std::map<std::string, ConfigDescription> locales_; | 
|  | std::map<uint16_t, ConfigDescription> densities_; | 
|  | std::vector<Element*> parent_stack_; | 
|  | int32_t target_sdk_ = 0; | 
|  |  | 
|  | std::unique_ptr<ManifestExtractor::Element> root_element_; | 
|  | std::vector<std::unique_ptr<ManifestExtractor::Element>> implied_permissions_; | 
|  | std::vector<FeatureGroup*> feature_groups_; | 
|  | Components components_; | 
|  | Architectures architectures_; | 
|  | const SupportsScreen* supports_screen_; | 
|  | }; | 
|  |  | 
|  | template<typename T> T* ElementCast(ManifestExtractor::Element* element); | 
|  |  | 
|  | /** Recurs through the children of the specified root in depth-first order. */ | 
|  | static void ForEachChild(ManifestExtractor::Element* root, | 
|  | std::function<void(ManifestExtractor::Element*)> f) { | 
|  | for (auto& child : root->children()) { | 
|  | f(child.get()); | 
|  | ForEachChild(child.get(), f); | 
|  | } | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Checks the element and its recursive children for an element that makes the specified | 
|  | * conditional function return true. Returns the first element that makes the conditional function | 
|  | * return true. | 
|  | **/ | 
|  | static ManifestExtractor::Element* FindElement(ManifestExtractor::Element* root, | 
|  | std::function<bool(ManifestExtractor::Element*)> f) { | 
|  | if (f(root)) { | 
|  | return root; | 
|  | } | 
|  | for (auto& child : root->children()) { | 
|  | if (auto b2 = FindElement(child.get(), f)) { | 
|  | return b2; | 
|  | } | 
|  | } | 
|  | return nullptr; | 
|  | } | 
|  |  | 
|  | /** Represents the <manifest> elements **/ | 
|  | class Manifest : public ManifestExtractor::Element { | 
|  | public: | 
|  | Manifest() = default; | 
|  | bool only_package_name; | 
|  | std::string package; | 
|  | int32_t versionCode; | 
|  | std::string versionName; | 
|  | const std::string* split = nullptr; | 
|  | const std::string* platformVersionName = nullptr; | 
|  | const std::string* platformVersionCode = nullptr; | 
|  | const int32_t* platformVersionNameInt = nullptr; | 
|  | const int32_t* platformVersionCodeInt = nullptr; | 
|  | const int32_t* compilesdkVersion = nullptr; | 
|  | const std::string* compilesdkVersionCodename = nullptr; | 
|  | const int32_t* installLocation = nullptr; | 
|  |  | 
|  | void Extract(xml::Element* manifest) override { | 
|  | package = GetAttributeStringDefault(FindAttribute(manifest, {}, "package"), ""); | 
|  | versionCode = GetAttributeIntegerDefault(FindAttribute(manifest, VERSION_CODE_ATTR), 0); | 
|  | versionName = GetAttributeStringDefault(FindAttribute(manifest, VERSION_NAME_ATTR), ""); | 
|  | split = GetAttributeString(FindAttribute(manifest, {}, "split")); | 
|  |  | 
|  | // Extract the platform build info | 
|  | platformVersionName = GetAttributeString(FindAttribute(manifest, {}, | 
|  | "platformBuildVersionName")); | 
|  | platformVersionCode = GetAttributeString(FindAttribute(manifest, {}, | 
|  | "platformBuildVersionCode")); | 
|  | platformVersionNameInt = GetAttributeInteger(FindAttribute(manifest, {}, | 
|  | "platformBuildVersionName")); | 
|  | platformVersionCodeInt = GetAttributeInteger(FindAttribute(manifest, {}, | 
|  | "platformBuildVersionCode")); | 
|  |  | 
|  | // Extract the compile sdk info | 
|  | compilesdkVersion = GetAttributeInteger(FindAttribute(manifest, COMPILE_SDK_VERSION_ATTR)); | 
|  | compilesdkVersionCodename = GetAttributeString( | 
|  | FindAttribute(manifest, COMPILE_SDK_VERSION_CODENAME_ATTR)); | 
|  | installLocation = GetAttributeInteger(FindAttribute(manifest, INSTALL_LOCATION_ATTR)); | 
|  | } | 
|  |  | 
|  | void ToProto(pb::Badging* out_badging) override { | 
|  | auto out_package = out_badging->mutable_package(); | 
|  | out_package->set_package(package); | 
|  | out_package->set_version_code(versionCode); | 
|  | out_package->set_version_name(versionName); | 
|  | if (compilesdkVersion) { | 
|  | out_package->set_compile_sdk_version(*compilesdkVersion); | 
|  | } | 
|  | if (compilesdkVersionCodename) { | 
|  | out_package->set_compile_sdk_version_codename(*compilesdkVersionCodename); | 
|  | } | 
|  | if (platformVersionName) { | 
|  | out_package->set_platform_version_name(*platformVersionName); | 
|  | } else if (platformVersionNameInt) { | 
|  | out_package->set_platform_version_name(std::to_string(*platformVersionNameInt)); | 
|  | } | 
|  | if (platformVersionCode) { | 
|  | out_package->set_platform_version_code(*platformVersionCode); | 
|  | } else if (platformVersionCodeInt) { | 
|  | out_package->set_platform_version_code(std::to_string(*platformVersionCodeInt)); | 
|  | } | 
|  |  | 
|  | if (installLocation) { | 
|  | switch (*installLocation) { | 
|  | case 0: | 
|  | out_package->set_install_location(pb::PackageInfo_InstallLocation_AUTO); | 
|  | break; | 
|  | case 1: | 
|  | out_package->set_install_location(pb::PackageInfo_InstallLocation_INTERNAL_ONLY); | 
|  | break; | 
|  | case 2: | 
|  | out_package->set_install_location(pb::PackageInfo_InstallLocation_PREFER_EXTERNAL); | 
|  | break; | 
|  | default: | 
|  | break; | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | void Print(text::Printer* printer) override { | 
|  | if (only_package_name) { | 
|  | printer->Println(StringPrintf("package: %s", package.data())); | 
|  | } else { | 
|  | PrintFull(printer); | 
|  | } | 
|  | } | 
|  |  | 
|  | void PrintFull(text::Printer* printer) { | 
|  | printer->Print(StringPrintf("package: name='%s' ", package.data())); | 
|  | printer->Print(StringPrintf("versionCode='%s' ", | 
|  | (versionCode > 0) ? std::to_string(versionCode).data() : "")); | 
|  | printer->Print(StringPrintf("versionName='%s'", versionName.data())); | 
|  |  | 
|  | if (split) { | 
|  | printer->Print(StringPrintf(" split='%s'", split->data())); | 
|  | } | 
|  | if (platformVersionName) { | 
|  | printer->Print(StringPrintf(" platformBuildVersionName='%s'", platformVersionName->data())); | 
|  | } else if (platformVersionNameInt) { | 
|  | printer->Print(StringPrintf(" platformBuildVersionName='%d'", *platformVersionNameInt)); | 
|  | } | 
|  | if (platformVersionCode) { | 
|  | printer->Print(StringPrintf(" platformBuildVersionCode='%s'", platformVersionCode->data())); | 
|  | } else if (platformVersionCodeInt) { | 
|  | printer->Print(StringPrintf(" platformBuildVersionCode='%d'", *platformVersionCodeInt)); | 
|  | } | 
|  | if (compilesdkVersion) { | 
|  | printer->Print(StringPrintf(" compileSdkVersion='%d'", *compilesdkVersion)); | 
|  | } | 
|  | if (compilesdkVersionCodename) { | 
|  | printer->Print(StringPrintf(" compileSdkVersionCodename='%s'", | 
|  | compilesdkVersionCodename->data())); | 
|  | } | 
|  | printer->Print("\n"); | 
|  |  | 
|  | if (installLocation) { | 
|  | switch (*installLocation) { | 
|  | case 0: | 
|  | printer->Print("install-location:'auto'\n"); | 
|  | break; | 
|  | case 1: | 
|  | printer->Print("install-location:'internalOnly'\n"); | 
|  | break; | 
|  | case 2: | 
|  | printer->Print("install-location:'preferExternal'\n"); | 
|  | break; | 
|  | default: | 
|  | break; | 
|  | } | 
|  | } | 
|  | } | 
|  | }; | 
|  |  | 
|  | /** Represents <application> elements. **/ | 
|  | class Application : public ManifestExtractor::Element { | 
|  | public: | 
|  | Application() = default; | 
|  | std::string label; | 
|  | std::string icon; | 
|  | std::string banner; | 
|  | int32_t is_game; | 
|  | int32_t debuggable; | 
|  | int32_t test_only; | 
|  | bool has_multi_arch; | 
|  |  | 
|  | /** Mapping from locales to app names. */ | 
|  | std::map<std::string, std::string> locale_labels; | 
|  |  | 
|  | /** Mapping from densities to app icons. */ | 
|  | std::map<uint16_t, std::string> density_icons; | 
|  |  | 
|  | void Extract(xml::Element* element) override { | 
|  | label = GetAttributeStringDefault(FindAttribute(element, LABEL_ATTR), ""); | 
|  | icon = GetAttributeStringDefault(FindAttribute(element, ICON_ATTR), ""); | 
|  | test_only = GetAttributeIntegerDefault(FindAttribute(element, TEST_ONLY_ATTR), 0); | 
|  | banner = GetAttributeStringDefault(FindAttribute(element, BANNER_ATTR), ""); | 
|  | is_game = GetAttributeIntegerDefault(FindAttribute(element, ISGAME_ATTR), 0); | 
|  | debuggable = GetAttributeIntegerDefault(FindAttribute(element, DEBUGGABLE_ATTR), 0); | 
|  |  | 
|  | // We must search by name because the multiArch flag hasn't been API | 
|  | // frozen yet. | 
|  | has_multi_arch = (GetAttributeIntegerDefault( | 
|  | FindAttribute(element, kAndroidNamespace, "multiArch"), 0) != 0); | 
|  |  | 
|  | // Retrieve the app names for every locale the app supports | 
|  | auto attr = FindAttribute(element, LABEL_ATTR); | 
|  | for (auto& config : extractor()->locales()) { | 
|  | if (auto label = GetAttributeString(attr, config.second)) { | 
|  | if (label) { | 
|  | locale_labels.insert(std::make_pair(config.first, *label)); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | // Retrieve the icons for the densities the app supports | 
|  | attr = FindAttribute(element, ICON_ATTR); | 
|  | for (auto& config : extractor()->densities()) { | 
|  | if (auto resource = GetAttributeString(attr, config.second)) { | 
|  | if (resource) { | 
|  | density_icons.insert(std::make_pair(config.first, *resource)); | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | void Print(text::Printer* printer) override { | 
|  | // Print the labels for every locale | 
|  | for (auto p : locale_labels) { | 
|  | if (p.first.empty()) { | 
|  | printer->Print(StringPrintf("application-label:'%s'\n", | 
|  | android::ResTable::normalizeForOutput(p.second.data()) | 
|  | .c_str())); | 
|  | } else { | 
|  | printer->Print(StringPrintf("application-label-%s:'%s'\n", p.first.data(), | 
|  | android::ResTable::normalizeForOutput(p.second.data()) | 
|  | .c_str())); | 
|  | } | 
|  | } | 
|  |  | 
|  | // Print the icon paths for every density | 
|  | for (auto p : density_icons) { | 
|  | printer->Print(StringPrintf("application-icon-%d:'%s'\n", p.first, p.second.data())); | 
|  | } | 
|  |  | 
|  | // Print the application info | 
|  | printer->Print(StringPrintf("application: label='%s' ", | 
|  | android::ResTable::normalizeForOutput(label.data()).c_str())); | 
|  | printer->Print(StringPrintf("icon='%s'", icon.data())); | 
|  | if (!banner.empty()) { | 
|  | printer->Print(StringPrintf(" banner='%s'", banner.data())); | 
|  | } | 
|  | printer->Print("\n"); | 
|  |  | 
|  | if (test_only != 0) { | 
|  | printer->Print(StringPrintf("testOnly='%d'\n", test_only)); | 
|  | } | 
|  | if (is_game != 0) { | 
|  | printer->Print("application-isGame\n"); | 
|  | } | 
|  | if (debuggable != 0) { | 
|  | printer->Print("application-debuggable\n"); | 
|  | } | 
|  | } | 
|  |  | 
|  | void ToProto(pb::Badging* out_badging) override { | 
|  | auto application = out_badging->mutable_application(); | 
|  | application->set_label(android::ResTable::normalizeForOutput(label.data())); | 
|  | application->set_icon(icon); | 
|  | application->set_banner(banner); | 
|  | application->set_test_only(test_only != 0); | 
|  | application->set_game(is_game != 0); | 
|  | application->set_debuggable(debuggable != 0); | 
|  |  | 
|  | auto out_locale_labels = application->mutable_locale_labels(); | 
|  | for (auto& p : locale_labels) { | 
|  | if (!p.first.empty()) { | 
|  | (*out_locale_labels)[p.first] = p.second; | 
|  | } | 
|  | } | 
|  | auto out_density_icons = application->mutable_density_icons(); | 
|  | for (auto& p : density_icons) { | 
|  | (*out_density_icons)[p.first] = p.second; | 
|  | } | 
|  | } | 
|  | }; | 
|  |  | 
|  | /** Represents <uses-sdk> elements. **/ | 
|  | class UsesSdkBadging : public ManifestExtractor::Element { | 
|  | public: | 
|  | UsesSdkBadging() = default; | 
|  | const int32_t* min_sdk = nullptr; | 
|  | const std::string* min_sdk_name = nullptr; | 
|  | const int32_t* max_sdk = nullptr; | 
|  | const int32_t* target_sdk = nullptr; | 
|  | const std::string* target_sdk_name = nullptr; | 
|  |  | 
|  | void Extract(xml::Element* element) override { | 
|  | min_sdk = GetAttributeInteger(FindAttribute(element, MIN_SDK_VERSION_ATTR)); | 
|  | min_sdk_name = GetAttributeString(FindAttribute(element, MIN_SDK_VERSION_ATTR)); | 
|  | max_sdk = GetAttributeInteger(FindAttribute(element, MAX_SDK_VERSION_ATTR)); | 
|  | target_sdk = GetAttributeInteger(FindAttribute(element, TARGET_SDK_VERSION_ATTR)); | 
|  | target_sdk_name = GetAttributeString(FindAttribute(element, TARGET_SDK_VERSION_ATTR)); | 
|  |  | 
|  | // Detect the target sdk of the element | 
|  | if  ((min_sdk_name && *min_sdk_name == "Donut") | 
|  | || (target_sdk_name && *target_sdk_name == "Donut")) { | 
|  | extractor()->RaiseTargetSdk(SDK_DONUT); | 
|  | } | 
|  | if (min_sdk) { | 
|  | extractor()->RaiseTargetSdk(*min_sdk); | 
|  | } | 
|  | if (target_sdk) { | 
|  | extractor()->RaiseTargetSdk(*target_sdk); | 
|  | } else if (target_sdk_name) { | 
|  | extractor()->RaiseTargetSdk(SDK_CUR_DEVELOPMENT); | 
|  | } | 
|  | } | 
|  |  | 
|  | void Print(text::Printer* printer) override { | 
|  | if (min_sdk) { | 
|  | printer->Print(StringPrintf("sdkVersion:'%d'\n", *min_sdk)); | 
|  | } else if (min_sdk_name) { | 
|  | printer->Print(StringPrintf("sdkVersion:'%s'\n", min_sdk_name->data())); | 
|  | } | 
|  | if (max_sdk) { | 
|  | printer->Print(StringPrintf("maxSdkVersion:'%d'\n", *max_sdk)); | 
|  | } | 
|  | if (target_sdk) { | 
|  | printer->Print(StringPrintf("targetSdkVersion:'%d'\n", *target_sdk)); | 
|  | } else if (target_sdk_name) { | 
|  | printer->Print(StringPrintf("targetSdkVersion:'%s'\n", target_sdk_name->data())); | 
|  | } | 
|  | } | 
|  |  | 
|  | void ToProto(pb::Badging* out_badging) override { | 
|  | auto out_sdks = out_badging->mutable_uses_sdk(); | 
|  | if (min_sdk) { | 
|  | out_sdks->set_min_sdk_version(*min_sdk); | 
|  | } else if (min_sdk_name) { | 
|  | out_sdks->set_min_sdk_version_name(*min_sdk_name); | 
|  | } | 
|  | if (max_sdk) { | 
|  | out_sdks->set_max_sdk_version(*max_sdk); | 
|  | } | 
|  | if (target_sdk) { | 
|  | out_sdks->set_target_sdk_version(*target_sdk); | 
|  | } else if (target_sdk_name) { | 
|  | out_sdks->set_target_sdk_version_name(*target_sdk_name); | 
|  | } | 
|  | } | 
|  | }; | 
|  |  | 
|  | /** Represents <uses-configuration> elements. **/ | 
|  | class UsesConfiguarion : public ManifestExtractor::Element { | 
|  | public: | 
|  | UsesConfiguarion() = default; | 
|  | int32_t req_touch_screen = 0; | 
|  | int32_t req_keyboard_type = 0; | 
|  | int32_t req_hard_keyboard = 0; | 
|  | int32_t req_navigation = 0; | 
|  | int32_t req_five_way_nav = 0; | 
|  |  | 
|  | void Extract(xml::Element* element) override { | 
|  | req_touch_screen = GetAttributeIntegerDefault( | 
|  | FindAttribute(element, REQ_TOUCH_SCREEN_ATTR), 0); | 
|  | req_keyboard_type = GetAttributeIntegerDefault( | 
|  | FindAttribute(element, REQ_KEYBOARD_TYPE_ATTR), 0); | 
|  | req_hard_keyboard = GetAttributeIntegerDefault( | 
|  | FindAttribute(element, REQ_HARD_KEYBOARD_ATTR), 0); | 
|  | req_navigation = GetAttributeIntegerDefault( | 
|  | FindAttribute(element, REQ_NAVIGATION_ATTR), 0); | 
|  | req_five_way_nav = GetAttributeIntegerDefault( | 
|  | FindAttribute(element, REQ_FIVE_WAY_NAV_ATTR), 0); | 
|  | } | 
|  |  | 
|  | void Print(text::Printer* printer) override { | 
|  | printer->Print("uses-configuration:"); | 
|  | if (req_touch_screen != 0) { | 
|  | printer->Print(StringPrintf(" reqTouchScreen='%d'", req_touch_screen)); | 
|  | } | 
|  | if (req_keyboard_type != 0) { | 
|  | printer->Print(StringPrintf(" reqKeyboardType='%d'", req_keyboard_type)); | 
|  | } | 
|  | if (req_hard_keyboard != 0) { | 
|  | printer->Print(StringPrintf(" reqHardKeyboard='%d'", req_hard_keyboard)); | 
|  | } | 
|  | if (req_navigation != 0) { | 
|  | printer->Print(StringPrintf(" reqNavigation='%d'", req_navigation)); | 
|  | } | 
|  | if (req_five_way_nav != 0) { | 
|  | printer->Print(StringPrintf(" reqFiveWayNav='%d'", req_five_way_nav)); | 
|  | } | 
|  | printer->Print("\n"); | 
|  | } | 
|  |  | 
|  | void ToProto(pb::Badging* out_badging) override { | 
|  | auto out_configuration = out_badging->mutable_uses_configuration(); | 
|  | out_configuration->set_req_touch_screen(req_touch_screen); | 
|  | out_configuration->set_req_keyboard_type(req_keyboard_type); | 
|  | out_configuration->set_req_hard_keyboard(req_hard_keyboard); | 
|  | out_configuration->set_req_navigation(req_navigation); | 
|  | out_configuration->set_req_five_way_nav(req_five_way_nav); | 
|  | } | 
|  | }; | 
|  |  | 
|  | /** Represents <supports-screen> elements. **/ | 
|  | class SupportsScreen : public ManifestExtractor::Element { | 
|  | public: | 
|  | SupportsScreen() = default; | 
|  | int32_t small_screen = 1; | 
|  | int32_t normal_screen = 1; | 
|  | int32_t large_screen  = 1; | 
|  | int32_t xlarge_screen = 1; | 
|  | int32_t any_density = 1; | 
|  | int32_t requires_smallest_width_dp = 0; | 
|  | int32_t compatible_width_limit_dp = 0; | 
|  | int32_t largest_width_limit_dp = 0; | 
|  |  | 
|  | void Extract(xml::Element* element) override { | 
|  | small_screen = GetAttributeIntegerDefault(FindAttribute(element, SMALL_SCREEN_ATTR), 1); | 
|  | normal_screen = GetAttributeIntegerDefault(FindAttribute(element, NORMAL_SCREEN_ATTR), 1); | 
|  | large_screen = GetAttributeIntegerDefault(FindAttribute(element, LARGE_SCREEN_ATTR), 1); | 
|  | xlarge_screen = GetAttributeIntegerDefault(FindAttribute(element, XLARGE_SCREEN_ATTR), 1); | 
|  | any_density = GetAttributeIntegerDefault(FindAttribute(element, ANY_DENSITY_ATTR), 1); | 
|  |  | 
|  | requires_smallest_width_dp = GetAttributeIntegerDefault( | 
|  | FindAttribute(element, REQUIRES_SMALLEST_WIDTH_DP_ATTR), 0); | 
|  | compatible_width_limit_dp = GetAttributeIntegerDefault( | 
|  | FindAttribute(element, COMPATIBLE_WIDTH_LIMIT_DP_ATTR), 0); | 
|  | largest_width_limit_dp = GetAttributeIntegerDefault( | 
|  | FindAttribute(element, LARGEST_WIDTH_LIMIT_DP_ATTR), 0); | 
|  |  | 
|  | // For modern apps, if screen size buckets haven't been specified | 
|  | // but the new width ranges have, then infer the buckets from them. | 
|  | if (small_screen > 0 && normal_screen > 0 && large_screen > 0 && xlarge_screen > 0 | 
|  | && requires_smallest_width_dp > 0) { | 
|  | int32_t compat_width = (compatible_width_limit_dp > 0) ? compatible_width_limit_dp | 
|  | : requires_smallest_width_dp; | 
|  | small_screen = (requires_smallest_width_dp <= 240 && compat_width >= 240) ? -1 : 0; | 
|  | normal_screen = (requires_smallest_width_dp <= 320 && compat_width >= 320) ? -1 : 0; | 
|  | large_screen = (requires_smallest_width_dp <= 480 && compat_width >= 480) ? -1 : 0; | 
|  | xlarge_screen = (requires_smallest_width_dp <= 720 && compat_width >= 720) ? -1 : 0; | 
|  | } | 
|  | } | 
|  |  | 
|  | void PrintScreens(text::Printer* printer, int32_t target_sdk) const { | 
|  | // Print the formatted screen info | 
|  | printer->Print("supports-screens:"); | 
|  | if (IsSmallScreenSupported(target_sdk)) { | 
|  | printer->Print(" 'small'"); | 
|  | } | 
|  | if (normal_screen != 0) { | 
|  | printer->Print(" 'normal'"); | 
|  | } | 
|  | if (IsLargeScreenSupported(target_sdk)) { | 
|  | printer->Print(" 'large'"); | 
|  | } | 
|  | if (IsXLargeScreenSupported(target_sdk)) { | 
|  | printer->Print(" 'xlarge'"); | 
|  | } | 
|  | printer->Print("\n"); | 
|  | printer->Print(StringPrintf("supports-any-density: '%s'\n", | 
|  | (IsAnyDensitySupported(target_sdk)) ? "true" : "false")); | 
|  | if (requires_smallest_width_dp > 0) { | 
|  | printer->Print(StringPrintf("requires-smallest-width:'%d'\n", requires_smallest_width_dp)); | 
|  | } | 
|  | if (compatible_width_limit_dp > 0) { | 
|  | printer->Print(StringPrintf("compatible-width-limit:'%d'\n", compatible_width_limit_dp)); | 
|  | } | 
|  | if (largest_width_limit_dp > 0) { | 
|  | printer->Print(StringPrintf("largest-width-limit:'%d'\n", largest_width_limit_dp)); | 
|  | } | 
|  | } | 
|  |  | 
|  | void ToProtoScreens(pb::Badging* out_badging, int32_t target_sdk) const { | 
|  | auto supports_screen = out_badging->mutable_supports_screen(); | 
|  | if (IsSmallScreenSupported(target_sdk)) { | 
|  | supports_screen->add_screens(pb::SupportsScreen_ScreenType_SMALL); | 
|  | } | 
|  | if (normal_screen != 0) { | 
|  | supports_screen->add_screens(pb::SupportsScreen_ScreenType_NORMAL); | 
|  | } | 
|  | if (IsLargeScreenSupported(target_sdk)) { | 
|  | supports_screen->add_screens(pb::SupportsScreen_ScreenType_LARGE); | 
|  | } | 
|  | if (IsXLargeScreenSupported(target_sdk)) { | 
|  | supports_screen->add_screens(pb::SupportsScreen_ScreenType_XLARGE); | 
|  | } | 
|  | supports_screen->set_supports_any_densities(IsAnyDensitySupported(target_sdk)); | 
|  | supports_screen->set_requires_smallest_width_dp(requires_smallest_width_dp); | 
|  | supports_screen->set_compatible_width_limit_dp(compatible_width_limit_dp); | 
|  | supports_screen->set_largest_width_limit_dp(largest_width_limit_dp); | 
|  | } | 
|  |  | 
|  | private: | 
|  | // Determine default values for any unspecified screen sizes, | 
|  | // based on the target SDK of the package.  As of 4 (donut) | 
|  | // the screen size support was introduced, so all default to | 
|  | // enabled. | 
|  | bool IsSmallScreenSupported(int32_t target_sdk) const { | 
|  | if (small_screen > 0) { | 
|  | return target_sdk >= SDK_DONUT; | 
|  | } | 
|  | return small_screen != 0; | 
|  | } | 
|  |  | 
|  | bool IsLargeScreenSupported(int32_t target_sdk) const { | 
|  | if (large_screen > 0) { | 
|  | return target_sdk >= SDK_DONUT; | 
|  | } | 
|  | return large_screen != 0; | 
|  | } | 
|  |  | 
|  | bool IsXLargeScreenSupported(int32_t target_sdk) const { | 
|  | if (xlarge_screen > 0) { | 
|  | return target_sdk >= SDK_GINGERBREAD; | 
|  | } | 
|  | return xlarge_screen != 0; | 
|  | } | 
|  |  | 
|  | bool IsAnyDensitySupported(int32_t target_sdk) const { | 
|  | if (any_density > 0) { | 
|  | return target_sdk >= SDK_DONUT || requires_smallest_width_dp > 0 || | 
|  | compatible_width_limit_dp > 0; | 
|  | } | 
|  | return any_density != 0; | 
|  | } | 
|  | }; | 
|  |  | 
|  | /** Represents <feature-group> elements. **/ | 
|  | class FeatureGroup : public ManifestExtractor::Element { | 
|  | public: | 
|  | FeatureGroup() = default; | 
|  | std::string label; | 
|  | int32_t open_gles_version = 0; | 
|  |  | 
|  | void Extract(xml::Element* element) override { | 
|  | label = GetAttributeStringDefault(FindAttribute(element, LABEL_ATTR), ""); | 
|  | } | 
|  |  | 
|  | virtual void PrintGroup(text::Printer* printer) { | 
|  | printer->Print(StringPrintf("feature-group: label='%s'\n", label.data())); | 
|  | if (open_gles_version > 0) { | 
|  | printer->Print(StringPrintf("  uses-gl-es: '0x%x'\n", open_gles_version)); | 
|  | } | 
|  |  | 
|  | for (auto feature : features_) { | 
|  | printer->Print(StringPrintf("  uses-feature%s: name='%s'", | 
|  | (feature.second.required ? "" : "-not-required"), | 
|  | feature.first.data())); | 
|  | if (feature.second.version > 0) { | 
|  | printer->Print(StringPrintf(" version='%d'", feature.second.version)); | 
|  | } | 
|  | printer->Print("\n"); | 
|  | } | 
|  | } | 
|  |  | 
|  | virtual void GroupToProto(pb::Badging* out_badging) { | 
|  | auto feature_group = out_badging->add_feature_groups(); | 
|  | feature_group->set_label(label); | 
|  | feature_group->set_open_gles_version(open_gles_version); | 
|  | for (auto& feature : features_) { | 
|  | auto out_feature = feature_group->add_features(); | 
|  | out_feature->set_name(feature.first); | 
|  | out_feature->set_required(feature.second.required); | 
|  | out_feature->set_version(feature.second.version); | 
|  | } | 
|  | } | 
|  |  | 
|  | /** Adds a feature to the feature group. */ | 
|  | void AddFeature(const std::string& name, bool required = true, int32_t version = -1) { | 
|  | features_.insert(std::make_pair(name, Feature{ required, version })); | 
|  | if (required) { | 
|  | if (name == "android.hardware.camera.autofocus" || | 
|  | name == "android.hardware.camera.flash") { | 
|  | AddFeature("android.hardware.camera", true); | 
|  | } else if (name == "android.hardware.location.gps" || | 
|  | name == "android.hardware.location.network") { | 
|  | AddFeature("android.hardware.location", true); | 
|  | } else if (name == "android.hardware.faketouch.multitouch") { | 
|  | AddFeature("android.hardware.faketouch", true); | 
|  | } else if (name == "android.hardware.faketouch.multitouch.distinct" || | 
|  | name == "android.hardware.faketouch.multitouch.jazzhands") { | 
|  | AddFeature("android.hardware.faketouch.multitouch", true); | 
|  | AddFeature("android.hardware.faketouch", true); | 
|  | } else if (name == "android.hardware.touchscreen.multitouch") { | 
|  | AddFeature("android.hardware.touchscreen", true); | 
|  | } else if (name == "android.hardware.touchscreen.multitouch.distinct" || | 
|  | name == "android.hardware.touchscreen.multitouch.jazzhands") { | 
|  | AddFeature("android.hardware.touchscreen.multitouch", true); | 
|  | AddFeature("android.hardware.touchscreen", true); | 
|  | } else if (name == "android.hardware.opengles.aep") { | 
|  | const int kOpenGLESVersion31 = 0x00030001; | 
|  | if (kOpenGLESVersion31 > open_gles_version) { | 
|  | open_gles_version = kOpenGLESVersion31; | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | /** Returns true if the feature group has the given feature. */ | 
|  | virtual bool HasFeature(const std::string& name) { | 
|  | return features_.find(name) != features_.end(); | 
|  | } | 
|  |  | 
|  | /** Merges the features of another feature group into this group. */ | 
|  | void Merge(FeatureGroup* group) { | 
|  | open_gles_version = std::max(open_gles_version, group->open_gles_version); | 
|  | for (auto& feature : group->features_) { | 
|  | features_.insert(feature); | 
|  | } | 
|  | } | 
|  |  | 
|  | protected: | 
|  | struct Feature { | 
|  | public: | 
|  | bool required = false; | 
|  | int32_t version = -1; | 
|  | }; | 
|  |  | 
|  | /* Mapping of feature names to their properties. */ | 
|  | std::map<std::string, Feature> features_; | 
|  | }; | 
|  |  | 
|  | /** | 
|  | * Represents the default feature group for the application if no <feature-group> elements are | 
|  | * present in the manifest. | 
|  | **/ | 
|  | class CommonFeatureGroup : public FeatureGroup { | 
|  | public: | 
|  | CommonFeatureGroup() = default; | 
|  | void PrintGroup(text::Printer* printer) override { | 
|  | FeatureGroup::PrintGroup(printer); | 
|  |  | 
|  | // Also print the implied features | 
|  | for (auto feature : implied_features_) { | 
|  | if (features_.find(feature.first) == features_.end()) { | 
|  | const char* sdk23 = feature.second.implied_from_sdk_k23 ? "-sdk-23" : ""; | 
|  | printer->Print(StringPrintf("  uses-feature%s: name='%s'\n", sdk23, feature.first.data())); | 
|  | printer->Print(StringPrintf("  uses-implied-feature%s: name='%s' reason='", sdk23, | 
|  | feature.first.data())); | 
|  |  | 
|  | // Print the reasons as a sentence | 
|  | size_t count = 0; | 
|  | for (auto reason : feature.second.reasons) { | 
|  | printer->Print(reason); | 
|  | if (count + 2 < feature.second.reasons.size()) { | 
|  | printer->Print(", "); | 
|  | } else if (count + 1 < feature.second.reasons.size()) { | 
|  | printer->Print(", and "); | 
|  | } | 
|  | count++; | 
|  | } | 
|  | printer->Print("'\n"); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | virtual void GroupToProto(pb::Badging* out_badging) override { | 
|  | FeatureGroup::GroupToProto(out_badging); | 
|  | auto feature_group = | 
|  | out_badging->mutable_feature_groups(out_badging->feature_groups_size() - 1); | 
|  | for (auto& feature : implied_features_) { | 
|  | if (features_.find(feature.first) == features_.end()) { | 
|  | auto out_feature = feature_group->add_features(); | 
|  | out_feature->set_name(feature.first); | 
|  | auto implied_data = out_feature->mutable_implied_data(); | 
|  | implied_data->set_from_sdk_23_permission(feature.second.implied_from_sdk_k23); | 
|  | for (auto& reason : feature.second.reasons) { | 
|  | implied_data->add_reasons(reason); | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | /** Returns true if the feature group has the given feature. */ | 
|  | bool HasFeature(const std::string& name) override { | 
|  | return FeatureGroup::HasFeature(name) | 
|  | || implied_features_.find(name) != implied_features_.end(); | 
|  | } | 
|  |  | 
|  | /** Adds a feature to a set of implied features not explicitly requested in the manifest. */ | 
|  | void addImpliedFeature(const std::string& name, const std::string& reason, bool sdk23 = false) { | 
|  | auto entry = implied_features_.find(name); | 
|  | if (entry == implied_features_.end()) { | 
|  | implied_features_.insert(std::make_pair(name, ImpliedFeature(sdk23))); | 
|  | entry = implied_features_.find(name); | 
|  | } | 
|  |  | 
|  | // A non-sdk 23 implied feature takes precedence. | 
|  | if (entry->second.implied_from_sdk_k23 && !sdk23) { | 
|  | entry->second.implied_from_sdk_k23 = false; | 
|  | } | 
|  |  | 
|  | entry->second.reasons.insert(reason); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Adds a feature to a set of implied features for all features that are implied by the presence | 
|  | * of the permission. | 
|  | **/ | 
|  | void addImpliedFeaturesForPermission(int32_t targetSdk, const std::string& name, bool sdk23) { | 
|  | if (name == "android.permission.CAMERA") { | 
|  | addImpliedFeature("android.hardware.camera", | 
|  | StringPrintf("requested %s permission", name.data()), | 
|  | sdk23); | 
|  |  | 
|  | } else if (name == "android.permission.ACCESS_FINE_LOCATION") { | 
|  | if (targetSdk < SDK_LOLLIPOP) { | 
|  | addImpliedFeature("android.hardware.location.gps", | 
|  | StringPrintf("requested %s permission", name.data()), | 
|  | sdk23); | 
|  | addImpliedFeature("android.hardware.location.gps", | 
|  | StringPrintf("targetSdkVersion < %d", SDK_LOLLIPOP), | 
|  | sdk23); | 
|  | } | 
|  | addImpliedFeature("android.hardware.location", | 
|  | StringPrintf("requested %s permission", name.data()), | 
|  | sdk23); | 
|  |  | 
|  | } else if (name == "android.permission.ACCESS_COARSE_LOCATION") { | 
|  | if (targetSdk < SDK_LOLLIPOP) { | 
|  | addImpliedFeature("android.hardware.location.network", | 
|  | StringPrintf("requested %s permission", name.data()), | 
|  | sdk23); | 
|  | addImpliedFeature("android.hardware.location.network", | 
|  | StringPrintf("targetSdkVersion < %d", SDK_LOLLIPOP), | 
|  | sdk23); | 
|  | } | 
|  | addImpliedFeature("android.hardware.location", | 
|  | StringPrintf("requested %s permission", name.data()), | 
|  | sdk23); | 
|  |  | 
|  | } else if (name == "android.permission.ACCESS_MOCK_LOCATION" || | 
|  | name == "android.permission.ACCESS_LOCATION_EXTRA_COMMANDS" || | 
|  | name == "android.permission.INSTALL_LOCATION_PROVIDER") { | 
|  | addImpliedFeature("android.hardware.location", | 
|  | StringPrintf("requested %s permission", name.data()), | 
|  | sdk23); | 
|  |  | 
|  | } else if (name == "android.permission.BLUETOOTH" || | 
|  | name == "android.permission.BLUETOOTH_ADMIN") { | 
|  | if (targetSdk > SDK_DONUT) { | 
|  | addImpliedFeature("android.hardware.bluetooth", | 
|  | StringPrintf("requested %s permission", name.data()), | 
|  | sdk23); | 
|  | addImpliedFeature("android.hardware.bluetooth", | 
|  | StringPrintf("targetSdkVersion > %d", SDK_DONUT), | 
|  | sdk23); | 
|  | } | 
|  |  | 
|  | } else if (name == "android.permission.RECORD_AUDIO") { | 
|  | addImpliedFeature("android.hardware.microphone", | 
|  | StringPrintf("requested %s permission", name.data()), | 
|  | sdk23); | 
|  |  | 
|  | } else if (name == "android.permission.ACCESS_WIFI_STATE" || | 
|  | name == "android.permission.CHANGE_WIFI_STATE" || | 
|  | name == "android.permission.CHANGE_WIFI_MULTICAST_STATE") { | 
|  | addImpliedFeature("android.hardware.wifi", | 
|  | StringPrintf("requested %s permission", name.data()), | 
|  | sdk23); | 
|  |  | 
|  | } else if (name == "android.permission.CALL_PHONE" || | 
|  | name == "android.permission.CALL_PRIVILEGED" || | 
|  | name == "android.permission.MODIFY_PHONE_STATE" || | 
|  | name == "android.permission.PROCESS_OUTGOING_CALLS" || | 
|  | name == "android.permission.READ_SMS" || | 
|  | name == "android.permission.RECEIVE_SMS" || | 
|  | name == "android.permission.RECEIVE_MMS" || | 
|  | name == "android.permission.RECEIVE_WAP_PUSH" || | 
|  | name == "android.permission.SEND_SMS" || | 
|  | name == "android.permission.WRITE_APN_SETTINGS" || | 
|  | name == "android.permission.WRITE_SMS") { | 
|  | addImpliedFeature("android.hardware.telephony", | 
|  | "requested a telephony permission", | 
|  | sdk23); | 
|  | } | 
|  | } | 
|  |  | 
|  | private: | 
|  | /** | 
|  | * Represents a feature that has been automatically added due to a pre-requisite or for some | 
|  | * other reason. | 
|  | */ | 
|  | struct ImpliedFeature { | 
|  | explicit ImpliedFeature(bool sdk23 = false) : implied_from_sdk_k23(sdk23) {} | 
|  |  | 
|  | /** List of human-readable reasons for why this feature was implied. */ | 
|  | std::set<std::string> reasons; | 
|  |  | 
|  | // Was this implied by a permission from SDK 23 (<uses-permission-sdk-23 />) | 
|  | bool implied_from_sdk_k23; | 
|  | }; | 
|  |  | 
|  | /* Mapping of implied feature names to their properties. */ | 
|  | std::map<std::string, ImpliedFeature> implied_features_; | 
|  | }; | 
|  |  | 
|  | /** Represents <uses-feature> elements. **/ | 
|  | class UsesFeature : public ManifestExtractor::Element { | 
|  | public: | 
|  | UsesFeature() = default; | 
|  | void Extract(xml::Element* element) override { | 
|  | const std::string* name = GetAttributeString(FindAttribute(element, NAME_ATTR)); | 
|  | int32_t* gl = GetAttributeInteger(FindAttribute(element, GL_ES_VERSION_ATTR)); | 
|  | bool required = GetAttributeIntegerDefault( | 
|  | FindAttribute(element, REQUIRED_ATTR), true) != 0; | 
|  | int32_t version = GetAttributeIntegerDefault( | 
|  | FindAttribute(element, kAndroidNamespace, "version"), 0); | 
|  |  | 
|  | // Add the feature to the parent feature group element if one exists; otherwise, add it to the | 
|  | // common feature group | 
|  | FeatureGroup* feature_group = ElementCast<FeatureGroup>(extractor()->parent_stack()[0]); | 
|  | if (!feature_group) { | 
|  | feature_group = extractor()->common_feature_group(); | 
|  | } else { | 
|  | // All features in side of <feature-group> elements are required. | 
|  | required = true; | 
|  | } | 
|  |  | 
|  | if (name) { | 
|  | feature_group->AddFeature(*name, required, version); | 
|  | } else if (gl) { | 
|  | feature_group->open_gles_version = std::max(feature_group->open_gles_version, *gl); | 
|  | } | 
|  | } | 
|  | }; | 
|  |  | 
|  | /** Represents <uses-permission> elements. **/ | 
|  | class UsesPermission : public ManifestExtractor::Element { | 
|  | public: | 
|  | UsesPermission() = default; | 
|  | bool implied; | 
|  | std::string name; | 
|  | std::vector<std::string> requiredFeatures; | 
|  | std::vector<std::string> requiredNotFeatures; | 
|  | int32_t required = true; | 
|  | int32_t maxSdkVersion = -1; | 
|  | int32_t usesPermissionFlags = 0; | 
|  | std::string impliedReason; | 
|  |  | 
|  | void Extract(xml::Element* element) override { | 
|  | name = GetAttributeStringDefault(FindAttribute(element, NAME_ATTR), ""); | 
|  | std::string feature = | 
|  | GetAttributeStringDefault(FindAttribute(element, REQUIRED_FEATURE_ATTR), ""); | 
|  | if (!feature.empty()) { | 
|  | requiredFeatures.push_back(feature); | 
|  | } | 
|  | feature = GetAttributeStringDefault(FindAttribute(element, REQUIRED_NOT_FEATURE_ATTR), ""); | 
|  | if (!feature.empty()) { | 
|  | requiredNotFeatures.push_back(feature); | 
|  | } | 
|  |  | 
|  | required = GetAttributeIntegerDefault(FindAttribute(element, REQUIRED_ATTR), 1); | 
|  | maxSdkVersion = GetAttributeIntegerDefault( | 
|  | FindAttribute(element, MAX_SDK_VERSION_ATTR), -1); | 
|  | usesPermissionFlags = GetAttributeIntegerDefault( | 
|  | FindAttribute(element, USES_PERMISSION_FLAGS_ATTR), 0); | 
|  |  | 
|  | if (!name.empty()) { | 
|  | CommonFeatureGroup* common = extractor()->common_feature_group(); | 
|  | common->addImpliedFeaturesForPermission(extractor()->target_sdk(), name, false); | 
|  | } | 
|  | } | 
|  |  | 
|  | void Print(text::Printer* printer) override { | 
|  | if (!name.empty()) { | 
|  | printer->Print(StringPrintf("uses-permission: name='%s'", name.data())); | 
|  | if (maxSdkVersion >= 0) { | 
|  | printer->Print(StringPrintf(" maxSdkVersion='%d'", maxSdkVersion)); | 
|  | } | 
|  | if ((usesPermissionFlags & kNeverForLocation) != 0) { | 
|  | printer->Print(StringPrintf(" usesPermissionFlags='neverForLocation'")); | 
|  | } | 
|  | printer->Print("\n"); | 
|  | for (const std::string& requiredFeature : requiredFeatures) { | 
|  | printer->Print(StringPrintf("  required-feature='%s'\n", requiredFeature.data())); | 
|  | } | 
|  | for (const std::string& requiredNotFeature : requiredNotFeatures) { | 
|  | printer->Print(StringPrintf("  required-not-feature='%s'\n", requiredNotFeature.data())); | 
|  | } | 
|  | if (required == 0) { | 
|  | printer->Print(StringPrintf("optional-permission: name='%s'", name.data())); | 
|  | if (maxSdkVersion >= 0) { | 
|  | printer->Print(StringPrintf(" maxSdkVersion='%d'", maxSdkVersion)); | 
|  | } | 
|  | if ((usesPermissionFlags & kNeverForLocation) != 0) { | 
|  | printer->Print(StringPrintf(" usesPermissionFlags='neverForLocation'")); | 
|  | } | 
|  | printer->Print("\n"); | 
|  | } | 
|  | } | 
|  | if (implied) { | 
|  | printer->Print(StringPrintf("uses-implied-permission: name='%s'", name.data())); | 
|  | if (maxSdkVersion >= 0) { | 
|  | printer->Print(StringPrintf(" maxSdkVersion='%d'", maxSdkVersion)); | 
|  | } | 
|  | if ((usesPermissionFlags & kNeverForLocation) != 0) { | 
|  | printer->Print(StringPrintf(" usesPermissionFlags='neverForLocation'")); | 
|  | } | 
|  | printer->Print(StringPrintf(" reason='%s'\n", impliedReason.data())); | 
|  | } | 
|  | } | 
|  |  | 
|  | void ToProto(pb::Badging* out_badging) override { | 
|  | if (!name.empty()) { | 
|  | auto permission = out_badging->add_uses_permissions(); | 
|  | permission->set_name(name); | 
|  | if (maxSdkVersion > 0) { | 
|  | permission->set_max_sdk_version(maxSdkVersion); | 
|  | } | 
|  | if ((usesPermissionFlags & kNeverForLocation) != 0) { | 
|  | permission->mutable_permission_flags()->set_never_for_location(true); | 
|  | } | 
|  | for (auto& requiredFeature : requiredFeatures) { | 
|  | permission->add_required_features(requiredFeature); | 
|  | } | 
|  | for (auto& requiredNotFeature : requiredNotFeatures) { | 
|  | permission->add_required_not_features(requiredNotFeature); | 
|  | } | 
|  | permission->set_required(required != 0); | 
|  | permission->set_implied(implied); | 
|  | } | 
|  | } | 
|  | }; | 
|  |  | 
|  | /** Represents <required-feature> elements. **/ | 
|  | class RequiredFeature : public ManifestExtractor::Element { | 
|  | public: | 
|  | RequiredFeature() = default; | 
|  | std::string name; | 
|  |  | 
|  | void Extract(xml::Element* element) override { | 
|  | name = GetAttributeStringDefault(FindAttribute(element, NAME_ATTR), ""); | 
|  | auto parent_stack = extractor()->parent_stack(); | 
|  | if (!name.empty() && ElementCast<UsesPermission>(parent_stack[0])) { | 
|  | UsesPermission* uses_permission = ElementCast<UsesPermission>(parent_stack[0]); | 
|  | uses_permission->requiredFeatures.push_back(name); | 
|  | } | 
|  | } | 
|  | }; | 
|  |  | 
|  | /** Represents <required-not-feature> elements. **/ | 
|  | class RequiredNotFeature : public ManifestExtractor::Element { | 
|  | public: | 
|  | RequiredNotFeature() = default; | 
|  | std::string name; | 
|  |  | 
|  | void Extract(xml::Element* element) override { | 
|  | name = GetAttributeStringDefault(FindAttribute(element, NAME_ATTR), ""); | 
|  | auto parent_stack = extractor()->parent_stack(); | 
|  | if (!name.empty() && ElementCast<UsesPermission>(parent_stack[0])) { | 
|  | UsesPermission* uses_permission = ElementCast<UsesPermission>(parent_stack[0]); | 
|  | uses_permission->requiredNotFeatures.push_back(name); | 
|  | } | 
|  | } | 
|  | }; | 
|  |  | 
|  | /** Represents <uses-permission-sdk-23> elements. **/ | 
|  | class UsesPermissionSdk23 : public ManifestExtractor::Element { | 
|  | public: | 
|  | UsesPermissionSdk23() = default; | 
|  | const std::string* name = nullptr; | 
|  | const int32_t* maxSdkVersion = nullptr; | 
|  |  | 
|  | void Extract(xml::Element* element) override { | 
|  | name = GetAttributeString(FindAttribute(element, NAME_ATTR)); | 
|  | maxSdkVersion = GetAttributeInteger(FindAttribute(element, MAX_SDK_VERSION_ATTR)); | 
|  |  | 
|  | if (name) { | 
|  | CommonFeatureGroup* common = extractor()->common_feature_group(); | 
|  | common->addImpliedFeaturesForPermission(extractor()->target_sdk(), *name, true); | 
|  | } | 
|  | } | 
|  |  | 
|  | void Print(text::Printer* printer) override { | 
|  | if (name) { | 
|  | printer->Print(StringPrintf("uses-permission-sdk-23: name='%s'", name->data())); | 
|  | if (maxSdkVersion) { | 
|  | printer->Print(StringPrintf(" maxSdkVersion='%d'", *maxSdkVersion)); | 
|  | } | 
|  | printer->Print("\n"); | 
|  | } | 
|  | } | 
|  |  | 
|  | void ToProto(pb::Badging* out_badging) override { | 
|  | if (name) { | 
|  | auto permission = out_badging->add_uses_permissions(); | 
|  | permission->set_sdk23_and_above(true); | 
|  | permission->set_name(*name); | 
|  | if (maxSdkVersion) { | 
|  | permission->set_max_sdk_version(*maxSdkVersion); | 
|  | } | 
|  | } | 
|  | } | 
|  | }; | 
|  |  | 
|  | /** Represents <permission> elements. These elements are only printing when dumping permissions. **/ | 
|  | class Permission : public ManifestExtractor::Element { | 
|  | public: | 
|  | Permission() = default; | 
|  | std::string name; | 
|  |  | 
|  | void Extract(xml::Element* element) override { | 
|  | name = GetAttributeStringDefault(FindAttribute(element, NAME_ATTR), ""); | 
|  | } | 
|  |  | 
|  | void Print(text::Printer* printer) override { | 
|  | if (extractor()->options_.only_permissions && !name.empty()) { | 
|  | printer->Print(StringPrintf("permission: %s\n", name.data())); | 
|  | } | 
|  | } | 
|  |  | 
|  | void ToProto(pb::Badging* out_badging) override { | 
|  | if (!name.empty()) { | 
|  | out_badging->add_permissions()->set_name(name); | 
|  | } | 
|  | } | 
|  | }; | 
|  |  | 
|  | /** Represents <activity> elements. **/ | 
|  | class Activity : public ManifestExtractor::Element { | 
|  | public: | 
|  | Activity() = default; | 
|  | std::string name; | 
|  | std::string icon; | 
|  | std::string label; | 
|  | std::string banner; | 
|  |  | 
|  | bool has_component_ = false; | 
|  | bool has_launcher_category = false; | 
|  | bool has_leanback_launcher_category = false; | 
|  | bool has_main_action = false; | 
|  |  | 
|  | void Extract(xml::Element* element) override { | 
|  | name = GetAttributeStringDefault(FindAttribute(element, NAME_ATTR), ""); | 
|  | label = GetAttributeStringDefault(FindAttribute(element, LABEL_ATTR), ""); | 
|  | icon = GetAttributeStringDefault(FindAttribute(element, ICON_ATTR), ""); | 
|  | banner = GetAttributeStringDefault(FindAttribute(element, BANNER_ATTR), ""); | 
|  |  | 
|  | // Retrieve the package name from the manifest | 
|  | std::string package; | 
|  | for (auto& parent : extractor()->parent_stack()) { | 
|  | if (auto manifest = ElementCast<Manifest>(parent)) { | 
|  | package = manifest->package; | 
|  | break; | 
|  | } | 
|  | } | 
|  |  | 
|  | // Fully qualify the activity name | 
|  | ssize_t idx = name.find('.'); | 
|  | if (idx == 0) { | 
|  | name = package + name; | 
|  | } else if (idx < 0) { | 
|  | name = package + "." + name; | 
|  | } | 
|  |  | 
|  | auto orientation = GetAttributeInteger(FindAttribute(element, SCREEN_ORIENTATION_ATTR)); | 
|  | if (orientation) { | 
|  | CommonFeatureGroup* common = extractor()->common_feature_group(); | 
|  | int orien = *orientation; | 
|  | if (orien == 0 || orien == 6 || orien == 8) { | 
|  | // Requests landscape, sensorLandscape, or reverseLandscape. | 
|  | common->addImpliedFeature("android.hardware.screen.landscape", | 
|  | "one or more activities have specified a landscape orientation", | 
|  | false); | 
|  | } else if (orien == 1 || orien == 7 || orien == 9) { | 
|  | // Requests portrait, sensorPortrait, or reversePortrait. | 
|  | common->addImpliedFeature("android.hardware.screen.portrait", | 
|  | "one or more activities have specified a portrait orientation", | 
|  | false); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | void Print(text::Printer* printer) override { | 
|  | // Print whether the activity has the HOME category and a the MAIN action | 
|  | if (has_main_action && has_launcher_category) { | 
|  | printer->Print("launchable-activity:"); | 
|  | if (!name.empty()) { | 
|  | printer->Print(StringPrintf(" name='%s' ", name.data())); | 
|  | } | 
|  | printer->Print(StringPrintf(" label='%s' icon='%s'\n", | 
|  | android::ResTable::normalizeForOutput(label.data()).c_str(), | 
|  | icon.data())); | 
|  | } | 
|  |  | 
|  | // Print wether the activity has the HOME category and a the MAIN action | 
|  | if (has_leanback_launcher_category) { | 
|  | printer->Print("leanback-launchable-activity:"); | 
|  | if (!name.empty()) { | 
|  | printer->Print(StringPrintf(" name='%s' ", name.data())); | 
|  | } | 
|  | printer->Print(StringPrintf(" label='%s' icon='%s' banner='%s'\n", | 
|  | android::ResTable::normalizeForOutput(label.data()).c_str(), | 
|  | icon.data(), banner.data())); | 
|  | } | 
|  | } | 
|  |  | 
|  | void ToProto(pb::Badging* out_badging) override { | 
|  | if (has_main_action && has_launcher_category) { | 
|  | auto activity = out_badging->mutable_launchable_activity(); | 
|  | activity->set_name(name); | 
|  | activity->set_label(android::ResTable::normalizeForOutput(label.data())); | 
|  | activity->set_icon(icon); | 
|  | } | 
|  | if (has_leanback_launcher_category) { | 
|  | auto activity = out_badging->mutable_leanback_launchable_activity(); | 
|  | activity->set_name(name); | 
|  | activity->set_label(android::ResTable::normalizeForOutput(label.data())); | 
|  | activity->set_icon(icon); | 
|  | activity->set_banner(banner); | 
|  | } | 
|  | } | 
|  | }; | 
|  |  | 
|  | /** Represents <intent-filter> elements. */ | 
|  | class IntentFilter : public ManifestExtractor::Element { | 
|  | public: | 
|  | IntentFilter() = default; | 
|  | }; | 
|  |  | 
|  | /** Represents <category> elements. */ | 
|  | class Category : public ManifestExtractor::Element { | 
|  | public: | 
|  | Category() = default; | 
|  | std::string component = ""; | 
|  |  | 
|  | void Extract(xml::Element* element) override { | 
|  | const std::string* category = GetAttributeString(FindAttribute(element, NAME_ATTR)); | 
|  |  | 
|  | auto parent_stack = extractor()->parent_stack(); | 
|  | if (category && ElementCast<IntentFilter>(parent_stack[0]) | 
|  | && ElementCast<Activity>(parent_stack[1])) { | 
|  | Activity* activity = ElementCast<Activity>(parent_stack[1]); | 
|  |  | 
|  | if (*category == "android.intent.category.LAUNCHER") { | 
|  | activity->has_launcher_category = true; | 
|  | } else if (*category == "android.intent.category.LEANBACK_LAUNCHER") { | 
|  | activity->has_leanback_launcher_category = true; | 
|  | } else if (*category == "android.intent.category.HOME") { | 
|  | component = "launcher"; | 
|  | } | 
|  | } | 
|  | } | 
|  | }; | 
|  |  | 
|  | /** | 
|  | * Represents <provider> elements. The elements may have an <intent-filter> which may have <action> | 
|  | * elements nested within. | 
|  | **/ | 
|  | class Provider : public ManifestExtractor::Element { | 
|  | public: | 
|  | Provider() = default; | 
|  | bool has_required_saf_attributes = false; | 
|  |  | 
|  | void Extract(xml::Element* element) override { | 
|  | const int32_t* exported = GetAttributeInteger(FindAttribute(element, EXPORTED_ATTR)); | 
|  | const int32_t* grant_uri_permissions = GetAttributeInteger( | 
|  | FindAttribute(element, GRANT_URI_PERMISSIONS_ATTR)); | 
|  | const std::string* permission = GetAttributeString( | 
|  | FindAttribute(element, PERMISSION_ATTR)); | 
|  |  | 
|  | has_required_saf_attributes = ((exported && *exported != 0) | 
|  | && (grant_uri_permissions && *grant_uri_permissions != 0) | 
|  | && (permission && *permission == "android.permission.MANAGE_DOCUMENTS")); | 
|  | } | 
|  | }; | 
|  |  | 
|  | /** Represents <receiver> elements. **/ | 
|  | class Receiver : public ManifestExtractor::Element { | 
|  | public: | 
|  | Receiver() = default; | 
|  | const std::string* permission = nullptr; | 
|  | bool has_component = false; | 
|  |  | 
|  | void Extract(xml::Element* element) override { | 
|  | permission = GetAttributeString(FindAttribute(element, PERMISSION_ATTR)); | 
|  | } | 
|  | }; | 
|  |  | 
|  | /**Represents <service> elements. **/ | 
|  | class Service : public ManifestExtractor::Element { | 
|  | public: | 
|  | Service() = default; | 
|  | const std::string* permission = nullptr; | 
|  | bool has_component = false; | 
|  |  | 
|  | void Extract(xml::Element* element) override { | 
|  | permission = GetAttributeString(FindAttribute(element, PERMISSION_ATTR)); | 
|  | } | 
|  | }; | 
|  |  | 
|  | /** Represents <uses-library> elements. **/ | 
|  | class UsesLibrary : public ManifestExtractor::Element { | 
|  | public: | 
|  | UsesLibrary() = default; | 
|  | std::string name; | 
|  | int required; | 
|  |  | 
|  | void Extract(xml::Element* element) override { | 
|  | auto parent_stack = extractor()->parent_stack(); | 
|  | if (parent_stack.size() > 0 && ElementCast<Application>(parent_stack[0])) { | 
|  | name = GetAttributeStringDefault(FindAttribute(element, NAME_ATTR), ""); | 
|  | required = GetAttributeIntegerDefault(FindAttribute(element, REQUIRED_ATTR), 1); | 
|  | } | 
|  | } | 
|  |  | 
|  | void Print(text::Printer* printer) override { | 
|  | if (!name.empty()) { | 
|  | printer->Print(StringPrintf("uses-library%s:'%s'\n", | 
|  | (required == 0) ? "-not-required" : "", name.data())); | 
|  | } | 
|  | } | 
|  |  | 
|  | void ToProto(pb::Badging* out_badging) override { | 
|  | if (!name.empty()) { | 
|  | auto uses_library = out_badging->add_uses_libraries(); | 
|  | uses_library->set_name(name); | 
|  | uses_library->set_required(required != 0); | 
|  | } | 
|  | } | 
|  | }; | 
|  |  | 
|  | /** Represents <static-library> elements. **/ | 
|  | class StaticLibrary : public ManifestExtractor::Element { | 
|  | public: | 
|  | StaticLibrary() = default; | 
|  | std::string name; | 
|  | int version; | 
|  | int versionMajor; | 
|  |  | 
|  | void Extract(xml::Element* element) override { | 
|  | auto parent_stack = extractor()->parent_stack(); | 
|  | if (parent_stack.size() > 0 && ElementCast<Application>(parent_stack[0])) { | 
|  | name = GetAttributeStringDefault(FindAttribute(element, NAME_ATTR), ""); | 
|  | version = GetAttributeIntegerDefault(FindAttribute(element, VERSION_ATTR), 0); | 
|  | versionMajor = GetAttributeIntegerDefault(FindAttribute(element, VERSION_MAJOR_ATTR), 0); | 
|  | } | 
|  | } | 
|  |  | 
|  | void Print(text::Printer* printer) override { | 
|  | printer->Print(StringPrintf( | 
|  | "static-library: name='%s' version='%d' versionMajor='%d'\n", | 
|  | name.data(), version, versionMajor)); | 
|  | } | 
|  |  | 
|  | void ToProto(pb::Badging* out_badging) override { | 
|  | auto static_library = out_badging->mutable_static_library(); | 
|  | static_library->set_name(name); | 
|  | static_library->set_version(version); | 
|  | static_library->set_version_major(versionMajor); | 
|  | } | 
|  | }; | 
|  |  | 
|  | /** Represents <uses-static-library> elements. **/ | 
|  | class UsesStaticLibrary : public ManifestExtractor::Element { | 
|  | public: | 
|  | UsesStaticLibrary() = default; | 
|  | std::string name; | 
|  | int version; | 
|  | int versionMajor; | 
|  | std::vector<std::string> certDigests; | 
|  |  | 
|  | void Extract(xml::Element* element) override { | 
|  | auto parent_stack = extractor()->parent_stack(); | 
|  | if (parent_stack.size() > 0 && ElementCast<Application>(parent_stack[0])) { | 
|  | name = GetAttributeStringDefault(FindAttribute(element, NAME_ATTR), ""); | 
|  | version = GetAttributeIntegerDefault(FindAttribute(element, VERSION_ATTR), 0); | 
|  | versionMajor = GetAttributeIntegerDefault(FindAttribute(element, VERSION_MAJOR_ATTR), 0); | 
|  | AddCertDigest(element); | 
|  | } | 
|  | } | 
|  |  | 
|  | void AddCertDigest(xml::Element* element) { | 
|  | std::string digest = GetAttributeStringDefault(FindAttribute(element, CERT_DIGEST_ATTR), ""); | 
|  | // We allow ":" delimiters in the SHA declaration as this is the format | 
|  | // emitted by the certtool making it easy for developers to copy/paste. | 
|  | digest.erase(std::remove(digest.begin(), digest.end(), ':'), digest.end()); | 
|  | if (!digest.empty()) { | 
|  | certDigests.push_back(digest); | 
|  | } | 
|  | } | 
|  |  | 
|  | void Print(text::Printer* printer) override { | 
|  | printer->Print(StringPrintf( | 
|  | "uses-static-library: name='%s' version='%d' versionMajor='%d'", | 
|  | name.data(), version, versionMajor)); | 
|  | for (size_t i = 0; i < certDigests.size(); i++) { | 
|  | printer->Print(StringPrintf(" certDigest='%s'", certDigests[i].data())); | 
|  | } | 
|  | printer->Print("\n"); | 
|  | } | 
|  |  | 
|  | void ToProto(pb::Badging* out_badging) override { | 
|  | auto uses_static_library = out_badging->add_uses_static_libraries(); | 
|  | uses_static_library->set_name(name); | 
|  | uses_static_library->set_version(version); | 
|  | uses_static_library->set_version_major(versionMajor); | 
|  | for (auto& cert : certDigests) { | 
|  | uses_static_library->add_certificates(cert); | 
|  | } | 
|  | } | 
|  | }; | 
|  |  | 
|  | /** Represents <sdk-library> elements. **/ | 
|  | class SdkLibrary : public ManifestExtractor::Element { | 
|  | public: | 
|  | SdkLibrary() = default; | 
|  | std::string name; | 
|  | int versionMajor; | 
|  |  | 
|  | void Extract(xml::Element* element) override { | 
|  | auto parent_stack = extractor()->parent_stack(); | 
|  | if (parent_stack.size() > 0 && ElementCast<Application>(parent_stack[0])) { | 
|  | name = GetAttributeStringDefault(FindAttribute(element, NAME_ATTR), ""); | 
|  | versionMajor = GetAttributeIntegerDefault(FindAttribute(element, VERSION_MAJOR_ATTR), 0); | 
|  | } | 
|  | } | 
|  |  | 
|  | void Print(text::Printer* printer) override { | 
|  | printer->Print( | 
|  | StringPrintf("sdk-library: name='%s' versionMajor='%d'\n", name.data(), versionMajor)); | 
|  | } | 
|  |  | 
|  | void ToProto(pb::Badging* out_badging) override { | 
|  | auto sdk_library = out_badging->mutable_sdk_library(); | 
|  | sdk_library->set_name(name); | 
|  | sdk_library->set_version_major(versionMajor); | 
|  | } | 
|  | }; | 
|  |  | 
|  | /** Represents <uses-sdk-library> elements. **/ | 
|  | class UsesSdkLibrary : public ManifestExtractor::Element { | 
|  | public: | 
|  | UsesSdkLibrary() = default; | 
|  | std::string name; | 
|  | int versionMajor; | 
|  | std::vector<std::string> certDigests; | 
|  |  | 
|  | void Extract(xml::Element* element) override { | 
|  | auto parent_stack = extractor()->parent_stack(); | 
|  | if (parent_stack.size() > 0 && ElementCast<Application>(parent_stack[0])) { | 
|  | name = GetAttributeStringDefault(FindAttribute(element, NAME_ATTR), ""); | 
|  | versionMajor = GetAttributeIntegerDefault(FindAttribute(element, VERSION_MAJOR_ATTR), 0); | 
|  | AddCertDigest(element); | 
|  | } | 
|  | } | 
|  |  | 
|  | void AddCertDigest(xml::Element* element) { | 
|  | std::string digest = GetAttributeStringDefault(FindAttribute(element, CERT_DIGEST_ATTR), ""); | 
|  | // We allow ":" delimiters in the SHA declaration as this is the format | 
|  | // emitted by the certtool making it easy for developers to copy/paste. | 
|  | digest.erase(std::remove(digest.begin(), digest.end(), ':'), digest.end()); | 
|  | if (!digest.empty()) { | 
|  | certDigests.push_back(digest); | 
|  | } | 
|  | } | 
|  |  | 
|  | void Print(text::Printer* printer) override { | 
|  | printer->Print( | 
|  | StringPrintf("uses-sdk-library: name='%s' versionMajor='%d'", name.data(), versionMajor)); | 
|  | for (size_t i = 0; i < certDigests.size(); i++) { | 
|  | printer->Print(StringPrintf(" certDigest='%s'", certDigests[i].data())); | 
|  | } | 
|  | printer->Print("\n"); | 
|  | } | 
|  |  | 
|  | void ToProto(pb::Badging* out_badging) override { | 
|  | auto uses_sdk_library = out_badging->add_uses_sdk_libraries(); | 
|  | uses_sdk_library->set_name(name); | 
|  | uses_sdk_library->set_version_major(versionMajor); | 
|  | for (auto& cert : certDigests) { | 
|  | uses_sdk_library->add_certificates(cert); | 
|  | } | 
|  | } | 
|  | }; | 
|  |  | 
|  | /** Represents <uses-native-library> elements. **/ | 
|  | class UsesNativeLibrary : public ManifestExtractor::Element { | 
|  | public: | 
|  | UsesNativeLibrary() = default; | 
|  | std::string name; | 
|  | int required; | 
|  |  | 
|  | void Extract(xml::Element* element) override { | 
|  | auto parent_stack = extractor()->parent_stack(); | 
|  | if (parent_stack.size() > 0 && ElementCast<Application>(parent_stack[0])) { | 
|  | name = GetAttributeStringDefault(FindAttribute(element, NAME_ATTR), ""); | 
|  | required = GetAttributeIntegerDefault(FindAttribute(element, REQUIRED_ATTR), 1); | 
|  | } | 
|  | } | 
|  |  | 
|  | void Print(text::Printer* printer) override { | 
|  | if (!name.empty()) { | 
|  | printer->Print(StringPrintf("uses-native-library%s:'%s'\n", | 
|  | (required == 0) ? "-not-required" : "", name.data())); | 
|  | } | 
|  | } | 
|  |  | 
|  | void ToProto(pb::Badging* out_badging) override { | 
|  | if (!name.empty()) { | 
|  | auto uses_native_library = out_badging->add_uses_native_libraries(); | 
|  | uses_native_library->set_name(name); | 
|  | uses_native_library->set_required(required != 0); | 
|  | } | 
|  | } | 
|  | }; | 
|  |  | 
|  | /** | 
|  | * Represents <meta-data> elements. These tags are only printed when a flag is passed in to | 
|  | * explicitly enable meta data printing. | 
|  | **/ | 
|  | class MetaData : public ManifestExtractor::Element { | 
|  | public: | 
|  | MetaData() = default; | 
|  | std::string name; | 
|  | std::string value; | 
|  | const int* value_int; | 
|  | std::string resource; | 
|  | const int* resource_int; | 
|  |  | 
|  | void Extract(xml::Element* element) override { | 
|  | name = GetAttributeStringDefault(FindAttribute(element, NAME_ATTR), ""); | 
|  | value = GetAttributeStringDefault(FindAttribute(element, VALUE_ATTR), ""); | 
|  | value_int = GetAttributeInteger(FindAttribute(element, VALUE_ATTR)); | 
|  | resource = GetAttributeStringDefault(FindAttribute(element, RESOURCE_ATTR), ""); | 
|  | resource_int = GetAttributeInteger(FindAttribute(element, RESOURCE_ATTR)); | 
|  | } | 
|  |  | 
|  | void Print(text::Printer* printer) override { | 
|  | if (extractor()->options_.include_meta_data && !name.empty()) { | 
|  | printer->Print(StringPrintf("meta-data: name='%s'", name.data())); | 
|  | if (!value.empty()) { | 
|  | printer->Print(StringPrintf(" value='%s'", value.data())); | 
|  | } else if (value_int) { | 
|  | printer->Print(StringPrintf(" value='%d'", *value_int)); | 
|  | } else { | 
|  | if (!resource.empty()) { | 
|  | printer->Print(StringPrintf(" resource='%s'", resource.data())); | 
|  | } else if (resource_int) { | 
|  | printer->Print(StringPrintf(" resource='%d'", *resource_int)); | 
|  | } | 
|  | } | 
|  | printer->Print("\n"); | 
|  | } | 
|  | } | 
|  |  | 
|  | void ToProto(pb::Badging* out_badging) override { | 
|  | if (!name.empty()) { | 
|  | auto metadata = out_badging->add_metadata(); | 
|  | metadata->set_name(name); | 
|  | if (!value.empty()) { | 
|  | metadata->set_value_string(value); | 
|  | } else if (value_int) { | 
|  | metadata->set_value_int(*value_int); | 
|  | } else { | 
|  | if (!resource.empty()) { | 
|  | metadata->set_resource_string(resource); | 
|  | } else if (resource_int) { | 
|  | metadata->set_resource_int(*resource_int); | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  | }; | 
|  |  | 
|  | /** | 
|  | * Represents <action> elements. Detects the presence of certain activity, provider, receiver, and | 
|  | * service components. | 
|  | **/ | 
|  | class Action : public ManifestExtractor::Element { | 
|  | public: | 
|  | Action() = default; | 
|  | std::string component = ""; | 
|  |  | 
|  | void Extract(xml::Element* element) override { | 
|  | auto parent_stack = extractor()->parent_stack(); | 
|  | std::string action = GetAttributeStringDefault(FindAttribute(element, NAME_ATTR), ""); | 
|  |  | 
|  | if (ElementCast<IntentFilter>(parent_stack[0])) { | 
|  | if (ElementCast<Activity>(parent_stack[1])) { | 
|  | // Detects the presence of a particular type of activity. | 
|  | Activity* activity = ElementCast<Activity>(parent_stack[1]); | 
|  | auto map = std::map<std::string, std::string>({ | 
|  | { "android.intent.action.MAIN" , "main" }, | 
|  | { "android.intent.action.VIDEO_CAMERA" , "camera" }, | 
|  | { "android.intent.action.STILL_IMAGE_CAMERA_SECURE" , "camera-secure" }, | 
|  | }); | 
|  |  | 
|  | auto entry = map.find(action); | 
|  | if (entry != map.end()) { | 
|  | component = entry->second; | 
|  | activity->has_component_ = true; | 
|  | } | 
|  |  | 
|  | if (action == "android.intent.action.MAIN") { | 
|  | activity->has_main_action = true; | 
|  | } | 
|  |  | 
|  | } else if (ElementCast<Receiver>(parent_stack[1])) { | 
|  | // Detects the presence of a particular type of receiver. If the action requires a | 
|  | // permission, then the receiver element is checked for the permission. | 
|  | Receiver* receiver = ElementCast<Receiver>(parent_stack[1]); | 
|  | auto map = std::map<std::string, std::string>({ | 
|  | { "android.appwidget.action.APPWIDGET_UPDATE" , "app-widget" }, | 
|  | { "android.app.action.DEVICE_ADMIN_ENABLED" , "device-admin" }, | 
|  | }); | 
|  |  | 
|  | auto permissions = std::map<std::string, std::string>({ | 
|  | { "android.app.action.DEVICE_ADMIN_ENABLED" , "android.permission.BIND_DEVICE_ADMIN" }, | 
|  | }); | 
|  |  | 
|  | auto entry = map.find(action); | 
|  | auto permission = permissions.find(action); | 
|  | if (entry != map.end() && (permission == permissions.end() | 
|  | || (receiver->permission && permission->second == *receiver->permission))) { | 
|  | receiver->has_component = true; | 
|  | component = entry->second; | 
|  | } | 
|  |  | 
|  | } else if (ElementCast<Service>(parent_stack[1])) { | 
|  | // Detects the presence of a particular type of service. If the action requires a | 
|  | // permission, then the service element is checked for the permission. | 
|  | Service* service = ElementCast<Service>(parent_stack[1]); | 
|  | auto map = std::map<std::string, std::string>({ | 
|  | { "android.view.InputMethod" , "ime" }, | 
|  | { "android.service.wallpaper.WallpaperService" , "wallpaper" }, | 
|  | { "android.accessibilityservice.AccessibilityService" , "accessibility" }, | 
|  | { "android.printservice.PrintService" , "print-service" }, | 
|  | { "android.nfc.cardemulation.action.HOST_APDU_SERVICE" , "host-apdu" }, | 
|  | { "android.nfc.cardemulation.action.OFF_HOST_APDU_SERVICE" , "offhost-apdu" }, | 
|  | { "android.service.notification.NotificationListenerService" ,"notification-listener" }, | 
|  | { "android.service.dreams.DreamService" , "dream" }, | 
|  | }); | 
|  |  | 
|  | auto permissions = std::map<std::string, std::string>({ | 
|  | { "android.accessibilityservice.AccessibilityService" , | 
|  | "android.permission.BIND_ACCESSIBILITY_SERVICE" }, | 
|  | { "android.printservice.PrintService" , "android.permission.BIND_PRINT_SERVICE" }, | 
|  | { "android.nfc.cardemulation.action.HOST_APDU_SERVICE" , | 
|  | "android.permission.BIND_NFC_SERVICE" }, | 
|  | { "android.nfc.cardemulation.action.OFF_HOST_APDU_SERVICE" , | 
|  | "android.permission.BIND_NFC_SERVICE" }, | 
|  | { "android.service.notification.NotificationListenerService" , | 
|  | "android.permission.BIND_NOTIFICATION_LISTENER_SERVICE" }, | 
|  | { "android.service.dreams.DreamService" , "android.permission.BIND_DREAM_SERVICE" }, | 
|  | }); | 
|  |  | 
|  | auto entry = map.find(action); | 
|  | auto permission = permissions.find(action); | 
|  | if (entry != map.end() && (permission == permissions.end() | 
|  | || (service->permission && permission->second == *service->permission))) { | 
|  | service->has_component= true; | 
|  | component = entry->second; | 
|  | } | 
|  |  | 
|  | } else if (ElementCast<Provider>(parent_stack[1])) { | 
|  | // Detects the presence of a particular type of receiver. If the provider requires a | 
|  | // permission, then the provider element is checked for the permission. | 
|  | // Detect whether this action | 
|  | Provider* provider = ElementCast<Provider>(parent_stack[1]); | 
|  | if (action == "android.content.action.DOCUMENTS_PROVIDER" | 
|  | && provider->has_required_saf_attributes) { | 
|  | component = "document-provider"; | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | // Represents a searchable interface | 
|  | if (action == "android.intent.action.SEARCH") { | 
|  | component = "search"; | 
|  | } | 
|  | } | 
|  | }; | 
|  |  | 
|  | /** | 
|  | * Represents <supports-input> elements. The element may have <input-type> elements nested within. | 
|  | **/ | 
|  | class SupportsInput : public ManifestExtractor::Element { | 
|  | public: | 
|  | SupportsInput() = default; | 
|  | std::vector<std::string> inputs; | 
|  |  | 
|  | void Print(text::Printer* printer) override { | 
|  | const size_t size = inputs.size(); | 
|  | if (size > 0) { | 
|  | printer->Print("supports-input: '"); | 
|  | for (size_t i = 0; i < size; i++) { | 
|  | printer->Print(StringPrintf("value='%s' ", inputs[i].data())); | 
|  | } | 
|  | printer->Print("\n"); | 
|  | } | 
|  | } | 
|  |  | 
|  | void ToProto(pb::Badging* out_badging) override { | 
|  | auto supports_input = out_badging->mutable_supports_input(); | 
|  | for (auto& input : inputs) { | 
|  | supports_input->add_inputs(input); | 
|  | } | 
|  | } | 
|  | }; | 
|  |  | 
|  | /** Represents <input-type> elements. **/ | 
|  | class InputType : public ManifestExtractor::Element { | 
|  | public: | 
|  | InputType() = default; | 
|  | void Extract(xml::Element* element) override { | 
|  | auto name = GetAttributeString(FindAttribute(element, NAME_ATTR)); | 
|  | auto parent_stack = extractor()->parent_stack(); | 
|  |  | 
|  | // Add the input to the set of supported inputs | 
|  | if (name && ElementCast<SupportsInput>(parent_stack[0])) { | 
|  | SupportsInput* supports = ElementCast<SupportsInput>(parent_stack[0]); | 
|  | supports->inputs.push_back(*name); | 
|  | } | 
|  | } | 
|  | }; | 
|  |  | 
|  | /** Represents <original-package> elements. **/ | 
|  | class OriginalPackage : public ManifestExtractor::Element { | 
|  | public: | 
|  | OriginalPackage() = default; | 
|  | const std::string* name = nullptr; | 
|  |  | 
|  | void Extract(xml::Element* element) override { | 
|  | name = GetAttributeString(FindAttribute(element, NAME_ATTR)); | 
|  | } | 
|  |  | 
|  | void Print(text::Printer* printer) override { | 
|  | if (name) { | 
|  | printer->Print(StringPrintf("original-package:'%s'\n", name->data())); | 
|  | } | 
|  | } | 
|  |  | 
|  | void ToProto(pb::Badging* out_badging) override { | 
|  | if (name) { | 
|  | out_badging->mutable_package()->set_original_package(*name); | 
|  | } | 
|  | } | 
|  | }; | 
|  |  | 
|  |  | 
|  | /** Represents <overlay> elements. **/ | 
|  | class Overlay : public ManifestExtractor::Element { | 
|  | public: | 
|  | Overlay() = default; | 
|  | const std::string* target_package = nullptr; | 
|  | int priority; | 
|  | bool is_static; | 
|  | const std::string* required_property_name = nullptr; | 
|  | const std::string* required_property_value = nullptr; | 
|  |  | 
|  | void Extract(xml::Element* element) override { | 
|  | target_package = GetAttributeString(FindAttribute(element, TARGET_PACKAGE_ATTR)); | 
|  | priority = GetAttributeIntegerDefault(FindAttribute(element, PRIORITY_ATTR), 0); | 
|  | is_static = GetAttributeIntegerDefault(FindAttribute(element, IS_STATIC_ATTR), false) != 0; | 
|  | required_property_name = GetAttributeString( | 
|  | FindAttribute(element, REQUIRED_SYSTEM_PROPERTY_NAME_ATTR)); | 
|  | required_property_value = GetAttributeString( | 
|  | FindAttribute(element, REQUIRED_SYSTEM_PROPERTY_VALUE_ATTR)); | 
|  | } | 
|  |  | 
|  | void Print(text::Printer* printer) override { | 
|  | printer->Print(StringPrintf("overlay:")); | 
|  | if (target_package) { | 
|  | printer->Print(StringPrintf(" targetPackage='%s'", target_package->c_str())); | 
|  | } | 
|  | printer->Print(StringPrintf(" priority='%d'", priority)); | 
|  | printer->Print(StringPrintf(" isStatic='%s'", is_static ? "true" : "false")); | 
|  | if (required_property_name) { | 
|  | printer->Print(StringPrintf(" requiredPropertyName='%s'", required_property_name->c_str())); | 
|  | } | 
|  | if (required_property_value) { | 
|  | printer->Print(StringPrintf(" requiredPropertyValue='%s'", required_property_value->c_str())); | 
|  | } | 
|  | printer->Print("\n"); | 
|  | } | 
|  |  | 
|  | void ToProto(pb::Badging* out_badging) override { | 
|  | auto overlay = out_badging->mutable_overlay(); | 
|  | if (target_package) { | 
|  | overlay->set_target_package(*target_package); | 
|  | } | 
|  | overlay->set_priority(priority); | 
|  | overlay->set_static_(is_static); | 
|  | if (required_property_name) { | 
|  | overlay->set_required_property_name(*required_property_name); | 
|  | } | 
|  | if (required_property_value) { | 
|  | overlay->set_required_property_value(*required_property_value); | 
|  | } | 
|  | } | 
|  | }; | 
|  |  | 
|  | /** * Represents <package-verifier> elements. **/ | 
|  | class PackageVerifier : public ManifestExtractor::Element { | 
|  | public: | 
|  | PackageVerifier() = default; | 
|  | const std::string* name = nullptr; | 
|  | const std::string* public_key = nullptr; | 
|  |  | 
|  | void Extract(xml::Element* element) override { | 
|  | name = GetAttributeString(FindAttribute(element, NAME_ATTR)); | 
|  | public_key = GetAttributeString(FindAttribute(element, PUBLIC_KEY_ATTR)); | 
|  | } | 
|  |  | 
|  | void Print(text::Printer* printer) override { | 
|  | if (name && public_key) { | 
|  | printer->Print(StringPrintf("package-verifier: name='%s' publicKey='%s'\n", | 
|  | name->data(), public_key->data())); | 
|  | } | 
|  | } | 
|  |  | 
|  | void ToProto(pb::Badging* out_badging) override { | 
|  | auto package_verifier = out_badging->mutable_package_verifier(); | 
|  | if (name && public_key) { | 
|  | package_verifier->set_name(*name); | 
|  | package_verifier->set_public_key(*public_key); | 
|  | } | 
|  | } | 
|  | }; | 
|  |  | 
|  | /** Represents <uses-package> elements. **/ | 
|  | class UsesPackage : public ManifestExtractor::Element { | 
|  | public: | 
|  | UsesPackage() = default; | 
|  | const std::string* packageType = nullptr; | 
|  | const std::string* name = nullptr; | 
|  | int version; | 
|  | int versionMajor; | 
|  | std::vector<std::string> certDigests; | 
|  |  | 
|  | void Extract(xml::Element* element) override { | 
|  | auto parent_stack = extractor()->parent_stack(); | 
|  | if (parent_stack.size() > 0 && ElementCast<Application>(parent_stack[0])) { | 
|  | packageType = GetAttributeString(FindAttribute(element, PACKAGE_TYPE_ATTR)); | 
|  | name = GetAttributeString(FindAttribute(element, NAME_ATTR)); | 
|  | version = GetAttributeIntegerDefault(FindAttribute(element, VERSION_ATTR), 0); | 
|  | versionMajor = GetAttributeIntegerDefault(FindAttribute(element, VERSION_MAJOR_ATTR), 0); | 
|  | AddCertDigest(element); | 
|  | } | 
|  | } | 
|  |  | 
|  | void AddCertDigest(xml::Element* element) { | 
|  | std::string digest = GetAttributeStringDefault(FindAttribute(element, CERT_DIGEST_ATTR), ""); | 
|  | // We allow ":" delimiters in the SHA declaration as this is the format | 
|  | // emitted by the certtool making it easy for developers to copy/paste. | 
|  | digest.erase(std::remove(digest.begin(), digest.end(), ':'), digest.end()); | 
|  | if (!digest.empty()) { | 
|  | certDigests.push_back(digest); | 
|  | } | 
|  | } | 
|  |  | 
|  | void Print(text::Printer* printer) override { | 
|  | if (name) { | 
|  | if (packageType) { | 
|  | printer->Print(StringPrintf( | 
|  | "uses-typed-package: type='%s' name='%s' version='%d' versionMajor='%d'", | 
|  | packageType->data(), name->data(), version, versionMajor)); | 
|  | for (size_t i = 0; i < certDigests.size(); i++) { | 
|  | printer->Print(StringPrintf(" certDigest='%s'", certDigests[i].data())); | 
|  | } | 
|  | printer->Print("\n"); | 
|  | } else { | 
|  | printer->Print(StringPrintf("uses-package:'%s'\n", name->data())); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | void ToProto(pb::Badging* out_badging) override { | 
|  | if (name) { | 
|  | auto uses_package = out_badging->add_uses_packages(); | 
|  | uses_package->set_name(*name); | 
|  | if (packageType) { | 
|  | uses_package->set_package_type(*packageType); | 
|  | uses_package->set_version(version); | 
|  | uses_package->set_version_major(versionMajor); | 
|  | for (auto& cert : certDigests) { | 
|  | uses_package->add_certificates(cert); | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  | }; | 
|  |  | 
|  | /** Represents <additional-certificate> elements. **/ | 
|  | class AdditionalCertificate : public ManifestExtractor::Element { | 
|  | public: | 
|  | AdditionalCertificate() = default; | 
|  |  | 
|  | void Extract(xml::Element* element) override { | 
|  | auto parent_stack = extractor()->parent_stack(); | 
|  | if (parent_stack.size() > 0) { | 
|  | if (ElementCast<UsesPackage>(parent_stack[0])) { | 
|  | UsesPackage* uses = ElementCast<UsesPackage>(parent_stack[0]); | 
|  | uses->AddCertDigest(element); | 
|  | } else if (ElementCast<UsesStaticLibrary>(parent_stack[0])) { | 
|  | UsesStaticLibrary* uses = ElementCast<UsesStaticLibrary>(parent_stack[0]); | 
|  | uses->AddCertDigest(element); | 
|  | } | 
|  | } | 
|  | } | 
|  | }; | 
|  |  | 
|  | /** Represents <screen> elements found in <compatible-screens> elements. */ | 
|  | class Screen : public ManifestExtractor::Element { | 
|  | public: | 
|  | Screen() = default; | 
|  | const int32_t* size = nullptr; | 
|  | const int32_t* density = nullptr; | 
|  |  | 
|  | void Extract(xml::Element* element) override { | 
|  | size = GetAttributeInteger(FindAttribute(element, SCREEN_SIZE_ATTR)); | 
|  | density = GetAttributeInteger(FindAttribute(element, SCREEN_DENSITY_ATTR)); | 
|  | } | 
|  |  | 
|  | void ToProto(pb::Badging* out_badging) override { | 
|  | if (size && density) { | 
|  | auto screen = out_badging->mutable_compatible_screens()->add_screens(); | 
|  | screen->set_density(*density); | 
|  | screen->set_size(*size); | 
|  | } | 
|  | } | 
|  | }; | 
|  |  | 
|  | /** | 
|  | * Represents <compatible-screens> elements. These elements have <screen> elements nested within | 
|  | * that each denote a supported screen size and screen density. | 
|  | **/ | 
|  | class CompatibleScreens : public ManifestExtractor::Element { | 
|  | public: | 
|  | CompatibleScreens() = default; | 
|  | void Print(text::Printer* printer) override { | 
|  | printer->Print("compatible-screens:"); | 
|  |  | 
|  | bool first = true; | 
|  | ForEachChild(this, [&printer, &first](ManifestExtractor::Element* el){ | 
|  | if (auto screen = ElementCast<Screen>(el)) { | 
|  | if (first) { | 
|  | first = false; | 
|  | } else { | 
|  | printer->Print(","); | 
|  | } | 
|  |  | 
|  | if (screen->size && screen->density) { | 
|  | printer->Print(StringPrintf("'%d/%d'", *screen->size, *screen->density)); | 
|  | } | 
|  | } | 
|  | }); | 
|  | printer->Print("\n"); | 
|  | } | 
|  | }; | 
|  |  | 
|  | /** Represents <supports-gl-texture> elements. **/ | 
|  | class SupportsGlTexture : public ManifestExtractor::Element { | 
|  | public: | 
|  | SupportsGlTexture() = default; | 
|  | const std::string* name = nullptr; | 
|  |  | 
|  | void Extract(xml::Element* element) override { | 
|  | name = GetAttributeString(FindAttribute(element, NAME_ATTR)); | 
|  | } | 
|  |  | 
|  | void Print(text::Printer* printer) override { | 
|  | if (name) { | 
|  | printer->Print(StringPrintf("supports-gl-texture:'%s'\n", name->data())); | 
|  | } | 
|  | } | 
|  |  | 
|  | void ToProto(pb::Badging* out_badging) override { | 
|  | if (name) { | 
|  | out_badging->mutable_supports_gl_texture()->add_name(*name); | 
|  | } | 
|  | } | 
|  | }; | 
|  |  | 
|  | /** Represents <property> elements. **/ | 
|  | class Property : public ManifestExtractor::Element { | 
|  | public: | 
|  | Property() = default; | 
|  | std::string name; | 
|  | std::string value; | 
|  | const int* value_int; | 
|  | std::string resource; | 
|  | const int* resource_int; | 
|  |  | 
|  | void Extract(xml::Element* element) override { | 
|  | name = GetAttributeStringDefault(FindAttribute(element, NAME_ATTR), ""); | 
|  | value = GetAttributeStringDefault(FindAttribute(element, VALUE_ATTR), ""); | 
|  | value_int = GetAttributeInteger(FindAttribute(element, VALUE_ATTR)); | 
|  | resource = GetAttributeStringDefault(FindAttribute(element, RESOURCE_ATTR), ""); | 
|  | resource_int = GetAttributeInteger(FindAttribute(element, RESOURCE_ATTR)); | 
|  | } | 
|  |  | 
|  | void Print(text::Printer* printer) override { | 
|  | printer->Print(StringPrintf("property: name='%s' ", name.data())); | 
|  | if (!value.empty()) { | 
|  | printer->Print(StringPrintf("value='%s' ", value.data())); | 
|  | } else if (value_int) { | 
|  | printer->Print(StringPrintf("value='%d' ", *value_int)); | 
|  | } else { | 
|  | if (!resource.empty()) { | 
|  | printer->Print(StringPrintf("resource='%s' ", resource.data())); | 
|  | } else if (resource_int) { | 
|  | printer->Print(StringPrintf("resource='%d' ", *resource_int)); | 
|  | } | 
|  | } | 
|  | printer->Print("\n"); | 
|  | } | 
|  |  | 
|  | void ToProto(pb::Badging* out_badging) override { | 
|  | if (!name.empty()) { | 
|  | auto property = out_badging->add_properties(); | 
|  | property->set_name(name); | 
|  | if (!value.empty()) { | 
|  | property->set_value_string(value); | 
|  | } else if (value_int) { | 
|  | property->set_value_int(*value_int); | 
|  | } else { | 
|  | if (!resource.empty()) { | 
|  | property->set_resource_string(resource); | 
|  | } else if (resource_int) { | 
|  | property->set_resource_int(*resource_int); | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  | }; | 
|  |  | 
|  | /** Recursively prints the extracted badging element. */ | 
|  | static void Print(ManifestExtractor::Element* el, text::Printer* printer) { | 
|  | el->Print(printer); | 
|  | for (auto &child : el->children()) { | 
|  | Print(child.get(), printer); | 
|  | } | 
|  | } | 
|  |  | 
|  | /** Recursively serializes extracted badging elements to proto. */ | 
|  | static void ToProto(ManifestExtractor::Element* el, pb::Badging* out_badging) { | 
|  | el->ToProto(out_badging); | 
|  | for (auto& child : el->children()) { | 
|  | ToProto(child.get(), out_badging); | 
|  | } | 
|  | } | 
|  |  | 
|  | bool ManifestExtractor::Extract(IDiagnostics* diag) { | 
|  | // Load the manifest | 
|  | doc_ = apk_->LoadXml("AndroidManifest.xml", diag); | 
|  | if (doc_ == nullptr) { | 
|  | diag->Error(DiagMessage() << "failed to find AndroidManifest.xml"); | 
|  | return false; | 
|  | } | 
|  |  | 
|  | xml::Element* element = doc_->root.get(); | 
|  | if (element->name != "manifest") { | 
|  | diag->Error(DiagMessage() << "manifest does not start with <manifest> tag"); | 
|  | return false; | 
|  | } | 
|  |  | 
|  | // Print only the <uses-permission>, <uses-permission-sdk23>, and <permission> elements if | 
|  | // printing only permission elements is requested | 
|  | if (options_.only_permissions) { | 
|  | root_element_ = ManifestExtractor::Element::Inflate(this, element); | 
|  |  | 
|  | if (auto manifest = ElementCast<Manifest>(root_element_.get())) { | 
|  | manifest->only_package_name = true; | 
|  |  | 
|  | for (xml::Element* child : element->GetChildElements()) { | 
|  | if (child->name == "uses-permission" || child->name == "uses-permission-sdk-23" | 
|  | || child->name == "permission") { | 
|  | // Inflate the element and its descendants | 
|  | auto permission_element = Visit(child); | 
|  | manifest->AddChild(permission_element); | 
|  | } | 
|  | } | 
|  | return true; | 
|  | } | 
|  | return false; | 
|  | } | 
|  |  | 
|  | // Collect information about the resource configurations | 
|  | if (apk_->GetResourceTable()) { | 
|  | for (auto &package : apk_->GetResourceTable()->packages) { | 
|  | for (auto &type : package->types) { | 
|  | for (auto &entry : type->entries) { | 
|  | for (auto &value : entry->values) { | 
|  | std::string locale_str = value->config.GetBcp47LanguageTag(); | 
|  |  | 
|  | // Collect all the unique locales of the apk | 
|  | if (locales_.find(locale_str) == locales_.end()) { | 
|  | ConfigDescription config = ManifestExtractor::DefaultConfig(); | 
|  | config.setBcp47Locale(locale_str.data()); | 
|  | locales_.insert(std::make_pair(locale_str, config)); | 
|  | } | 
|  |  | 
|  | // Collect all the unique density of the apk | 
|  | uint16_t density = (value->config.density == 0) ? (uint16_t) 160 | 
|  | : value->config.density; | 
|  | if (densities_.find(density) == densities_.end()) { | 
|  | ConfigDescription config = ManifestExtractor::DefaultConfig(); | 
|  | config.density = density; | 
|  | densities_.insert(std::make_pair(density, config)); | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | // Extract badging information | 
|  | root_element_ = Visit(element); | 
|  |  | 
|  | // Filter out all "uses-sdk" tags besides the very last tag. The android runtime only uses the | 
|  | // attribute values from the last defined tag. | 
|  | std::vector<UsesSdkBadging*> filtered_uses_sdk_tags; | 
|  | for (const auto& child : root_element_->children()) { | 
|  | if (auto uses_sdk = ElementCast<UsesSdkBadging>(child.get())) { | 
|  | filtered_uses_sdk_tags.emplace_back(uses_sdk); | 
|  | } | 
|  | } | 
|  | if (filtered_uses_sdk_tags.size() >= 2U) { | 
|  | filtered_uses_sdk_tags.pop_back(); | 
|  | root_element_->Filter([&](const ManifestExtractor::Element* e) { | 
|  | return std::find(filtered_uses_sdk_tags.begin(), filtered_uses_sdk_tags.end(), e) != | 
|  | filtered_uses_sdk_tags.end(); | 
|  | }); | 
|  | } | 
|  |  | 
|  | /** Recursively checks the extracted elements for the specified permission. **/ | 
|  | auto FindPermission = [&](ManifestExtractor::Element* root, | 
|  | const std::string& name) -> ManifestExtractor::Element* { | 
|  | return FindElement(root, [&](ManifestExtractor::Element* el) -> bool { | 
|  | if (UsesPermission* permission = ElementCast<UsesPermission>(el)) { | 
|  | return permission->name == name; | 
|  | } | 
|  | return false; | 
|  | }); | 
|  | }; | 
|  |  | 
|  | auto AddImpliedPermission = [&](const std::string& name, const std::string& reason, | 
|  | int32_t max_sdk_version) -> void { | 
|  | auto permission = util::make_unique<UsesPermission>(); | 
|  | permission->name = name; | 
|  | permission->maxSdkVersion = max_sdk_version; | 
|  | permission->implied = true; | 
|  | permission->impliedReason = reason; | 
|  | implied_permissions_.push_back(std::move(permission)); | 
|  | }; | 
|  |  | 
|  | // Implied permissions | 
|  | // Pre-1.6 implicitly granted permission compatibility logic | 
|  | bool insert_write_external = false; | 
|  | auto write_external_permission = ElementCast<UsesPermission>( | 
|  | FindPermission(root_element_.get(), "android.permission.WRITE_EXTERNAL_STORAGE")); | 
|  |  | 
|  | if (target_sdk() < SDK_DONUT) { | 
|  | if (!write_external_permission) { | 
|  | AddImpliedPermission("android.permission.WRITE_EXTERNAL_STORAGE", "targetSdkVersion < 4", -1); | 
|  | insert_write_external = true; | 
|  | } | 
|  |  | 
|  | if (!FindPermission(root_element_.get(), "android.permission.READ_PHONE_STATE")) { | 
|  | AddImpliedPermission("android.permission.READ_PHONE_STATE", "targetSdkVersion < 4", -1); | 
|  | } | 
|  | } | 
|  |  | 
|  | // If the application has requested WRITE_EXTERNAL_STORAGE, we will | 
|  | // force them to always take READ_EXTERNAL_STORAGE as well.  We always | 
|  | // do this (regardless of target API version) because we can't have | 
|  | // an app with write permission but not read permission. | 
|  | auto read_external = | 
|  | FindPermission(root_element_.get(), "android.permission.READ_EXTERNAL_STORAGE"); | 
|  | if (!read_external && (insert_write_external || write_external_permission)) { | 
|  | AddImpliedPermission( | 
|  | "android.permission.READ_EXTERNAL_STORAGE", "requested WRITE_EXTERNAL_STORAGE", | 
|  | (write_external_permission) ? write_external_permission->maxSdkVersion : -1); | 
|  | } | 
|  |  | 
|  | // Pre-JellyBean call log permission compatibility. | 
|  | if (target_sdk() < SDK_JELLY_BEAN) { | 
|  | if (!FindPermission(root_element_.get(), "android.permission.READ_CALL_LOG") && | 
|  | FindPermission(root_element_.get(), "android.permission.READ_CONTACTS")) { | 
|  | AddImpliedPermission("android.permission.READ_CALL_LOG", | 
|  | "targetSdkVersion < 16 and requested READ_CONTACTS", -1); | 
|  | } | 
|  |  | 
|  | if (!FindPermission(root_element_.get(), "android.permission.WRITE_CALL_LOG") && | 
|  | FindPermission(root_element_.get(), "android.permission.WRITE_CONTACTS")) { | 
|  | AddImpliedPermission("android.permission.WRITE_CALL_LOG", | 
|  | "targetSdkVersion < 16 and requested WRITE_CONTACTS", -1); | 
|  | } | 
|  | } | 
|  |  | 
|  | // If the app hasn't declared the touchscreen as a feature requirement (either | 
|  | // directly or implied, required or not), then the faketouch feature is implied. | 
|  | if (!common_feature_group()->HasFeature("android.hardware.touchscreen")) { | 
|  | common_feature_group()->addImpliedFeature("android.hardware.faketouch", | 
|  | "default feature for all apps", false); | 
|  | } | 
|  |  | 
|  | // Only print the common feature group if no feature group is defined | 
|  | std::vector<FeatureGroup*> feature_groups; | 
|  | ForEachChild(root_element_.get(), [&feature_groups](ManifestExtractor::Element* el) -> void { | 
|  | if (auto feature_group = ElementCast<FeatureGroup>(el)) { | 
|  | feature_groups.push_back(feature_group); | 
|  | } | 
|  | }); | 
|  |  | 
|  | if (feature_groups.empty()) { | 
|  | feature_groups_.push_back(common_feature_group()); | 
|  | } else { | 
|  | // Merge the common feature group into the feature group | 
|  | for (auto& feature_group : feature_groups) { | 
|  | feature_group->Merge(common_feature_group()); | 
|  | feature_groups_.push_back(feature_group); | 
|  | } | 
|  | }; | 
|  |  | 
|  | // Collect the component types of the application | 
|  | ForEachChild(root_element_.get(), [&](ManifestExtractor::Element* el) -> void { | 
|  | if (ElementCast<Action>(el)) { | 
|  | auto action = ElementCast<Action>(el); | 
|  | if (!action->component.empty()) { | 
|  | components_.discovered_components.insert(action->component); | 
|  | return; | 
|  | } | 
|  | } | 
|  |  | 
|  | if (ElementCast<Category>(el)) { | 
|  | auto category = ElementCast<Category>(el); | 
|  | if (!category->component.empty()) { | 
|  | components_.discovered_components.insert(category->component); | 
|  | return; | 
|  | } | 
|  | } | 
|  | }); | 
|  |  | 
|  | // Check for the payment component | 
|  | ForEachChild(root_element_.get(), [this, &diag](ManifestExtractor::Element* el) -> void { | 
|  | if (auto service = ElementCast<Service>(el)) { | 
|  | auto host_apdu_action = ElementCast<Action>(FindElement(service, | 
|  | [&](ManifestExtractor::Element* el) -> bool { | 
|  | if (auto action = ElementCast<Action>(el)) { | 
|  | return (action->component == "host-apdu"); | 
|  | } | 
|  | return false; | 
|  | })); | 
|  |  | 
|  | auto offhost_apdu_action = ElementCast<Action>(FindElement(service, | 
|  | [&](ManifestExtractor::Element* el) -> bool { | 
|  | if (auto action = ElementCast<Action>(el)) { | 
|  | return (action->component == "offhost-apdu"); | 
|  | } | 
|  | return false; | 
|  | })); | 
|  |  | 
|  | ForEachChild(service, | 
|  | [this, &diag, &host_apdu_action, | 
|  | &offhost_apdu_action](ManifestExtractor::Element* el) -> void { | 
|  | if (auto meta_data = ElementCast<MetaData>(el)) { | 
|  | if ((meta_data->name == "android.nfc.cardemulation.host_apdu_service" && | 
|  | host_apdu_action) || | 
|  | (meta_data->name == "android.nfc.cardemulation.off_host_apdu_service" && | 
|  | offhost_apdu_action)) { | 
|  | // Attempt to load the resource file | 
|  | if (!meta_data->resource.empty()) { | 
|  | return; | 
|  | } | 
|  | auto resource = this->apk_->LoadXml(meta_data->resource, diag); | 
|  | if (!resource) { | 
|  | return; | 
|  | } | 
|  |  | 
|  | // Look for the payment category on an <aid-group> element | 
|  | auto& root = resource.get()->root; | 
|  | if ((host_apdu_action && root->name == "host-apdu-service") || | 
|  | (offhost_apdu_action && root->name == "offhost-apdu-service")) { | 
|  | for (auto& child : root->GetChildElements()) { | 
|  | if (child->name == "aid-group") { | 
|  | auto category = FindAttribute(child, CATEGORY_ATTR); | 
|  | if (category && category->value == "payment") { | 
|  | this->components_.discovered_components.insert("payment"); | 
|  | return; | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  | }); | 
|  | } | 
|  | }); | 
|  |  | 
|  | // Print presence of activities, receivers, and services with no special components | 
|  | FindElement(root_element_.get(), [&](ManifestExtractor::Element* el) -> bool { | 
|  | if (auto activity = ElementCast<Activity>(el)) { | 
|  | if (!activity->has_component_) { | 
|  | components_.other_activities = true; | 
|  | return true; | 
|  | } | 
|  | } | 
|  | return false; | 
|  | }); | 
|  |  | 
|  | FindElement(root_element_.get(), [&](ManifestExtractor::Element* el) -> bool { | 
|  | if (auto receiver = ElementCast<Receiver>(el)) { | 
|  | if (!receiver->has_component) { | 
|  | components_.other_receivers = true; | 
|  | return true; | 
|  | } | 
|  | } | 
|  | return false; | 
|  | }); | 
|  |  | 
|  | FindElement(root_element_.get(), [&](ManifestExtractor::Element* el) -> bool { | 
|  | if (auto service = ElementCast<Service>(el)) { | 
|  | if (!service->has_component) { | 
|  | components_.other_services = true; | 
|  | return true; | 
|  | } | 
|  | } | 
|  | return false; | 
|  | }); | 
|  |  | 
|  | // Gather the supported screens | 
|  | const static SupportsScreen default_screens{}; | 
|  | SupportsScreen* screen = ElementCast<SupportsScreen>( | 
|  | FindElement(root_element_.get(), [&](ManifestExtractor::Element* el) -> bool { | 
|  | return ElementCast<SupportsScreen>(el) != nullptr; | 
|  | })); | 
|  | supports_screen_ = screen ? screen : &default_screens; | 
|  |  | 
|  | // Gather the supported architectures_ of the app | 
|  | std::set<std::string> architectures_from_apk; | 
|  | auto it = apk_->GetFileCollection()->Iterator(); | 
|  | while (it->HasNext()) { | 
|  | auto file_path = it->Next()->GetSource().path; | 
|  | size_t pos = file_path.find("lib/"); | 
|  | if (pos != std::string::npos) { | 
|  | file_path = file_path.substr(pos + 4); | 
|  | pos = file_path.find('/'); | 
|  | if (pos != std::string::npos) { | 
|  | file_path = file_path.substr(0, pos); | 
|  | } | 
|  |  | 
|  | architectures_from_apk.insert(file_path); | 
|  | } | 
|  | } | 
|  |  | 
|  | // Determine if the application has multiArch supports | 
|  | auto has_multi_arch = | 
|  | FindElement(root_element_.get(), [&](ManifestExtractor::Element* el) -> bool { | 
|  | if (auto application = ElementCast<Application>(el)) { | 
|  | return application->has_multi_arch; | 
|  | } | 
|  | return false; | 
|  | }); | 
|  |  | 
|  | bool output_alt_native_code = false; | 
|  | // A multiArch package is one that contains 64-bit and | 
|  | // 32-bit versions of native code and expects 3rd-party | 
|  | // apps to load these native code libraries. Since most | 
|  | // 64-bit systems also support 32-bit apps, the apps | 
|  | // loading this multiArch package's code may be either | 
|  | if (has_multi_arch) { | 
|  | // If this is a multiArch package, report the 64-bit | 
|  | // version only. Then as a separate entry, report the | 
|  | // rest. | 
|  | // | 
|  | // If we report the 32-bit architecture, this APK will | 
|  | // be installed on a 32-bit device, causing a large waste | 
|  | // of bandwidth and disk space. This assumes that | 
|  | // the developer of the multiArch package has also | 
|  | // made a version that is 32-bit only. | 
|  | const std::string kIntel64 = "x86_64"; | 
|  | const std::string kArm64 = "arm64-v8a"; | 
|  |  | 
|  | auto arch = architectures_from_apk.find(kIntel64); | 
|  | if (arch == architectures_from_apk.end()) { | 
|  | arch = architectures_from_apk.find(kArm64); | 
|  | } | 
|  |  | 
|  | if (arch != architectures_from_apk.end()) { | 
|  | architectures_.architectures.insert(*arch); | 
|  | architectures_from_apk.erase(arch); | 
|  | output_alt_native_code = true; | 
|  | } | 
|  | } | 
|  | for (auto& arch : architectures_from_apk) { | 
|  | if (output_alt_native_code) { | 
|  | architectures_.alt_architectures.insert(arch); | 
|  | } else { | 
|  | architectures_.architectures.insert(arch); | 
|  | } | 
|  | } | 
|  | return true; | 
|  | } | 
|  |  | 
|  | bool ManifestExtractor::Dump(text::Printer* printer) { | 
|  | Print(root_element_.get(), printer); | 
|  | if (options_.only_permissions) { | 
|  | return true; | 
|  | } | 
|  |  | 
|  | for (auto& implied_permission : implied_permissions_) { | 
|  | implied_permission->Print(printer); | 
|  | } | 
|  | for (auto& feature_group : feature_groups_) { | 
|  | feature_group->PrintGroup(printer); | 
|  | } | 
|  | components_.Print(printer); | 
|  | supports_screen_->PrintScreens(printer, target_sdk_); | 
|  |  | 
|  | // Print all the unique locales of the apk | 
|  | printer->Print("locales:"); | 
|  | for (auto& config : locales_) { | 
|  | if (config.first.empty()) { | 
|  | printer->Print(" '--_--'"); | 
|  | } else { | 
|  | printer->Print(StringPrintf(" '%s'", config.first.data())); | 
|  | } | 
|  | } | 
|  | printer->Print("\n"); | 
|  |  | 
|  | // Print all the densities locales of the apk | 
|  | printer->Print("densities:"); | 
|  | for (auto& config : densities_) { | 
|  | printer->Print(StringPrintf(" '%d'", config.first)); | 
|  | } | 
|  | printer->Print("\n"); | 
|  |  | 
|  | architectures_.Print(printer); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | bool ManifestExtractor::DumpProto(pb::Badging* out_badging) { | 
|  | ToProto(root_element_.get(), out_badging); | 
|  | for (auto& implied_permission : implied_permissions_) { | 
|  | implied_permission->ToProto(out_badging); | 
|  | } | 
|  | for (auto& feature_group : feature_groups_) { | 
|  | feature_group->GroupToProto(out_badging); | 
|  | } | 
|  | components_.ToProto(out_badging); | 
|  | supports_screen_->ToProtoScreens(out_badging, target_sdk_); | 
|  |  | 
|  | for (auto& config : locales_) { | 
|  | if (!config.first.empty()) { | 
|  | out_badging->add_locales(config.first); | 
|  | } | 
|  | } | 
|  | for (auto& config : densities_) { | 
|  | out_badging->add_densities(config.first); | 
|  | } | 
|  |  | 
|  | architectures_.ToProto(out_badging); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Returns the element casted to the type if the element is of that type. Otherwise, returns a null | 
|  | * pointer. | 
|  | **/ | 
|  | template<typename T> | 
|  | T* ElementCast(ManifestExtractor::Element* element) { | 
|  | if (element == nullptr) { | 
|  | return nullptr; | 
|  | } | 
|  |  | 
|  | const std::unordered_map<std::string, bool> kTagCheck = { | 
|  | {"action", std::is_base_of<Action, T>::value}, | 
|  | {"activity", std::is_base_of<Activity, T>::value}, | 
|  | {"additional-certificate", std::is_base_of<AdditionalCertificate, T>::value}, | 
|  | {"application", std::is_base_of<Application, T>::value}, | 
|  | {"category", std::is_base_of<Category, T>::value}, | 
|  | {"compatible-screens", std::is_base_of<CompatibleScreens, T>::value}, | 
|  | {"feature-group", std::is_base_of<FeatureGroup, T>::value}, | 
|  | {"input-type", std::is_base_of<InputType, T>::value}, | 
|  | {"intent-filter", std::is_base_of<IntentFilter, T>::value}, | 
|  | {"meta-data", std::is_base_of<MetaData, T>::value}, | 
|  | {"manifest", std::is_base_of<Manifest, T>::value}, | 
|  | {"original-package", std::is_base_of<OriginalPackage, T>::value}, | 
|  | {"overlay", std::is_base_of<Overlay, T>::value}, | 
|  | {"package-verifier", std::is_base_of<PackageVerifier, T>::value}, | 
|  | {"permission", std::is_base_of<Permission, T>::value}, | 
|  | {"property", std::is_base_of<Property, T>::value}, | 
|  | {"provider", std::is_base_of<Provider, T>::value}, | 
|  | {"receiver", std::is_base_of<Receiver, T>::value}, | 
|  | {"required-feature", std::is_base_of<RequiredFeature, T>::value}, | 
|  | {"required-not-feature", std::is_base_of<RequiredNotFeature, T>::value}, | 
|  | {"screen", std::is_base_of<Screen, T>::value}, | 
|  | {"service", std::is_base_of<Service, T>::value}, | 
|  | {"sdk-library", std::is_base_of<SdkLibrary, T>::value}, | 
|  | {"static-library", std::is_base_of<StaticLibrary, T>::value}, | 
|  | {"supports-gl-texture", std::is_base_of<SupportsGlTexture, T>::value}, | 
|  | {"supports-input", std::is_base_of<SupportsInput, T>::value}, | 
|  | {"supports-screens", std::is_base_of<SupportsScreen, T>::value}, | 
|  | {"uses-configuration", std::is_base_of<UsesConfiguarion, T>::value}, | 
|  | {"uses-feature", std::is_base_of<UsesFeature, T>::value}, | 
|  | {"uses-library", std::is_base_of<UsesLibrary, T>::value}, | 
|  | {"uses-native-library", std::is_base_of<UsesNativeLibrary, T>::value}, | 
|  | {"uses-package", std::is_base_of<UsesPackage, T>::value}, | 
|  | {"uses-permission", std::is_base_of<UsesPermission, T>::value}, | 
|  | {"uses-permission-sdk-23", std::is_base_of<UsesPermissionSdk23, T>::value}, | 
|  | {"uses-sdk", std::is_base_of<UsesSdkBadging, T>::value}, | 
|  | {"uses-sdk-library", std::is_base_of<UsesSdkLibrary, T>::value}, | 
|  | {"uses-static-library", std::is_base_of<UsesStaticLibrary, T>::value}, | 
|  | }; | 
|  |  | 
|  | auto check = kTagCheck.find(element->tag()); | 
|  | if (check != kTagCheck.end() && check->second) { | 
|  | return static_cast<T*>(element); | 
|  | } | 
|  | return nullptr; | 
|  | } | 
|  |  | 
|  | template<typename T> | 
|  | std::unique_ptr<T> CreateType() { | 
|  | return std::move(util::make_unique<T>()); | 
|  | } | 
|  |  | 
|  | std::unique_ptr<ManifestExtractor::Element> ManifestExtractor::Element::Inflate( | 
|  | ManifestExtractor* extractor, xml::Element* el) { | 
|  | const std::unordered_map<std::string, | 
|  | std::function<std::unique_ptr<ManifestExtractor::Element>()>> | 
|  | kTagCheck = { | 
|  | {"action", &CreateType<Action>}, | 
|  | {"activity", &CreateType<Activity>}, | 
|  | {"additional-certificate", &CreateType<AdditionalCertificate>}, | 
|  | {"application", &CreateType<Application>}, | 
|  | {"category", &CreateType<Category>}, | 
|  | {"compatible-screens", &CreateType<CompatibleScreens>}, | 
|  | {"feature-group", &CreateType<FeatureGroup>}, | 
|  | {"input-type", &CreateType<InputType>}, | 
|  | {"intent-filter", &CreateType<IntentFilter>}, | 
|  | {"manifest", &CreateType<Manifest>}, | 
|  | {"meta-data", &CreateType<MetaData>}, | 
|  | {"original-package", &CreateType<OriginalPackage>}, | 
|  | {"overlay", &CreateType<Overlay>}, | 
|  | {"package-verifier", &CreateType<PackageVerifier>}, | 
|  | {"permission", &CreateType<Permission>}, | 
|  | {"property", &CreateType<Property>}, | 
|  | {"provider", &CreateType<Provider>}, | 
|  | {"receiver", &CreateType<Receiver>}, | 
|  | {"required-feature", &CreateType<RequiredFeature>}, | 
|  | {"required-not-feature", &CreateType<RequiredNotFeature>}, | 
|  | {"screen", &CreateType<Screen>}, | 
|  | {"service", &CreateType<Service>}, | 
|  | {"sdk-library", &CreateType<SdkLibrary>}, | 
|  | {"static-library", &CreateType<StaticLibrary>}, | 
|  | {"supports-gl-texture", &CreateType<SupportsGlTexture>}, | 
|  | {"supports-input", &CreateType<SupportsInput>}, | 
|  | {"supports-screens", &CreateType<SupportsScreen>}, | 
|  | {"uses-configuration", &CreateType<UsesConfiguarion>}, | 
|  | {"uses-feature", &CreateType<UsesFeature>}, | 
|  | {"uses-library", &CreateType<UsesLibrary>}, | 
|  | {"uses-native-library", &CreateType<UsesNativeLibrary>}, | 
|  | {"uses-package", &CreateType<UsesPackage>}, | 
|  | {"uses-permission", &CreateType<UsesPermission>}, | 
|  | {"uses-permission-sdk-23", &CreateType<UsesPermissionSdk23>}, | 
|  | {"uses-sdk", &CreateType<UsesSdkBadging>}, | 
|  | {"uses-sdk-library", &CreateType<UsesSdkLibrary>}, | 
|  | {"uses-static-library", &CreateType<UsesStaticLibrary>}, | 
|  | }; | 
|  |  | 
|  | // Attempt to map the xml tag to a element inflater | 
|  | std::unique_ptr<ManifestExtractor::Element> element; | 
|  | auto check = kTagCheck.find(el->name); | 
|  | if (check != kTagCheck.end()) { | 
|  | element = check->second(); | 
|  | } else { | 
|  | element = util::make_unique<ManifestExtractor::Element>(); | 
|  | } | 
|  |  | 
|  | element->extractor_ = extractor; | 
|  | element->tag_ = el->name; | 
|  | element->Extract(el); | 
|  | return element; | 
|  | } | 
|  |  | 
|  | std::unique_ptr<ManifestExtractor::Element> ManifestExtractor::Visit(xml::Element* el) { | 
|  | auto element = ManifestExtractor::Element::Inflate(this, el); | 
|  | parent_stack_.insert(parent_stack_.begin(), element.get()); | 
|  |  | 
|  | // Process the element and recursively visit the children | 
|  | for (xml::Element* child : el->GetChildElements()) { | 
|  | auto v = Visit(child); | 
|  | element->AddChild(v); | 
|  | } | 
|  |  | 
|  | parent_stack_.erase(parent_stack_.begin()); | 
|  | return element; | 
|  | } | 
|  |  | 
|  | int DumpManifest(LoadedApk* apk, DumpManifestOptions& options, text::Printer* printer, | 
|  | IDiagnostics* diag) { | 
|  | ManifestExtractor extractor(apk, options); | 
|  | if (!extractor.Extract(diag)) { | 
|  | return 1; | 
|  | } | 
|  | return extractor.Dump(printer) ? 0 : 1; | 
|  | } | 
|  |  | 
|  | int DumpBadgingProto(LoadedApk* apk, pb::Badging* out_badging, IDiagnostics* diag) { | 
|  | DumpManifestOptions options{/* include_meta_data= */ true, | 
|  | /* only_permissions= */ false}; | 
|  | ManifestExtractor extractor(apk, options); | 
|  | if (!extractor.Extract(diag)) { | 
|  | return 1; | 
|  | } | 
|  | return extractor.DumpProto(out_badging) ? 0 : 1; | 
|  | } | 
|  |  | 
|  | } // namespace aapt |