/* * Copyright (C) 2015 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 "Logger.h" #include "ManifestValidator.h" #include "Maybe.h" #include "Source.h" #include "Util.h" #include <androidfw/ResourceTypes.h> namespace aapt { ManifestValidator::ManifestValidator(const android::ResTable& table) : mTable(table) { } bool ManifestValidator::validate(const Source& source, android::ResXMLParser* parser) { SourceLogger logger(source); android::ResXMLParser::event_code_t code; while ((code = parser->next()) != android::ResXMLParser::END_DOCUMENT && code != android::ResXMLParser::BAD_DOCUMENT) { if (code != android::ResXMLParser::START_TAG) { continue; } size_t len = 0; const StringPiece16 namespaceUri(parser->getElementNamespace(&len), len); if (!namespaceUri.empty()) { continue; } const StringPiece16 name(parser->getElementName(&len), len); if (name.empty()) { logger.error(parser->getLineNumber()) << "failed to get the element name." << std::endl; return false; } if (name == u"manifest") { if (!validateManifest(source, parser)) { return false; } } } return true; } Maybe<StringPiece16> ManifestValidator::getAttributeValue(android::ResXMLParser* parser, size_t idx) { android::Res_value value; if (parser->getAttributeValue(idx, &value) < 0) { return StringPiece16(); } const android::ResStringPool* pool = &parser->getStrings(); if (value.dataType == android::Res_value::TYPE_REFERENCE) { ssize_t strIdx = mTable.resolveReference(&value, 0x10000000u); if (strIdx < 0) { return {}; } pool = mTable.getTableStringBlock(strIdx); } if (value.dataType != android::Res_value::TYPE_STRING || !pool) { return {}; } return util::getString(*pool, value.data); } Maybe<StringPiece16> ManifestValidator::getAttributeInlineValue(android::ResXMLParser* parser, size_t idx) { android::Res_value value; if (parser->getAttributeValue(idx, &value) < 0) { return StringPiece16(); } if (value.dataType != android::Res_value::TYPE_STRING) { return {}; } return util::getString(parser->getStrings(), value.data); } bool ManifestValidator::validateInlineAttribute(android::ResXMLParser* parser, size_t idx, SourceLogger& logger, const StringPiece16& charSet) { size_t len = 0; StringPiece16 element(parser->getElementName(&len), len); StringPiece16 attributeName(parser->getAttributeName(idx, &len), len); Maybe<StringPiece16> result = getAttributeInlineValue(parser, idx); if (!result) { logger.error(parser->getLineNumber()) << "<" << element << "> must have a '" << attributeName << "' attribute with a string literal value." << std::endl; return false; } return validateAttributeImpl(element, attributeName, result.value(), charSet, parser->getLineNumber(), logger); } bool ManifestValidator::validateAttribute(android::ResXMLParser* parser, size_t idx, SourceLogger& logger, const StringPiece16& charSet) { size_t len = 0; StringPiece16 element(parser->getElementName(&len), len); StringPiece16 attributeName(parser->getAttributeName(idx, &len), len); Maybe<StringPiece16> result = getAttributeValue(parser, idx); if (!result) { logger.error(parser->getLineNumber()) << "<" << element << "> must have a '" << attributeName << "' attribute that points to a string." << std::endl; return false; } return validateAttributeImpl(element, attributeName, result.value(), charSet, parser->getLineNumber(), logger); } bool ManifestValidator::validateAttributeImpl(const StringPiece16& element, const StringPiece16& attributeName, const StringPiece16& attributeValue, const StringPiece16& charSet, size_t lineNumber, SourceLogger& logger) { StringPiece16::const_iterator badIter = util::findNonAlphaNumericAndNotInSet(attributeValue, charSet); if (badIter != attributeValue.end()) { logger.error(lineNumber) << "tag <" << element << "> attribute '" << attributeName << "' has invalid character '" << StringPiece16(badIter, 1) << "'." << std::endl; return false; } if (!attributeValue.empty()) { StringPiece16 trimmed = util::trimWhitespace(attributeValue); if (attributeValue.begin() != trimmed.begin()) { logger.error(lineNumber) << "tag <" << element << "> attribute '" << attributeName << "' can not start with whitespace." << std::endl; return false; } if (attributeValue.end() != trimmed.end()) { logger.error(lineNumber) << "tag <" << element << "> attribute '" << attributeName << "' can not end with whitespace." << std::endl; return false; } } return true; } constexpr const char16_t* kPackageIdentSet = u"._"; bool ManifestValidator::validateManifest(const Source& source, android::ResXMLParser* parser) { bool error = false; SourceLogger logger(source); const StringPiece16 kAndroid = u"android"; const StringPiece16 kPackage = u"package"; const StringPiece16 kSharedUserId = u"sharedUserId"; ssize_t idx; idx = parser->indexOfAttribute(nullptr, 0, kPackage.data(), kPackage.size()); if (idx < 0) { logger.error(parser->getLineNumber()) << "missing package attribute." << std::endl; error = true; } else { error |= !validateInlineAttribute(parser, idx, logger, kPackageIdentSet); } idx = parser->indexOfAttribute(kAndroid.data(), kAndroid.size(), kSharedUserId.data(), kSharedUserId.size()); if (idx >= 0) { error |= !validateInlineAttribute(parser, idx, logger, kPackageIdentSet); } return !error; } } // namespace aapt