Make type_has_non_canonicalized_subtype() tighter

type_has_non_canonicalized_subtype() gives up too quickly.

For instance, suppose it's looking a type 'foo'.  If foo has no
canonicalized type yet and has a data member which type is foo* (for
instance), then type_has_non_canonicalized_subtype() just sees that
type 'foo*' has no canonicalized type, and so it returns, saying that
he found a non-canonicalized subtype for foo.

In that case though, what type_has_non_canonicalized_subtype() should
do is detect that foo* is a pointer to foo itself, so it shouldn't
count as a non-canonicalized sub-type.  It should keep going and look
for other meaningful non-canonicalized sub-types.

And this what this patch does.  It changes the sub-type walker that
type_has_non_canonicalized_subtype() uses, so that

   - it doesn't flag sub-types that refer to the type we are looking
     at as non-canonicalized sub-types.  This is for sub-types that
     are combinations of pointers, references and typedefs.

   - it doesn't consider sub-types of member functions of the type we
     are looking at, unless that member function is virtual.

The result is that more types are canonicalized early during DWARF
reading, and so there are less types to store on the side for late
canonicalization.  This can have a big impact on, e.g, C++ libraries
with tens of thousands of types.

	* include/abg-fwd.h (is_typedef, is_pointer_type)
	(is_reference_type): Declare new overloads.
	(peel_typedef_type): Renamed get_typedef_underlying_type into
	this.
	(peel_pointer_type, peel_reference_type)
	(peel_typedef_pointer_or_reference_type): Declare new functions.
	* src/abg-ir.cc (peel_typedef_type): Renamed
	get_typedef_underlying_type into this.
	(is_typedef, is_pointer_type, is_reference_type): Define new
	overloads.
	(peel_pointer_type, peel_reference_type)
	(peel_typedef_pointer_or_reference_type): Define new functions.
	(non_canonicalized_subtype_detector::has_non_canonical_type_):
	Make the type of this data member be a type_base*, not a bool.
	This is so that we can return the first non-canonicalized subtype
	of the type we are looking at.
	(non_canonicalized_subtype_detector::non_canonicalized_subtype_detector):
	Adjust the data member initialization.
	(non_canonicalized_subtype_detector::visit_begin): Add an overload
	for function_decl*, to avoid looking into non-virtual member
	functions.
	In the overload for type_base*, peel typedefs, pointers and
	reference of each sub-type that has no canonical type, to see if
	refers to the type we are actually walking.  If yes, then keep
	going.
	(type_has_non_canonicalized_subtype): Return the non-canonicalized
	sub-type found.
	* src/abg-comparison.cc (type_suppression::suppresses_diff):
	Adjust for the get_typedef_underlying_type -> peel_typedef_type
	renaming.

Signed-off-by: Dodji Seketeli <dodji@redhat.com>
This commit is contained in:
Dodji Seketeli 2015-08-19 16:15:28 +02:00
parent 39b2e8b7d5
commit bd161caa52
3 changed files with 299 additions and 12 deletions

View File

