blob: c187bedf4f51a4e8ce91f303b31c6c08de9fa9c4 [file] [log] [blame]
// Copyright (C) 2016 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 "diff/abi_diff.h"
#include "utils/config_file.h"
#include <llvm/ADT/SmallString.h>
#include <llvm/Support/CommandLine.h>
#include <llvm/Support/FileSystem.h>
#include <llvm/Support/Path.h>
#include <llvm/Support/raw_ostream.h>
#include <fstream>
using header_checker::diff::HeaderAbiDiff;
using header_checker::repr::CompatibilityStatusIR;
using header_checker::repr::DiffPolicyOptions;
using header_checker::repr::TextFormatIR;
using header_checker::utils::ConfigFile;
using header_checker::utils::ConfigSection;
static llvm::cl::OptionCategory header_checker_category(
"header-abi-diff options");
static llvm::cl::opt<std::string> compatibility_report(
"o", llvm::cl::desc("<compatibility report>"), llvm::cl::Required,
llvm::cl::cat(header_checker_category));
static llvm::cl::opt<std::string> lib_name(
"lib", llvm::cl::desc("<lib name>"), llvm::cl::Required,
llvm::cl::cat(header_checker_category));
static llvm::cl::opt<std::string> arch(
"arch", llvm::cl::desc("<arch>"), llvm::cl::Required,
llvm::cl::cat(header_checker_category));
static llvm::cl::opt<std::string> new_dump(
"new", llvm::cl::desc("<new dump>"), llvm::cl::Required,
llvm::cl::cat(header_checker_category));
static llvm::cl::opt<std::string> old_dump(
"old", llvm::cl::desc("<old dump>"), llvm::cl::Required,
llvm::cl::cat(header_checker_category));
static llvm::cl::opt<std::string> ignore_symbol_list(
"ignore-symbols", llvm::cl::desc("ignore symbols"), llvm::cl::Optional,
llvm::cl::cat(header_checker_category));
static llvm::cl::opt<bool> advice_only(
"advice-only", llvm::cl::desc("Advisory mode only"), llvm::cl::Optional,
llvm::cl::cat(header_checker_category));
static llvm::cl::opt<bool> elf_unreferenced_symbol_errors(
"elf-unreferenced-symbol-errors",
llvm::cl::desc("This option is deprecated and has no effect."),
llvm::cl::Optional, llvm::cl::cat(header_checker_category));
static llvm::cl::opt<bool> check_all_apis(
"check-all-apis",
llvm::cl::desc("All apis, whether referenced or not, by exported symbols in"
" the dynsym table of a shared library are checked"),
llvm::cl::Optional, llvm::cl::cat(header_checker_category));
static llvm::cl::opt<bool> allow_extensions(
"allow-extensions",
llvm::cl::desc("Do not return a non zero status on extensions"),
llvm::cl::Optional, llvm::cl::cat(header_checker_category));
static llvm::cl::opt<bool> allow_unreferenced_elf_symbol_changes(
"allow-unreferenced-elf-symbol-changes",
llvm::cl::desc("Do not return a non zero status on changes to elf symbols"
"not referenced by metadata in exported headers"),
llvm::cl::Optional, llvm::cl::cat(header_checker_category));
static llvm::cl::opt<bool> allow_unreferenced_changes(
"allow-unreferenced-changes",
llvm::cl::desc("Do not return a non zero status on changes to data"
" structures which are not directly referenced by exported"
" APIs."),
llvm::cl::Optional, llvm::cl::cat(header_checker_category));
static llvm::cl::opt<bool> allow_adding_removing_referenced_apis(
"allow-adding-removing-referenced-apis",
llvm::cl::desc("Do not return a non zero status on addition or removal of "
"functions and variables that have corresponding symbols. "
"This option ignores the functions built in runtime "
"libraries and may not be declared in headers."),
llvm::cl::Optional, llvm::cl::cat(header_checker_category));
static llvm::cl::opt<bool> consider_opaque_types_different(
"consider-opaque-types-different",
llvm::cl::desc("Consider opaque types with different names as different. "
"This should not be used while comparing C++ library ABIs"),
llvm::cl::Optional, llvm::cl::cat(header_checker_category));
static llvm::cl::opt<TextFormatIR> text_format_old(
"input-format-old", llvm::cl::desc("Specify input format of old abi dump"),
llvm::cl::values(clEnumValN(TextFormatIR::ProtobufTextFormat,
"ProtobufTextFormat", "ProtobufTextFormat"),
clEnumValN(TextFormatIR::Json, "Json", "JSON")),
llvm::cl::init(TextFormatIR::Json),
llvm::cl::cat(header_checker_category));
static llvm::cl::opt<TextFormatIR> text_format_new(
"input-format-new", llvm::cl::desc("Specify input format of new abi dump"),
llvm::cl::values(clEnumValN(TextFormatIR::ProtobufTextFormat,
"ProtobufTextFormat", "ProtobufTextFormat"),
clEnumValN(TextFormatIR::Json, "Json", "JSON")),
llvm::cl::init(TextFormatIR::Json),
llvm::cl::cat(header_checker_category));
static llvm::cl::opt<TextFormatIR> text_format_diff(
"text-format-diff", llvm::cl::desc("Specify text format of abi-diff"),
llvm::cl::values(clEnumValN(TextFormatIR::ProtobufTextFormat,
"ProtobufTextFormat", "ProtobufTextFormat")),
llvm::cl::init(TextFormatIR::ProtobufTextFormat),
llvm::cl::cat(header_checker_category));
static llvm::cl::opt<bool> allow_adding_removing_weak_symbols(
"allow-adding-removing-weak-symbols",
llvm::cl::desc("Do not treat addition or removal of weak symbols as "
"incompatible changes."),
llvm::cl::init(false), llvm::cl::Optional,
llvm::cl::cat(header_checker_category));
static llvm::cl::opt<std::string> target_version(
"target-version",
llvm::cl::desc(
"Load the flags for <target version> and <lib name> from config.json in "
"the old dump's parent directory."
),
llvm::cl::init("current"), llvm::cl::Optional,
llvm::cl::cat(header_checker_category));
static llvm::cl::list<std::string> ignore_linker_set_keys(
"ignore-linker-set-key",
llvm::cl::desc("Ignore a specific type or function in the comparison."),
llvm::cl::ZeroOrMore, llvm::cl::cat(header_checker_category));
static std::set<std::string> LoadIgnoredSymbols(std::string &symbol_list_path) {
std::ifstream symbol_ifstream(symbol_list_path);
std::set<std::string> ignored_symbols;
if (!symbol_ifstream) {
llvm::errs() << "Failed to open file containing symbols to ignore\n";
::exit(1);
}
std::string line = "";
while (std::getline(symbol_ifstream, line)) {
ignored_symbols.insert(line);
}
return ignored_symbols;
}
static std::string GetConfigFilePath(const std::string &dump_file_path) {
llvm::SmallString<128> config_file_path(dump_file_path);
llvm::sys::path::remove_filename(config_file_path);
llvm::sys::path::append(config_file_path, "config.json");
return std::string(config_file_path);
}
static void UpdateFlags(const ConfigSection &section) {
for (auto &&i : section.GetIgnoredLinkerSetKeys()) {
ignore_linker_set_keys.push_back(i);
}
for (auto &&p : section) {
auto &&key = p.first;
bool value_bool = p.second;
if (key == "allow_adding_removing_weak_symbols") {
allow_adding_removing_weak_symbols = value_bool;
} else if (key == "advice_only") {
advice_only = value_bool;
} else if (key == "elf_unreferenced_symbol_errors") {
elf_unreferenced_symbol_errors = value_bool;
} else if (key == "check_all_apis") {
check_all_apis = value_bool;
} else if (key == "allow_extensions") {
allow_extensions = value_bool;
} else if (key == "allow_unreferenced_elf_symbol_changes") {
allow_unreferenced_elf_symbol_changes = value_bool;
} else if (key == "allow_unreferenced_changes") {
allow_unreferenced_changes = value_bool;
} else if (key == "allow_adding_removing_referenced_apis") {
allow_adding_removing_referenced_apis = value_bool;
} else if (key == "consider_opaque_types_different") {
consider_opaque_types_different = value_bool;
}
}
}
static void ReadConfigFile(const std::string &config_file_path) {
ConfigFile cfg;
if (!cfg.Load(config_file_path)) {
::exit(1);
}
if (cfg.HasGlobalSection()) {
UpdateFlags(cfg.GetGlobalSection());
}
if (cfg.HasSection(lib_name, target_version)) {
UpdateFlags(cfg.GetSection(lib_name, target_version));
}
}
static std::string GetErrorMessage(CompatibilityStatusIR status) {
if (status & CompatibilityStatusIR::Incompatible) {
return "INCOMPATIBLE CHANGES";
}
if (!allow_unreferenced_elf_symbol_changes &&
(status & CompatibilityStatusIR::ElfIncompatible)) {
return "ELF Symbols not referenced by exported headers removed";
}
if (!allow_extensions && (status & CompatibilityStatusIR::Extension)) {
return "EXTENDING CHANGES";
}
if (!allow_extensions && !allow_unreferenced_elf_symbol_changes &&
(status & CompatibilityStatusIR::ElfExtension)) {
return "ELF Symbols not referenced by exported headers added";
}
if (!allow_unreferenced_changes &&
(status & CompatibilityStatusIR::UnreferencedChanges)) {
return "changes in exported headers, which are not directly referenced "
"by exported symbols. This MIGHT be an ABI breaking change due to "
"internal typecasts";
}
return "";
}
int main(int argc, const char **argv) {
llvm::cl::ParseCommandLineOptions(argc, argv, "header-checker");
const std::string config_file_path = GetConfigFilePath(old_dump);
if (llvm::sys::fs::exists(config_file_path)) {
ReadConfigFile(config_file_path);
}
std::set<std::string> ignored_symbols;
if (llvm::sys::fs::exists(ignore_symbol_list)) {
ignored_symbols = LoadIgnoredSymbols(ignore_symbol_list);
}
std::set<std::string> ignored_linker_set_keys_set(
ignore_linker_set_keys.begin(), ignore_linker_set_keys.end());
const DiffPolicyOptions diff_policy_options{
.consider_opaque_types_different = consider_opaque_types_different,
.allow_adding_removing_referenced_apis =
allow_adding_removing_referenced_apis};
HeaderAbiDiff judge(lib_name, arch, old_dump, new_dump, compatibility_report,
ignored_symbols, ignored_linker_set_keys_set,
allow_adding_removing_weak_symbols, diff_policy_options,
check_all_apis, text_format_old, text_format_new,
text_format_diff);
CompatibilityStatusIR status = judge.GenerateCompatibilityReport();
std::string status_str = GetErrorMessage(status);
if (!status_str.empty()) {
llvm::errs() << "******************************************************\n"
<< "\033[31;1merror: \033[0m"
<< lib_name
<< "'s ABI has "
<< status_str
<< ". Please check compatibility report at: "
<< compatibility_report << "\n"
<< "******************************************************\n";
}
return (advice_only || status_str.empty()) ? CompatibilityStatusIR::Compatible
: status;
}