/* * 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 "idmap2/ResourceUtils.h" #include #include #include "androidfw/StringPiece.h" #include "androidfw/Util.h" #include "idmap2/Result.h" #include "idmap2/XmlParser.h" #include "idmap2/ZipFile.h" using android::StringPiece16; using android::idmap2::Result; using android::idmap2::XmlParser; using android::idmap2::ZipFile; using android::util::Utf16ToUtf8; namespace android::idmap2::utils { bool IsReference(uint8_t data_type) { return data_type == Res_value::TYPE_REFERENCE || data_type == Res_value::TYPE_DYNAMIC_REFERENCE; } StringPiece DataTypeToString(uint8_t data_type) { switch (data_type) { case Res_value::TYPE_NULL: return "null"; case Res_value::TYPE_REFERENCE: return "reference"; case Res_value::TYPE_ATTRIBUTE: return "attribute"; case Res_value::TYPE_STRING: return "string"; case Res_value::TYPE_FLOAT: return "float"; case Res_value::TYPE_DIMENSION: return "dimension"; case Res_value::TYPE_FRACTION: return "fraction"; case Res_value::TYPE_DYNAMIC_REFERENCE: return "reference (dynamic)"; case Res_value::TYPE_DYNAMIC_ATTRIBUTE: return "attribute (dynamic)"; case Res_value::TYPE_INT_DEC: case Res_value::TYPE_INT_HEX: return "integer"; case Res_value::TYPE_INT_BOOLEAN: return "boolean"; case Res_value::TYPE_INT_COLOR_ARGB8: case Res_value::TYPE_INT_COLOR_RGB8: case Res_value::TYPE_INT_COLOR_RGB4: return "color"; default: return "unknown"; } } Result ResToTypeEntryName(const AssetManager2& am, uint32_t resid) { const auto name = am.GetResourceName(resid); if (!name.has_value()) { return Error("no resource 0x%08x in asset manager", resid); } std::string out; if (name->type != nullptr) { out.append(name->type, name->type_len); } else { out += Utf16ToUtf8(StringPiece16(name->type16, name->type_len)); } out.append("/"); if (name->entry != nullptr) { out.append(name->entry, name->entry_len); } else { out += Utf16ToUtf8(StringPiece16(name->entry16, name->entry_len)); } return out; } Result ExtractOverlayManifestInfo(const std::string& path, bool assert_overlay) { std::unique_ptr zip = ZipFile::Open(path); if (!zip) { return Error("failed to open %s as a zip file", path.c_str()); } std::unique_ptr entry = zip->Uncompress("AndroidManifest.xml"); if (!entry) { return Error("failed to uncompress AndroidManifest.xml from %s", path.c_str()); } Result> xml = XmlParser::Create(entry->buf, entry->size); if (!xml) { return Error("failed to parse AndroidManifest.xml from %s", path.c_str()); } auto manifest_it = (*xml)->tree_iterator(); if (manifest_it->event() != XmlParser::Event::START_TAG || manifest_it->name() != "manifest") { return Error("root element tag is not in AndroidManifest.xml of %s", path.c_str()); } auto overlay_it = std::find_if(manifest_it.begin(), manifest_it.end(), [](const auto& it) { return it.event() == XmlParser::Event::START_TAG && it.name() == "overlay"; }); OverlayManifestInfo info{}; if (overlay_it == manifest_it.end()) { if (!assert_overlay) { return info; } return Error(" missing from AndroidManifest.xml of %s", path.c_str()); } if (auto result_str = overlay_it->GetAttributeStringValue("targetPackage")) { info.target_package = *result_str; } else { return Error("android:targetPackage missing from of %s: %s", path.c_str(), result_str.GetErrorMessage().c_str()); } if (auto result_str = overlay_it->GetAttributeStringValue("targetName")) { info.target_name = *result_str; } if (auto result_value = overlay_it->GetAttributeValue("resourcesMap")) { if (IsReference((*result_value).dataType)) { info.resource_mapping = (*result_value).data; } else { return Error("android:resourcesMap is not a reference in AndroidManifest.xml of %s", path.c_str()); } } if (auto result_value = overlay_it->GetAttributeValue("isStatic")) { if ((*result_value).dataType >= Res_value::TYPE_FIRST_INT && (*result_value).dataType <= Res_value::TYPE_LAST_INT) { info.is_static = (*result_value).data != 0U; } else { return Error("android:isStatic is not a boolean in AndroidManifest.xml of %s", path.c_str()); } } if (auto result_value = overlay_it->GetAttributeValue("priority")) { if ((*result_value).dataType >= Res_value::TYPE_FIRST_INT && (*result_value).dataType <= Res_value::TYPE_LAST_INT) { info.priority = (*result_value).data; } else { return Error("android:priority is not an integer in AndroidManifest.xml of %s", path.c_str()); } } if (auto result_str = overlay_it->GetAttributeStringValue("requiredSystemPropertyName")) { info.requiredSystemPropertyName = *result_str; } if (auto result_str = overlay_it->GetAttributeStringValue("requiredSystemPropertyValue")) { info.requiredSystemPropertyValue = *result_str; } return info; } } // namespace android::idmap2::utils