mirror of
git://sourceware.org/git/libabigail.git
synced 2025-01-22 09:12:55 +00:00
2d1e324727
* include/abg-comp-filter.h: Update year in copyright notice. * include/abg-comparison.h: Likewise. * include/abg-config.h: Likewise. * include/abg-corpus.h: Likewise. * include/abg-ctf-reader.h: Likewise. * include/abg-cxx-compat.h: Likewise. * include/abg-diff-utils.h: Likewise. * include/abg-dwarf-reader.h: Likewise. * include/abg-elf-reader-common.h: Likewise. * include/abg-fwd.h: Likewise. * include/abg-hash.h: Likewise. * include/abg-ini.h: Likewise. * include/abg-interned-str.h: Likewise. * include/abg-ir.h: Likewise. * include/abg-libxml-utils.h: Likewise. * include/abg-reader.h: Likewise. * include/abg-regex.h: Likewise. * include/abg-reporter.h: Likewise. * include/abg-sptr-utils.h: Likewise. * include/abg-suppression.h: Likewise. * include/abg-tools-utils.h: Likewise. * include/abg-traverse.h: Likewise. * include/abg-viz-common.h: Likewise. * include/abg-viz-dot.h: Likewise. * include/abg-viz-svg.h: Likewise. * include/abg-workers.h: Likewise. * include/abg-writer.h: Likewise. * src/abg-comp-filter.cc: Likewise. * src/abg-comparison-priv.h: Likewise. * src/abg-comparison.cc: Likewise. * src/abg-config.cc: Likewise. * src/abg-corpus-priv.h: Likewise. * src/abg-corpus.cc: Likewise. * src/abg-ctf-reader.cc: Likewise. * src/abg-default-reporter.cc: Likewise. * src/abg-diff-utils.cc: Likewise. * src/abg-dwarf-reader.cc: Likewise. * src/abg-elf-helpers.cc: Likewise. * src/abg-elf-helpers.h: Likewise. * src/abg-elf-reader-common.cc: Likewise. * src/abg-hash.cc: Likewise. * src/abg-ini.cc: Likewise. * src/abg-internal.h: Likewise. * src/abg-ir-priv.h: Likewise. * src/abg-ir.cc: Likewise. * src/abg-leaf-reporter.cc: Likewise. * src/abg-libxml-utils.cc: Likewise. * src/abg-reader.cc: Likewise. * src/abg-regex.cc: Likewise. * src/abg-reporter-priv.cc: Likewise. * src/abg-reporter-priv.h: Likewise. * src/abg-suppression-priv.h: Likewise. * src/abg-suppression.cc: Likewise. * src/abg-symtab-reader.cc: Likewise. * src/abg-symtab-reader.h: Likewise. * src/abg-tools-utils.cc: Likewise. * src/abg-traverse.cc: Likewise. * src/abg-viz-common.cc: Likewise. * src/abg-viz-dot.cc: Likewise. * src/abg-viz-svg.cc: Likewise. * src/abg-workers.cc: Likewise. * src/abg-writer.cc: Likewise. * tests/print-diff-tree.cc: Likewise. * tests/test-abicompat.cc: Likewise. * tests/test-abidiff-exit.cc: Likewise. * tests/test-abidiff.cc: Likewise. * tests/test-alt-dwarf-file.cc: Likewise. * tests/test-core-diff.cc: Likewise. * tests/test-cxx-compat.cc: Likewise. * tests/test-diff-dwarf-abixml.cc: Likewise. * tests/test-diff-dwarf.cc: Likewise. * tests/test-diff-filter.cc: Likewise. * tests/test-diff-pkg.cc: Likewise. * tests/test-diff-suppr.cc: Likewise. * tests/test-diff2.cc: Likewise. * tests/test-elf-helpers.cc: Likewise. * tests/test-ini.cc: Likewise. * tests/test-ir-walker.cc: Likewise. * tests/test-kmi-whitelist.cc: Likewise. * tests/test-lookup-syms.cc: Likewise. * tests/test-read-ctf.cc: Likewise. * tests/test-read-dwarf.cc: Likewise. * tests/test-read-write.cc: Likewise. * tests/test-symtab-reader.cc: Likewise. * tests/test-symtab.cc: Likewise. * tests/test-tools-utils.cc: Likewise. * tests/test-types-stability.cc: Likewise. * tests/test-utils.cc: Likewise. * tests/test-utils.h: Likewise. * tools/abicompat.cc: Likewise. * tools/abidiff.cc: Likewise. * tools/abidw.cc: Likewise. * tools/abilint.cc: Likewise. * tools/abipkgdiff.cc: Likewise. * tools/abisym.cc: Likewise. * tools/binilint.cc: Likewise. * tools/kmidiff.cc: Likewise. * update-copyright.sh: Likewise. Signed-off-by: Dodji Seketeli <dodji@redhat.com>
180 lines
4.2 KiB
C++
180 lines
4.2 KiB
C++
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
|
// -*- Mode: C++ -*-
|
|
//
|
|
// Copyright (C) 2013-2022 Red Hat, Inc.
|
|
|
|
#include <string>
|
|
#include <fstream>
|
|
#include <iostream>
|
|
#include <cstdlib>
|
|
#include "abg-dwarf-reader.h"
|
|
#include "test-utils.h"
|
|
|
|
using std::string;
|
|
using std::ofstream;
|
|
using std::cerr;
|
|
using std::cout;
|
|
|
|
///@file
|
|
///
|
|
/// This example shows how to walk the Internal Representation (IR)
|
|
/// graph of the ABI of a binary (called an ABI Corpus) and perform
|
|
/// actions on each node of the graph.
|
|
///
|
|
/// Basically, one has to define a "visitor" which carries member
|
|
/// functions that are called during the traversal of the graph.
|
|
///
|
|
/// On the visitor, there is potentially one member function pair per
|
|
/// type of node traversed. Each time a given node is visited, the
|
|
/// corresponding member function pair is called by the traversal
|
|
/// machinery. In other words, the visitor is notified each time a
|
|
/// node is traversed.
|
|
///
|
|
/// To define a visitor, one has to create a type which implements
|
|
/// (inherits) the abigail::ir_node_visitor interface. The visitor
|
|
/// must have a pair of node_begin() and node_end() function per type
|
|
/// of node that we wish to be notified for.
|
|
///
|
|
/// Once the visitor is defined, we can load an elf file and build an
|
|
/// ABI corpus out of it by using the
|
|
/// libabigail::dwarf_reader::read_corpus_from_elf() function, for
|
|
/// instance.
|
|
///
|
|
/// Then we enumerate the translation units comprised in
|
|
/// that ABI corpus and we invoke their "traverse()" method, using
|
|
/// and instance of the visitor that we just defined.
|
|
///
|
|
/// Enjoy!
|
|
|
|
struct name_printing_visitor : public abigail::ir_node_visitor
|
|
{
|
|
unsigned level_;
|
|
|
|
name_printing_visitor()
|
|
: level_()
|
|
{
|
|
// Using this visitor, the IR walker will visit each type only
|
|
// once.
|
|
allow_visiting_already_visited_type_node(false);
|
|
}
|
|
|
|
void
|
|
build_level_prefix(string& str)
|
|
{
|
|
str.clear();
|
|
for (unsigned i = 0; i < level_; ++i)
|
|
str += ' ';
|
|
}
|
|
|
|
string
|
|
build_level_prefix()
|
|
{
|
|
string prefix;
|
|
build_level_prefix(prefix);
|
|
return prefix;
|
|
}
|
|
|
|
bool
|
|
visit_begin(abigail::namespace_decl* ns)
|
|
{
|
|
string prefix = build_level_prefix();
|
|
|
|
cout << prefix << ns->get_pretty_representation() << "\n"
|
|
<< prefix << "{\n";
|
|
++level_;
|
|
return true;
|
|
}
|
|
|
|
bool
|
|
visit_end(abigail::namespace_decl*)
|
|
{
|
|
string prefix = build_level_prefix();
|
|
cout << prefix << "}\n";
|
|
--level_;
|
|
return true;
|
|
}
|
|
|
|
bool
|
|
visit_begin(abigail::class_decl* klass)
|
|
{
|
|
string prefix = build_level_prefix();
|
|
|
|
cout << prefix << klass->get_pretty_representation() << "\n"
|
|
<< prefix << "{\n";
|
|
++level_;
|
|
return true;
|
|
}
|
|
|
|
bool
|
|
visit_end(abigail::class_decl*)
|
|
{
|
|
string prefix = build_level_prefix();
|
|
cout << prefix << "}\n";
|
|
--level_;
|
|
return true;
|
|
}
|
|
|
|
bool
|
|
visit_begin(abigail::function_decl* f)
|
|
{
|
|
string prefix = build_level_prefix();
|
|
cout << prefix << f->get_pretty_representation() << "\n";
|
|
++level_;
|
|
return true;
|
|
}
|
|
|
|
bool
|
|
visit_end(abigail::function_decl*)
|
|
{
|
|
--level_;
|
|
return true;
|
|
}
|
|
|
|
bool
|
|
visit_begin(abigail::var_decl* v)
|
|
{
|
|
string prefix = build_level_prefix();
|
|
cout << prefix << v->get_pretty_representation() << "\n";
|
|
++level_;
|
|
return true;
|
|
}
|
|
|
|
bool
|
|
visit_end(abigail::var_decl*)
|
|
{
|
|
--level_;
|
|
return true;
|
|
}
|
|
};
|
|
|
|
int
|
|
main(int argc, char **argv)
|
|
{
|
|
if (argc < 2)
|
|
return 0;
|
|
|
|
string file_name = argv[1];
|
|
|
|
abigail::ir::environment_sptr env(new abigail::ir::environment);
|
|
abigail::corpus_sptr c;
|
|
abigail::elf_reader::status status = abigail::elf_reader::STATUS_OK;
|
|
std::vector<char**> di_roots;
|
|
if (!(c = abigail::dwarf_reader::read_corpus_from_elf(file_name, di_roots,
|
|
env.get(),
|
|
/*load_all_type=*/false,
|
|
status)))
|
|
{
|
|
cerr << "failed to read " << file_name << "\n";
|
|
return 1;
|
|
}
|
|
|
|
name_printing_visitor v;
|
|
// Now traverse each translation unit of the corpus using our
|
|
// instance of name_printing_visitor
|
|
for (abigail::ir::translation_units::const_iterator tu_iterator =
|
|
c->get_translation_units().begin();
|
|
tu_iterator != c->get_translation_units().end();
|
|
++tu_iterator)
|
|
(*tu_iterator)->traverse(v);
|
|
}
|