mirror of
git://sourceware.org/git/libabigail.git
synced 2025-01-17 06:41:18 +00:00
18166 - Abidiff fails with internal on Libtirpc ABI in XML format
It turns out the support for reading the native libabigail XML format is falling short since the work on (late) type canonicalizing. The type ID -> XML node map is wrongly considered as being per corpus data while it should be per translation unit data; type IDs are unique only for a given translation unit. The code to walk a XML sub-tree to perform the ID -> node mapping is wrongly walking all the XML node *after* the sub-tree node too; that is, it walks the entire corpus file starting from the XML sub-tree node it's given. It should only walk the sub-tree it's given. These two issues normally solve the crash reported here. But then there are other related issues too. The native XML format reader doesn't populate the set of exported declarations for the current corpus it's building. This patch addresses all these issues and makes tests/test-abidiff.cc supports corpus files for a new regression test. * src/abg-reader.cc (read_context::m_exported_decls_builder_): New data member. (read_context::read_context): Initialize it. (read_context::{type_is_from_translation_unit, get_exported_decls_builder, set_exported_decls_builder, maybe_add_fn_to_exported_decls, maybe_add_fn_to_exported_decls, type_id_new_in_translation_unit}): New member functions. (read_context::clear_per_translation_unit_data): Clear id->xml node map here ... (read_context::clear_per_corpus_data): ... not here. (read_context::walk_xml_node_to_map_type_ids): Only walk the sub-tree we are asked to walk. (read_translation_unit_from_input): Cleanup. (read_corpus_from_input): Wire populating of exported declarations of the current corpus. (build_function_decl, build_var_decl): Populate exported declarations of the current corpus here. (build_type_decl, build_qualified_type_decl) (build_pointer_type_def, build_reference_type_def) (build_array_type_def, build_enum_type_decl, build_type_decl) (build_template_tparameter): Adjust assert on ID to make sure it's the first type it's being defined in the current translation unit. * tests/data/test-abidiff/test-corpus0-report0.txt: New test reference output. * tests/data/test-abidiff/test-corpus0-v{0,1}.so.abi: New test input. * tests/test-abidiff.cc (specs): Add the test inputs above to the list of inputs over which to run the test harness. (main): Support reading corpora too, as this test harness was reading just translation units before. (tests/data/Makefile.am): Add test material above to source distribution. Signed-off-by: Dodji Seketeli <dodji@redhat.com>
This commit is contained in:
parent
9dbd3bb90a
commit
b9fa3b7fd1
@ -102,6 +102,7 @@ private:
|
||||
xml::reader_sptr m_reader;
|
||||
deque<shared_ptr<decl_base> > m_decls_stack;
|
||||
corpus_sptr m_corpus;
|
||||
corpus::exported_decls_builder* m_exported_decls_builder_;
|
||||
|
||||
public:
|
||||
read_context(xml::reader_sptr reader) : m_reader(reader)
|
||||
@ -322,6 +323,26 @@ public:
|
||||
return 0;
|
||||
}
|
||||
|
||||
/// Test if a given type is from the current translation unit.
|
||||
///
|
||||
/// @param type the type to consider.
|
||||
///
|
||||
/// @return true iff the type is from the current translation unit.
|
||||
bool
|
||||
type_is_from_translation_unit(type_base_sptr type)
|
||||
{
|
||||
decl_base_sptr d = get_type_declaration(type);
|
||||
if (d)
|
||||
return (ir::get_translation_unit(d) == get_translation_unit());
|
||||
else
|
||||
// TODO: This is likely a function type as these don't have
|
||||
// declarations. For now we don't support lookup function type
|
||||
// from translation units. It'll be supported in a subsequent
|
||||
// patch. At that time we should be able to update this code
|
||||
// here.
|
||||
return false;
|
||||
}
|
||||
|
||||
void
|
||||
push_decl(decl_base_sptr d)
|
||||
{
|
||||
@ -581,12 +602,59 @@ public:
|
||||
set_corpus(corpus_sptr c)
|
||||
{m_corpus = c;}
|
||||
|
||||
/// Getter for the object that determines if a given declaration
|
||||
/// ought to be put in the set of exported decls of the current
|
||||
/// corpus.
|
||||
///
|
||||
/// @return the exported decls builder.
|
||||
corpus::exported_decls_builder*
|
||||
get_exported_decls_builder()
|
||||
{return m_exported_decls_builder_;}
|
||||
|
||||
/// Setter for the object that determines if a given declaration
|
||||
/// ought to be put in the set of exported decls of the current
|
||||
/// corpus.
|
||||
///
|
||||
/// @param d the new exported decls builder.
|
||||
///
|
||||
/// @return the exported decls builder.
|
||||
void
|
||||
set_exported_decls_builder(corpus::exported_decls_builder* d)
|
||||
{m_exported_decls_builder_ = d;}
|
||||
|
||||
/// Add a given function to the set of exported functions of the
|
||||
/// current corpus, if the function satisfies the different
|
||||
/// constraints requirements.
|
||||
///
|
||||
/// @param fn the function to consider.
|
||||
void
|
||||
maybe_add_fn_to_exported_decls(function_decl* fn)
|
||||
{
|
||||
if (fn)
|
||||
if (corpus::exported_decls_builder* b = get_exported_decls_builder())
|
||||
b->maybe_add_fn_to_exported_fns(fn);
|
||||
}
|
||||
|
||||
/// Add a given variable to the set of exported functions of the
|
||||
/// current corpus, if the function satisfies the different
|
||||
/// constraints requirements.
|
||||
///
|
||||
/// @param var the variable to consider.
|
||||
void
|
||||
maybe_add_var_to_exported_decls(var_decl* var)
|
||||
{
|
||||
if (var)
|
||||
if (corpus::exported_decls_builder* b = get_exported_decls_builder())
|
||||
b->maybe_add_var_to_exported_vars(var);
|
||||
}
|
||||
|
||||
/// Clear all the data that must absolutely be cleared at the end of
|
||||
/// the parsing of a translation unit.
|
||||
void
|
||||
clear_per_translation_unit_data()
|
||||
{
|
||||
clear_xml_node_decl_map();
|
||||
clear_id_xml_node_map();
|
||||
clear_decls_stack();
|
||||
}
|
||||
|
||||
@ -595,7 +663,6 @@ public:
|
||||
void
|
||||
clear_per_corpus_data()
|
||||
{
|
||||
clear_id_xml_node_map();
|
||||
clear_type_map();
|
||||
clear_types_to_canonicalize();
|
||||
}
|
||||
@ -654,6 +721,19 @@ public:
|
||||
canonicalize(*i);
|
||||
}
|
||||
|
||||
/// Test if a type ID is new in the current translation unit.
|
||||
///
|
||||
/// @param id the translation unit ID to test for.
|
||||
///
|
||||
/// @return true iff the type ID is new in the current translation
|
||||
/// unit.
|
||||
bool
|
||||
type_id_new_in_translation_unit(const string& id)
|
||||
{
|
||||
type_base_sptr t = get_type_decl(id);
|
||||
return !t || !type_is_from_translation_unit(t);
|
||||
}
|
||||
|
||||
};// end class read_context
|
||||
|
||||
static int advance_cursor(read_context&);
|
||||
@ -889,14 +969,19 @@ advance_cursor(read_context& ctxt)
|
||||
/// Walk an entire XML sub-tree to build a map where the key is the
|
||||
/// the value of the 'id' attribute (for type definitions) and the key
|
||||
/// is the xml node containing the 'id' attribute.
|
||||
///
|
||||
/// @param ctxt the context of the reader.
|
||||
///
|
||||
/// @param node the XML sub-tree node to walk. It must be an element
|
||||
/// node.
|
||||
static void
|
||||
walk_xml_node_to_map_type_ids(read_context& ctxt,
|
||||
xmlNodePtr node)
|
||||
{
|
||||
for (xmlNodePtr n = node; n; n = n->next)
|
||||
{
|
||||
if (n->type != XML_ELEMENT_NODE)
|
||||
continue;
|
||||
xmlNodePtr n = node;
|
||||
|
||||
if (!n || n->type != XML_ELEMENT_NODE)
|
||||
return;
|
||||
|
||||
if (xml_char_sptr s = XML_NODE_GET_ATTRIBUTE(n, "id"))
|
||||
{
|
||||
@ -904,8 +989,8 @@ walk_xml_node_to_map_type_ids(read_context& ctxt,
|
||||
ctxt.map_id_and_node(id, n);
|
||||
}
|
||||
|
||||
walk_xml_node_to_map_type_ids(ctxt, n->children);
|
||||
}
|
||||
for (n = n->children; n; n = n->next)
|
||||
walk_xml_node_to_map_type_ids(ctxt, n);
|
||||
}
|
||||
|
||||
/// Parse the input XML document containing a translation_unit,
|
||||
@ -939,8 +1024,6 @@ read_translation_unit_from_input(read_context& ctxt,
|
||||
if (!node)
|
||||
return false;
|
||||
|
||||
walk_xml_node_to_map_type_ids(ctxt, node);
|
||||
|
||||
xml::xml_char_sptr addrsize_str =
|
||||
XML_NODE_GET_ATTRIBUTE(node, "address-size");
|
||||
if (addrsize_str)
|
||||
@ -958,6 +1041,8 @@ read_translation_unit_from_input(read_context& ctxt,
|
||||
ctxt.push_decl(tu.get_global_scope());
|
||||
ctxt.map_xml_node_to_decl(node, tu.get_global_scope());
|
||||
|
||||
walk_xml_node_to_map_type_ids(ctxt, node);
|
||||
|
||||
for (xmlNodePtr n = node->children; n; n = n->next)
|
||||
{
|
||||
if (n->type != XML_ELEMENT_NODE)
|
||||
@ -1142,6 +1227,7 @@ read_corpus_from_input(read_context& ctxt)
|
||||
ctxt.clear_per_corpus_data();
|
||||
|
||||
corpus& corp = *ctxt.get_corpus();
|
||||
ctxt.set_exported_decls_builder(corp.get_exported_decls_builder().get());
|
||||
|
||||
xml::xml_char_sptr path_str = XML_READER_GET_ATTRIBUTE(reader, "path");
|
||||
if (path_str)
|
||||
@ -2087,6 +2173,8 @@ build_function_decl(read_context& ctxt,
|
||||
fn_decl->set_type(fn_type);
|
||||
ctxt.maybe_canonicalize_type(fn_type);
|
||||
|
||||
ctxt.maybe_add_fn_to_exported_decls(fn_decl.get());
|
||||
|
||||
return fn_decl;
|
||||
}
|
||||
|
||||
@ -2146,6 +2234,8 @@ build_var_decl(read_context& ctxt,
|
||||
if (decl->get_symbol() && decl->get_symbol()->is_public())
|
||||
decl->set_is_in_public_symbol_table(true);
|
||||
|
||||
ctxt.maybe_add_var_to_exported_decls(decl.get());
|
||||
|
||||
return decl;
|
||||
}
|
||||
|
||||
@ -2184,7 +2274,7 @@ build_type_decl(read_context& ctxt,
|
||||
string id;
|
||||
if (xml_char_sptr s = XML_NODE_GET_ATTRIBUTE(node, "id"))
|
||||
id = CHAR_STR(s);
|
||||
assert(!id.empty());
|
||||
assert(!id.empty() && ctxt.type_id_new_in_translation_unit(id));
|
||||
|
||||
size_t size_in_bits= 0;
|
||||
if (xml_char_sptr s = XML_NODE_GET_ATTRIBUTE(node, "size-in-bits"))
|
||||
@ -2298,7 +2388,7 @@ build_qualified_type_decl(read_context& ctxt,
|
||||
location loc;
|
||||
read_location(ctxt, node, loc);
|
||||
|
||||
assert(!id.empty());
|
||||
assert(!id.empty() && ctxt.type_id_new_in_translation_unit(id));
|
||||
|
||||
qualified_type_def_sptr decl;
|
||||
|
||||
@ -2379,7 +2469,7 @@ build_pointer_type_def(read_context& ctxt,
|
||||
string id;
|
||||
if (xml_char_sptr s = XML_NODE_GET_ATTRIBUTE(node, "id"))
|
||||
id = CHAR_STR(s);
|
||||
assert(!id.empty());
|
||||
assert(!id.empty() && ctxt.type_id_new_in_translation_unit(id));
|
||||
if (type_base_sptr d = ctxt.get_type_decl(id))
|
||||
{
|
||||
pointer_type_def_sptr ty = is_pointer_type(d);
|
||||
@ -2467,7 +2557,7 @@ build_reference_type_def(read_context& ctxt,
|
||||
string id;
|
||||
if (xml_char_sptr s = XML_NODE_GET_ATTRIBUTE(node, "id"))
|
||||
id = CHAR_STR(s);
|
||||
assert(!id.empty());
|
||||
assert(!id.empty() && ctxt.type_id_new_in_translation_unit(id));
|
||||
|
||||
if (type_base_sptr d = ctxt.get_type_decl(id))
|
||||
{
|
||||
@ -2611,7 +2701,7 @@ build_array_type_def(read_context& ctxt,
|
||||
string id;
|
||||
if (xml_char_sptr s = XML_NODE_GET_ATTRIBUTE(node, "id"))
|
||||
id = CHAR_STR(s);
|
||||
assert(!id.empty());
|
||||
assert(!id.empty() && ctxt.type_id_new_in_translation_unit(id));
|
||||
|
||||
if (type_base_sptr d = ctxt.get_type_decl(id))
|
||||
{
|
||||
@ -2705,7 +2795,7 @@ build_enum_type_decl(read_context& ctxt,
|
||||
if (xml_char_sptr s = XML_NODE_GET_ATTRIBUTE(node, "id"))
|
||||
id = CHAR_STR(s);
|
||||
|
||||
assert(!id.empty() && !ctxt.get_type_decl(id));
|
||||
assert(!id.empty() && ctxt.type_id_new_in_translation_unit(id));
|
||||
|
||||
string base_type_id;
|
||||
enum_type_decl::enumerators enums;
|
||||
@ -2784,7 +2874,7 @@ build_typedef_decl(read_context& ctxt,
|
||||
string id;
|
||||
if (xml_char_sptr s = XML_NODE_GET_ATTRIBUTE(node, "id"))
|
||||
id = CHAR_STR(s);
|
||||
assert(!id.empty());
|
||||
assert(!id.empty() && ctxt.type_id_new_in_translation_unit(id));
|
||||
|
||||
string name;
|
||||
if (xml_char_sptr s = XML_NODE_GET_ATTRIBUTE(node, "name"))
|
||||
@ -3453,7 +3543,7 @@ build_template_tparameter(read_context& ctxt,
|
||||
if (xml_char_sptr s = XML_NODE_GET_ATTRIBUTE(node, "id"))
|
||||
id = CHAR_STR(s);
|
||||
// Bail out if a type with the same ID already exists.
|
||||
assert(!id.empty() && !ctxt.get_type_decl(id));
|
||||
assert(!id.empty() && !ctxt.type_id_new_in_translation_unit(id));
|
||||
|
||||
string type_id;
|
||||
if (xml_char_sptr s = XML_NODE_GET_ATTRIBUTE(node, "type-id"))
|
||||
|
@ -67,6 +67,9 @@ test-abidiff/test-struct1-report.txt \
|
||||
test-abidiff/test-var0-v0.cc.bi \
|
||||
test-abidiff/test-var0-v1.cc.bi \
|
||||
test-abidiff/test-var0-report.txt \
|
||||
test-abidiff/test-corpus0-v0.so.abi \
|
||||
test-abidiff/test-corpus0-v1.so.abi \
|
||||
test-abidiff/test-corpus0-report0.txt \
|
||||
\
|
||||
test-diff-dwarf/test0-v0.cc \
|
||||
test-diff-dwarf/test0-v0.o \
|
||||
|
3
tests/data/test-abidiff/test-corpus0-report0.txt
Normal file
3
tests/data/test-abidiff/test-corpus0-report0.txt
Normal file
@ -0,0 +1,3 @@
|
||||
Functions changes summary: 0 Removed, 0 Changed, 0 Added function
|
||||
Variables changes summary: 0 Removed, 0 Changed, 0 Added variable
|
||||
|
16411
tests/data/test-abidiff/test-corpus0-v0.so.abi
Normal file
16411
tests/data/test-abidiff/test-corpus0-v0.so.abi
Normal file
File diff suppressed because it is too large
Load Diff
14938
tests/data/test-abidiff/test-corpus0-v1.so.abi
Normal file
14938
tests/data/test-abidiff/test-corpus0-v1.so.abi
Normal file
File diff suppressed because it is too large
Load Diff
@ -37,6 +37,7 @@
|
||||
#include "abg-reader.h"
|
||||
#include "test-utils.h"
|
||||
#include "abg-comparison.h"
|
||||
#include "abg-corpus.h"
|
||||
|
||||
using std::string;
|
||||
using std::ofstream;
|
||||
@ -88,6 +89,12 @@ static InOutSpec specs[] =
|
||||
"data/test-abidiff/test-var0-report.txt",
|
||||
"output/test-abidiff/test-var0-report.txt"
|
||||
},
|
||||
{
|
||||
"data/test-abidiff/test-corpus0-v0.so.abi",
|
||||
"data/test-abidiff/test-corpus0-v1.so.abi",
|
||||
"data/test-abidiff/test-corpus0-report0.txt",
|
||||
"output/test-abidiff/test-corpus0-report0.txt"
|
||||
},
|
||||
// This should be the last entry.
|
||||
{0, 0, 0, 0}
|
||||
};
|
||||
@ -98,8 +105,15 @@ static InOutSpec specs[] =
|
||||
using std::string;
|
||||
using std::cerr;
|
||||
using std::ofstream;
|
||||
using abigail::tools_utils::file_type;
|
||||
using abigail::tools_utils::check_file;
|
||||
using abigail::tools_utils::guess_file_type;
|
||||
using abigail::corpus_sptr;
|
||||
using abigail::translation_unit;
|
||||
using abigail::translation_unit_sptr;
|
||||
using abigail::xml_reader::read_translation_unit_from_file;
|
||||
using abigail::xml_reader::read_corpus_from_native_xml_file;
|
||||
using abigail::comparison::corpus_diff_sptr;
|
||||
using abigail::comparison::translation_unit_diff_sptr;
|
||||
using abigail::comparison::compute_diff;
|
||||
|
||||
@ -134,25 +148,42 @@ main(int, char*[])
|
||||
continue;
|
||||
}
|
||||
|
||||
translation_unit_sptr tu1 =
|
||||
abigail::xml_reader::read_translation_unit_from_file(first_in_path);
|
||||
if (!tu1)
|
||||
translation_unit_sptr tu1, tu2;
|
||||
corpus_sptr corpus1, corpus2;
|
||||
file_type t = guess_file_type(first_in_path);
|
||||
if (t == abigail::tools_utils::FILE_TYPE_NATIVE_BI)
|
||||
tu1 = read_translation_unit_from_file(first_in_path);
|
||||
else if (t == abigail::tools_utils::FILE_TYPE_XML_CORPUS)
|
||||
corpus1 = read_corpus_from_native_xml_file(first_in_path);
|
||||
else
|
||||
abort();
|
||||
if (!tu1 && !corpus1)
|
||||
{
|
||||
cerr << "failed to read " << tu1->get_path() << "\n";
|
||||
cerr << "failed to read " << first_in_path << "\n";
|
||||
is_ok = false;
|
||||
continue;
|
||||
}
|
||||
|
||||
translation_unit_sptr tu2 =
|
||||
abigail::xml_reader::read_translation_unit_from_file(second_in_path);
|
||||
if (!tu2)
|
||||
t = guess_file_type(second_in_path);
|
||||
if (t == abigail::tools_utils::FILE_TYPE_NATIVE_BI)
|
||||
tu2 = read_translation_unit_from_file(second_in_path);
|
||||
else if (t == abigail::tools_utils::FILE_TYPE_XML_CORPUS)
|
||||
corpus2 = read_corpus_from_native_xml_file(second_in_path);
|
||||
else
|
||||
abort();
|
||||
if (!tu2 && !corpus2)
|
||||
{
|
||||
cerr << "failed to read " << tu1->get_path() << "\n";
|
||||
cerr << "failed to read " << second_in_path << "\n";
|
||||
is_ok = false;
|
||||
continue;
|
||||
}
|
||||
|
||||
translation_unit_diff_sptr d = compute_diff(tu1, tu2);
|
||||
translation_unit_diff_sptr d1;
|
||||
corpus_diff_sptr d2;
|
||||
if (tu1)
|
||||
d1= compute_diff(tu1, tu2);
|
||||
else
|
||||
d2 = compute_diff(corpus1, corpus2);
|
||||
ofstream of(out_path.c_str(), std::ios_base::trunc);
|
||||
if (!of.is_open())
|
||||
{
|
||||
@ -161,7 +192,10 @@ main(int, char*[])
|
||||
continue;
|
||||
}
|
||||
|
||||
d->report(of);
|
||||
if (d1)
|
||||
d1->report(of);
|
||||
else
|
||||
d2->report(of);
|
||||
of.close();
|
||||
|
||||
string cmd = "diff -u " + ref_diff_path + " " + out_path;
|
||||
|
Loading…
Reference in New Issue
Block a user