178 lines
5.8 KiB
C++
178 lines
5.8 KiB
C++
/*
|
|
* 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 <memory>
|
|
#include <string>
|
|
|
|
#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<std::string> ResToTypeEntryName(const AssetManager2& am, uint32_t resid) {
|
|
AssetManager2::ResourceName name;
|
|
if (!am.GetResourceName(resid, &name)) {
|
|
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<OverlayManifestInfo> ExtractOverlayManifestInfo(const std::string& path,
|
|
bool assert_overlay) {
|
|
std::unique_ptr<const ZipFile> zip = ZipFile::Open(path);
|
|
if (!zip) {
|
|
return Error("failed to open %s as a zip file", path.c_str());
|
|
}
|
|
|
|
std::unique_ptr<const MemoryChunk> entry = zip->Uncompress("AndroidManifest.xml");
|
|
if (!entry) {
|
|
return Error("failed to uncompress AndroidManifest.xml from %s", path.c_str());
|
|
}
|
|
|
|
Result<std::unique_ptr<const XmlParser>> 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 <manifest> 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("<overlay> 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 <overlay> 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
|