// -*- Mode: C++ -*- // // Copyright (C) 2013-2017 Red Hat, Inc. // // This file is part of the GNU Application Binary Interface Generic // Analysis and Instrumentation Library (libabigail). This library is // free software; you can redistribute it and/or modify it under the // terms of the GNU Lesser General Public License as published by the // Free Software Foundation; either version 3, or (at your option) any // later version. // This library is distributed in the hope that it will be useful, but // WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU // General Lesser Public License for more details. // You should have received a copy of the GNU Lesser General Public // License along with this program; see the file COPYING-LGPLV3. If // not, see . // // Author: Dodji Seketeli /// @file /// /// This program reads an elf file, try to load its debug info (in /// DWARF format) and emit it back in a set of "text sections" in native /// libabigail XML format. #include #include #include #include #include #include #include #include #include #include #include "abg-config.h" #include "abg-tools-utils.h" #include "abg-corpus.h" #include "abg-dwarf-reader.h" #include "abg-writer.h" #include "abg-reader.h" #include "abg-comparison.h" using std::string; using std::cerr; using std::cout; using std::ostream; using std::ofstream; using std::vector; using std::tr1::shared_ptr; using abigail::tools_utils::emit_prefix; using abigail::tools_utils::temp_file; using abigail::tools_utils::temp_file_sptr; using abigail::tools_utils::check_file; using abigail::ir::environment_sptr; using abigail::ir::environment; using abigail::corpus; using abigail::corpus_sptr; using abigail::translation_units; using abigail::suppr::suppression_sptr; using abigail::suppr::suppressions_type; using abigail::suppr::read_suppressions; using abigail::comparison::corpus_diff; using abigail::comparison::corpus_diff_sptr; using abigail::comparison::compute_diff; using abigail::comparison::diff_context_sptr; using abigail::comparison::diff_context; using abigail::xml_writer::write_corpus_to_native_xml; using abigail::xml_reader::read_corpus_from_native_xml_file; using abigail::dwarf_reader::read_context; using abigail::dwarf_reader::read_context_sptr; using abigail::dwarf_reader::read_corpus_from_elf; using abigail::dwarf_reader::create_read_context; using namespace abigail; struct options { string wrong_option; string in_file_path; string out_file_path; shared_ptr di_root_path; string headers_dir; vector suppression_paths; bool display_version; bool check_alt_debug_info_path; bool show_base_name_alt_debug_info_path; bool write_architecture; bool write_corpus_path; bool load_all_types; bool linux_kernel_mode; bool show_stats; bool noout; bool show_locs; bool abidiff; bool annotate; bool do_log; options() : display_version(), check_alt_debug_info_path(), show_base_name_alt_debug_info_path(), write_architecture(true), write_corpus_path(true), load_all_types(), linux_kernel_mode(true), show_stats(), noout(), show_locs(true), abidiff(), do_log() {} }; static void display_usage(const string& prog_name, ostream& out) { emit_prefix(prog_name, out) << "usage: " << prog_name << " [options] []\n" << " where options can be: \n" << " --help|-h display this message\n" << " --version|-v display program version information and exit\n" << " --debug-info-dir|-d look for debug info under 'dir-path'\n" << " --headers-dir|--hd the path to headers of the elf file\n" << " --out-file write the output to 'file-path'\n" << " --noout do not emit anything after reading the binary\n" << " --suppressions|--suppr specify a suppression file\n" << " --no-architecture do not emit architecture info in the output\n" << " --no-corpus-path do not take the path to the corpora into account\n" << " --no-show-locs do now show location information\n" << " --check-alternate-debug-info check alternate debug info " "of \n" << " --check-alternate-debug-info-base-name check alternate " "debug info of , and show its base name\n" << " --load-all-types read all types including those not reachable from " "exported declarations\n" << " --no-linux-kernel-mode don't consider the input binary as " "a Linux Kernel binary\n" << " --abidiff compare the loaded ABI against itself\n" << " --annotate annotate the ABI artifacts emitted in the output\n" << " --stats show statistics about various internal stuff\n" << " --verbose show verbose messages about internal stuff\n"; ; } static bool parse_command_line(int argc, char* argv[], options& opts) { if (argc < 2) return false; for (int i = 1; i < argc; ++i) { if (argv[i][0] != '-') { if (opts.in_file_path.empty()) opts.in_file_path = argv[i]; else return false; } else if (!strcmp(argv[i], "--version") || !strcmp(argv[i], "-v")) opts.display_version = true; else if (!strcmp(argv[i], "--debug-info-dir") || !strcmp(argv[i], "-d")) { if (argc <= i + 1 || argv[i + 1][0] == '-') return false; // elfutils wants the root path to the debug info to be // absolute. opts.di_root_path = abigail::tools_utils::make_path_absolute(argv[i + 1]); ++i; } else if (!strcmp(argv[i], "--headers-dir") || !strcmp(argv[i], "--hd")) { int j = i + 1; if (j >= argc) return false; opts.headers_dir = argv[j]; ++i; } else if (!strcmp(argv[i], "--out-file")) { if (argc <= i + 1 || argv[i + 1][0] == '-' || !opts.out_file_path.empty()) return false; opts.out_file_path = argv[i + 1]; ++i; } else if (!strcmp(argv[i], "--suppressions") || !strcmp(argv[i], "--suppr")) { int j = i + 1; if (j >= argc) return false; opts.suppression_paths.push_back(argv[j]); ++i; } else if (!strcmp(argv[i], "--noout")) opts.noout = true; else if (!strcmp(argv[i], "--no-architecture")) opts.write_architecture = false; else if (!strcmp(argv[i], "--no-corpus-path")) opts.write_corpus_path = false; else if (!strcmp(argv[i], "--no-show-locs")) opts.show_locs = false; else if (!strcmp(argv[i], "--check-alternate-debug-info") || !strcmp(argv[i], "--check-alternate-debug-info-base-name")) { if (argc <= i + 1 || argv[i + 1][0] == '-' || !opts.in_file_path.empty()) return false; if (!strcmp(argv[i], "--check-alternate-debug-info-base-name")) opts.show_base_name_alt_debug_info_path = true; opts.check_alt_debug_info_path = true; opts.in_file_path = argv[i + 1]; ++i; } else if (!strcmp(argv[i], "--load-all-types")) opts.load_all_types = true; else if (!strcmp(argv[i], "--no-linux-kernel-mode")) opts.linux_kernel_mode = false; else if (!strcmp(argv[i], "--abidiff")) opts.abidiff = true; else if (!strcmp(argv[i], "--annotate")) opts.annotate = true; else if (!strcmp(argv[i], "--stats")) opts.show_stats = true; else if (!strcmp(argv[i], "--verbose")) opts.do_log = true; else if (!strcmp(argv[i], "--help") || !strcmp(argv[i], "--h")) return false; else { if (strlen(argv[i]) >= 2 && argv[i][0] == '-' && argv[i][1] == '-') opts.wrong_option = argv[i]; return false; } } return true; } /// Initialize the context use for driving ABI comparison. /// /// @param ctxt the context to initialize. static void set_diff_context(diff_context_sptr& ctxt) { ctxt->default_output_stream(&cerr); ctxt->error_output_stream(&cerr); // Filter out changes that are not meaningful from an ABI // standpoint, from the diff output. ctxt->switch_categories_off (abigail::comparison::ACCESS_CHANGE_CATEGORY | abigail::comparison::COMPATIBLE_TYPE_CHANGE_CATEGORY | abigail::comparison::HARMLESS_DECL_NAME_CHANGE_CATEGORY); } /// Check that the suppression specification files supplied are /// present. If not, emit an error on stderr. /// /// @param opts the options instance to use. /// /// @return true if all suppression specification files are present, /// false otherwise. static bool maybe_check_suppression_files(const options& opts) { for (vector::const_iterator i = opts.suppression_paths.begin(); i != opts.suppression_paths.end(); ++i) if (!check_file(*i, cerr, "abidiff")) return false; return true; } /// Set suppression specifications to the @p read_context used to load /// the ABI corpus from the ELF/DWARF file. /// /// These suppression specifications are going to be applied to drop /// some ABI artifacts on the floor (while reading the ELF/DWARF file) /// and thus minimize the size of the resulting ABI corpus. /// /// @param read_ctxt the read context to apply the suppression /// specifications to. /// /// @param opts the options where to get the suppression /// specifications from. static void set_suppressions(read_context& read_ctxt, const options& opts) { suppressions_type supprs; for (vector::const_iterator i = opts.suppression_paths.begin(); i != opts.suppression_paths.end(); ++i) read_suppressions(*i, supprs); suppression_sptr suppr = abigail::tools_utils::gen_suppr_spec_from_headers(opts.headers_dir); if (suppr) supprs.push_back(suppr); add_read_context_suppressions(read_ctxt, supprs); } int main(int argc, char* argv[]) { options opts; if (!parse_command_line(argc, argv, opts) || (opts.in_file_path.empty() && !opts.display_version)) { if (!opts.wrong_option.empty()) emit_prefix(argv[0], cerr) << "unrecognized option: " << opts.wrong_option << "\n"; display_usage(argv[0], cerr); return 1; } if (opts.display_version) { string major, minor, revision; abigail::abigail_get_library_version(major, minor, revision); cout << major << "." << minor << "." << revision << "\n"; return 0; } assert(!opts.in_file_path.empty()); if (!abigail::tools_utils::check_file(opts.in_file_path, cerr, argv[0])) return 1; if (!maybe_check_suppression_files(opts)) return 1; abigail::tools_utils::file_type type = abigail::tools_utils::guess_file_type(opts.in_file_path); if (type != abigail::tools_utils::FILE_TYPE_ELF && type != abigail::tools_utils::FILE_TYPE_AR) { emit_prefix(argv[0], cerr) << opts.in_file_path << " is not an ELF file\n"; return 1; } char* p = opts.di_root_path.get(); environment_sptr env(new environment); corpus_sptr corp; read_context_sptr c = create_read_context(opts.in_file_path, &p, env.get(), opts.load_all_types, opts.linux_kernel_mode); read_context& ctxt = *c; set_show_stats(ctxt, opts.show_stats); set_suppressions(ctxt, opts); abigail::dwarf_reader::set_do_log(ctxt, opts.do_log); if (opts.check_alt_debug_info_path) { bool has_alt_di = false; string alt_di_path; abigail::dwarf_reader::status status = abigail::dwarf_reader::has_alt_debug_info(ctxt, has_alt_di, alt_di_path); if (status & abigail::dwarf_reader::STATUS_OK) { if (alt_di_path.empty()) ; else { cout << "found the alternate debug info file"; if (opts.show_base_name_alt_debug_info_path) { tools_utils::base_name(alt_di_path, alt_di_path); cout << " '" << alt_di_path << "'"; } cout << "\n"; } return 0; } else { emit_prefix(argv[0], cerr) << "could not find alternate debug info file\n"; return 1; } } dwarf_reader::status s = dwarf_reader::STATUS_UNKNOWN; corp = read_corpus_from_elf(ctxt, s); c.reset(); if (!corp) { if (s == dwarf_reader::STATUS_DEBUG_INFO_NOT_FOUND) { if (p == 0) { emit_prefix(argv[0], cerr) << "Could not read debug info from " << opts.in_file_path << "\n"; emit_prefix(argv[0], cerr) << "You might want to supply the root directory where " "to search debug info from, using the " "--debug-info-dir option " "(e.g --debug-info-dir /usr/lib/debug)\n"; } else { emit_prefix(argv[0], cerr) << "Could not read debug info for '" << opts.in_file_path << "' from debug info root directory '" << p << "'\n"; } } else if (s == dwarf_reader::STATUS_NO_SYMBOLS_FOUND) emit_prefix(argv[0], cerr) << "Could not read ELF symbol information from " << opts.in_file_path << "\n"; return 1; } else { if (opts.abidiff) { // Save the abi in abixml format in a temporary file, read // it back, and compare the ABI of what we've read back // against the ABI of the input ELF file. temp_file_sptr tmp_file = temp_file::create(); write_corpus_to_native_xml(corp, 0, tmp_file->get_stream(), opts.annotate); tmp_file->get_stream().flush(); corpus_sptr corp2 = read_corpus_from_native_xml_file(tmp_file->get_path(), env.get()); if (!corp2) { emit_prefix(argv[0], cerr) << "Could not read temporary XML representation of " "elf file back\n"; return 1; } diff_context_sptr ctxt(new diff_context); set_diff_context(ctxt); ctxt->show_locs(opts.show_locs); corpus_diff_sptr diff = compute_diff(corp, corp2, ctxt); bool has_error = diff->has_incompatible_changes(); if (has_error) { diff->report(cerr); return 1; } return 0; } if (opts.noout) return 0; if (!opts.write_architecture) corp->set_architecture_name(""); if (!opts.write_corpus_path) corp->set_path(""); if (!opts.out_file_path.empty()) { ofstream of(opts.out_file_path.c_str(), std::ios_base::trunc); if (!of.is_open()) { emit_prefix(argv[0], cerr) << "could not open output file '" << opts.out_file_path << "'\n"; return 1; } abigail::xml_writer::write_corpus_to_native_xml(corp, 0, of, opts.annotate); of.close(); return 0; } else abigail::xml_writer::write_corpus_to_native_xml(corp, 0, cout, opts.annotate); } return 0; }