@ -204,6 +204,12 @@ is_typedef(const shared_ptr<type_base>);
shared_ptr<typedef_decl>
is_typedef(const shared_ptr<decl_base>);
const typedef_decl*
is_typedef(const type_base*);
typedef_decl*
is_typedef(type_base*);
shared_ptr<enum_type_decl>
is_enum_type(const shared_ptr<type_base>&);
@ -228,12 +234,18 @@ is_compatible_with_class_type(const shared_ptr<type_base>);
pointer_type_def*
is_pointer_type(type_base*);
const pointer_type_def*
is_pointer_type(const type_base*);
shared_ptr<pointer_type_def>
is_pointer_type(const shared_ptr<type_base>);
reference_type_def*
is_reference_type(type_base*);
const reference_type_def*
is_reference_type(const type_base*);
shared_ptr<reference_type_def>
is_reference_type(const shared_ptr<type_base>);
@ -439,7 +451,28 @@ shared_ptr<type_base>
strip_typedef(const shared_ptr<type_base>);
shared_ptr<type_base>
get_typedef_underlying_type(const shared_ptr<type_base>&);
peel_typedef_type(const shared_ptr<type_base>&);
const type_base*
peel_typedef_type(const type_base*);
shared_ptr<type_base>
peel_pointer_type(const shared_ptr<type_base>&);
const type_base*
peel_pointer_type(const type_base*);
shared_ptr<type_base>
peel_reference_type(const shared_ptr<type_base>&);
const type_base*
peel_reference_type(const type_base*);
shared_ptr<type_base>
peel_typedef_pointer_or_reference_type(const shared_ptr<type_base>);
type_base*
peel_typedef_pointer_or_reference_type(const type_base*);
string
get_name(const shared_ptr<type_or_decl_base>&,
@ -600,7 +633,7 @@ type_or_void(const shared_ptr<type_base>);
shared_ptr<type_base>
canonicalize(shared_ptr<type_base>);
bool
type_base*
type_has_non_canonicalized_subtype(shared_ptr<type_base> t);
bool

View File

@ -963,8 +963,8 @@ type_suppression::suppresses_diff(const diff* diff) const
if (!suppresses_type(ft, d->context())
&& !suppresses_type(st, d->context()))
{
ft = get_typedef_underlying_type(ft);
st = get_typedef_underlying_type(st);
ft = peel_typedef_type(ft);
st = peel_typedef_type(st);
if (!suppresses_type(ft, d->context())
&& !suppresses_type(st, d->context()))

View File

@ -3007,17 +3007,201 @@ strip_typedef(const type_base_sptr type)
///
/// @return the leaf underlying type node of a @p type.
type_base_sptr
get_typedef_underlying_type(const type_base_sptr& type)
peel_typedef_type(const type_base_sptr& type)
{
typedef_decl_sptr t = is_typedef(type);
if (!t)
return type;
if (is_typedef(t->get_underlying_type()))
return get_typedef_underlying_type(t->get_underlying_type());
return peel_typedef_type(t->get_underlying_type());
return t->get_underlying_type();
}
/// Return the leaf underlying type node of a @ref typedef_decl node.
///
/// If the underlying type of a @ref typedef_decl node is itself a
/// @ref typedef_decl node, then recursively look at the underlying
/// type nodes to get the first one that is not a a @ref typedef_decl
/// node. This is what a leaf underlying type node means.
///
/// Otherwise, if the underlying type node of @ref typedef_decl is
/// *NOT* a @ref typedef_decl node, then just return the underlying
/// type node.
///
/// And if the type node considered is not a @ref typedef_decl node,
/// then just return it.
///
/// @return the leaf underlying type node of a @p type.
const type_base*
peel_typedef_type(const type_base* type)
{
const typedef_decl* t = is_typedef(type);
if (!t)
return t;
return peel_typedef_type(t->get_underlying_type()).get();
}
/// Return the leaf pointed-to type node of a @ref pointer_type_def
/// node.
///
/// If the pointed-to type of a @ref pointer_type_def node is itself a
/// @ref pointer_type_def node, then recursively look at the
/// pointed-to type nodes to get the first one that is not a a @ref
/// pointer_type_def node. This is what a leaf pointed-to type node
/// means.
///
/// Otherwise, if the pointed-to type node of @ref pointer_type_def is
/// *NOT* a @ref pointer_type_def node, then just return the
/// pointed-to type node.
///
/// And if the type node considered is not a @ref pointer_type_def
/// node, then just return it.
///
/// @return the leaf pointed-to type node of a @p type.
type_base_sptr
peel_pointer_type(const type_base_sptr& type)
{
pointer_type_def_sptr t = is_pointer_type(type);
if (!t)
return type;
if (is_pointer_type(t->get_pointed_to_type()))
return peel_pointer_type(t->get_pointed_to_type());
return t->get_pointed_to_type();
}
/// Return the leaf pointed-to type node of a @ref pointer_type_def
/// node.
///
/// If the pointed-to type of a @ref pointer_type_def node is itself a
/// @ref pointer_type_def node, then recursively look at the
/// pointed-to type nodes to get the first one that is not a a @ref
/// pointer_type_def node. This is what a leaf pointed-to type node
/// means.
///
/// Otherwise, if the pointed-to type node of @ref pointer_type_def is
/// *NOT* a @ref pointer_type_def node, then just return the
/// pointed-to type node.
///
/// And if the type node considered is not a @ref pointer_type_def
/// node, then just return it.
///
/// @return the leaf pointed-to type node of a @p type.
const type_base*
peel_pointer_type(const type_base* type)
{
const pointer_type_def* t = is_pointer_type(type);
if (!t)
return type;
return peel_pointer_type(t->get_pointed_to_type()).get();
}
/// Return the leaf pointed-to type node of a @ref reference_type_def
/// node.
///
/// If the pointed-to type of a @ref reference_type_def node is itself
/// a @ref reference_type_def node, then recursively look at the
/// pointed-to type nodes to get the first one that is not a a @ref
/// reference_type_def node. This is what a leaf pointed-to type node
/// means.
///
/// Otherwise, if the pointed-to type node of @ref reference_type_def
/// is *NOT* a @ref reference_type_def node, then just return the
/// pointed-to type node.
///
/// And if the type node considered is not a @ref reference_type_def
/// node, then just return it.
///
/// @return the leaf pointed-to type node of a @p type.
type_base_sptr
peel_reference_type(const type_base_sptr& type)
{
reference_type_def_sptr t = is_reference_type(type);
if (!t)
return type;
if (is_reference_type(t->get_pointed_to_type()))
return peel_reference_type(t->get_pointed_to_type());
return t->get_pointed_to_type();
}
/// Return the leaf pointed-to type node of a @ref reference_type_def
/// node.
///
/// If the pointed-to type of a @ref reference_type_def node is itself
/// a @ref reference_type_def node, then recursively look at the
/// pointed-to type nodes to get the first one that is not a a @ref
/// reference_type_def node. This is what a leaf pointed-to type node
/// means.
///
/// Otherwise, if the pointed-to type node of @ref reference_type_def
/// is *NOT* a @ref reference_type_def node, then just return the
/// pointed-to type node.
///
/// And if the type node considered is not a @ref reference_type_def
/// node, then just return it.
///
/// @return the leaf pointed-to type node of a @p type.
const type_base*
peel_reference_type(const type_base* type)
{
const reference_type_def* t = is_reference_type(type);
if (!t)
return type;
return peel_reference_type(t->get_pointed_to_type()).get();
}
/// Return the leaf underlying or pointed-to type node of a @ref
/// typedef_decl, @ref pointer_type_def or @ref reference_type_def
/// node.
///
/// @return the leaf underlying or pointed-to type node of @p type.
type_base_sptr
peel_typedef_pointer_or_reference_type(const type_base_sptr type)
{
type_base_sptr typ = type;
while (is_typedef(type) || is_pointer_type(type) || is_reference_type(type))
{
if (typedef_decl_sptr t = is_typedef(typ))
typ = peel_typedef_type(t);
if (pointer_type_def_sptr t = is_pointer_type(typ))
typ = peel_pointer_type(t);
if (reference_type_def_sptr t = is_reference_type(typ))
typ = peel_reference_type(t);
}
return typ;
}
/// Return the leaf underlying or pointed-to type node of a @ref
/// typedef_decl, @ref pointer_type_def or @ref reference_type_def
/// node.
///
/// @return the leaf underlying or pointed-to type node of @p type.
type_base*
peel_typedef_pointer_or_reference_type(const type_base* type)
{
while (is_typedef(type) || is_pointer_type(type) || is_reference_type(type))
{
if (const typedef_decl* t = is_typedef(type))
type = peel_typedef_type(t);
if (const pointer_type_def* t = is_pointer_type(type))
type = peel_pointer_type(t);
if (const reference_type_def* t = is_reference_type(type))
type = peel_reference_type(t);
}
return const_cast<type_base*>(type);
}
/// Add a member decl to this scope. Note that user code should not
/// use this, but rather use add_decl_to_scope.
///
@ -4048,6 +4232,26 @@ typedef_decl_sptr
is_typedef(const decl_base_sptr d)
{return is_typedef(is_type(d));}
/// Test whether a type is a typedef.
///
/// @param t the declaration of the type to test for.
///
/// @return the typedef declaration of the @p t, or NULL if it's not a
/// typedef.
const typedef_decl*
is_typedef(const type_base* t)
{return dynamic_cast<const typedef_decl*>(t);}
/// Test whether a type is a typedef.
///
/// @param t the declaration of the type to test for.
///
/// @return the typedef declaration of the @p t, or NULL if it's not a
/// typedef.
typedef_decl*
is_typedef(type_base* t)
{return dynamic_cast<typedef_decl*>(t);}
/// Test if a decl is an enum_type_decl
///
/// @param d the decl to test for.
@ -4134,6 +4338,16 @@ pointer_type_def*
is_pointer_type(type_base* t)
{return dynamic_cast<pointer_type_def*>(t);}
/// Test whether a type is a pointer_type_def.
///
/// @param t the type to test.
///
/// @return the @ref pointer_type_def_sptr if @p t is a
/// pointer_type_def, null otherwise.
const pointer_type_def*
is_pointer_type(const type_base* t)
{return dynamic_cast<const pointer_type_def*>(t);}
/// Test whether a type is a pointer_type_def.
///
/// @param t the type to test.
@ -4144,10 +4358,26 @@ pointer_type_def_sptr
is_pointer_type(const type_base_sptr t)
{return dynamic_pointer_cast<pointer_type_def>(t);}
/// Test whether a type is a reference_type_def.
///
/// @param t the type to test.
///
/// @return the @ref reference_type_def_sptr if @p t is a
/// reference_type_def, null otherwise.
reference_type_def*
is_reference_type(type_base* t)
{return dynamic_cast<reference_type_def*>(t);}
/// Test whether a type is a reference_type_def.
///
/// @param t the type to test.
///
/// @return the @ref reference_type_def_sptr if @p t is a
/// reference_type_def, null otherwise.
const reference_type_def*
is_reference_type(const type_base* t)
{return dynamic_cast<const reference_type_def*>(t);}
/// Test whether a type is a reference_type_def.
///
/// @param t the type to test.
@ -11062,7 +11292,7 @@ class_tdecl::~class_tdecl()
class non_canonicalized_subtype_detector : public ir::ir_node_visitor
{
type_base* type_;
bool has_non_canonical_type_;
type_base* has_non_canonical_type_;
private:
non_canonicalized_subtype_detector();
@ -11070,7 +11300,7 @@ private:
public:
non_canonicalized_subtype_detector(type_base* type)
: type_(type),
has_non_canonical_type_(false)
has_non_canonical_type_()
{}
/// Return true if the visitor detected that there is a
@ -11078,10 +11308,22 @@ public:
///
/// @return true if the visitor detected that there is a
/// non-canonicalized sub-type.
bool
type_base*
has_non_canonical_type() const
{return has_non_canonical_type_;}
/// The intent of this visitor handler is to avoid looking into
/// sub-types of member functions of the type we are traversing.
bool
visit_begin(function_decl* f)
{
// Do not look at sub-types of non-virtual member functions.
if (is_member_function(f)
&& get_member_function_is_virtual(*f))
return false;
return true;
}
/// When visiting a sub-type, if it's *NOT* been canonicalized, set
/// the 'has_non_canonical_type' flag. And in any case, when
/// visiting a sub-type, do not visit its children nodes. So this
@ -11095,8 +11337,20 @@ public:
{
if (t != type_)
{
// If 'type_' references a type which is a combination of
// pointer to, reference to, or typedef of himself, then do
// not look in there for a sub-type which doesn't have a
// canonical type.
type_base* type = peel_typedef_pointer_or_reference_type(t);
if (get_type_name(type, true) == get_type_name(type_, true))
return true;
if (!t->get_canonical_type())
has_non_canonical_type_ = true;
// We are looking a sub-type of 'type_' which has no
// canonical type. So tada! we found one! Get out right
// now with the trophy.
has_non_canonical_type_ = t;
return false;
}
return true;
@ -11123,11 +11377,11 @@ public:
/// @param t the type which sub-types to consider.
///
/// @return true if a type has sub-types that are non-canonicalized.
bool
type_base*
type_has_non_canonicalized_subtype(type_base_sptr t)
{
if (!t)
return false;
return 0;
non_canonicalized_subtype_detector v(t.get());
t->traverse(v);