blob: a8301743aab3ad738a2aa6c885f3cde57b800114 [file] [log] [blame]
/*
* 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.
*/
#ifndef ART_TOOLS_VERIDEX_HIDDEN_API_H_
#define ART_TOOLS_VERIDEX_HIDDEN_API_H_
#include "api_list_filter.h"
#include "base/hiddenapi_flags.h"
#include "dex/method_reference.h"
#include <map>
#include <ostream>
#include <string>
namespace art {
class DexFile;
enum class SignatureSource {
UNKNOWN,
BOOT,
APP,
};
/**
* Helper class for logging if a method/field is in a hidden API list.
*/
class HiddenApi {
public:
HiddenApi(const char* flags_file, const ApiListFilter& api_list_filter);
hiddenapi::ApiList GetApiList(const std::string& name) const {
auto it = api_list_.find(name);
return (it == api_list_.end()) ? hiddenapi::ApiList() : it->second;
}
bool ShouldReport(const std::string& signature) const {
return api_list_filter_.Matches(GetApiList(signature));
}
void AddSignatureSource(const std::string &signature, SignatureSource source) {
const auto type = GetApiClassName(signature);
auto it = source_.find(type);
if (it == source_.end() || it->second == SignatureSource::UNKNOWN) {
source_[type] = source;
} else if (it->second != source) {
LOG(WARNING) << type << "is present both in boot and in app.";
if (source == SignatureSource::BOOT) {
// Runtime resolves to boot type, so it takes precedence.
it->second = source;
}
} else {
// Already exists with the same source.
}
}
SignatureSource GetSignatureSource(const std::string& signature) const {
auto it = source_.find(GetApiClassName(signature));
return (it == source_.end()) ? SignatureSource::UNKNOWN : it->second;
}
bool IsInBoot(const std::string& signature) const {
return SignatureSource::BOOT == GetSignatureSource(signature);
}
static std::string GetApiMethodName(const DexFile& dex_file, uint32_t method_index);
static std::string GetApiFieldName(const DexFile& dex_file, uint32_t field_index);
static std::string GetApiMethodName(MethodReference ref) {
return HiddenApi::GetApiMethodName(*ref.dex_file, ref.index);
}
static std::string ToInternalName(const std::string& str) {
std::string val = str;
std::replace(val.begin(), val.end(), '.', '/');
return "L" + val + ";";
}
private:
void AddSignatureToApiList(const std::string& signature, hiddenapi::ApiList membership);
static std::string GetApiClassName(const std::string& signature) {
size_t pos = signature.find("->");
if (pos != std::string::npos) {
return signature.substr(0, pos);
}
return signature;
}
const ApiListFilter& api_list_filter_;
std::map<std::string, hiddenapi::ApiList> api_list_;
std::map<std::string, SignatureSource> source_;
};
struct HiddenApiStats {
uint32_t count = 0;
uint32_t reflection_count = 0;
uint32_t linking_count = 0;
// Ensure enough space for kInvalid as well, and initialize all to zero
uint32_t api_counts[hiddenapi::ApiList::kValueSize] = {};
};
} // namespace art
#endif // ART_TOOLS_VERIDEX_HIDDEN_API_H_