diff --git a/gcc/cgraph.h b/gcc/cgraph.h index 47b5c02d770..4a1f89920f5 100644 --- a/gcc/cgraph.h +++ b/gcc/cgraph.h @@ -319,6 +319,9 @@ public: /* Return node that alias is aliasing. */ inline symtab_node *get_alias_target (void); + /* Return DECL that alias is aliasing. */ + inline tree get_alias_target_tree (); + /* Set section for symbol and its aliases. */ void set_section (const char *section); @@ -2665,6 +2668,17 @@ symtab_node::get_alias_target (void) return ref->referred; } +/* Return the DECL (or identifier) that alias is aliasing. Unlike the above, + this works whether or not the alias has been analyzed already. */ + +inline tree +symtab_node::get_alias_target_tree () +{ + if (alias_target) + return alias_target; + return get_alias_target ()->decl; +} + /* Return next reachable static symbol with initializer after the node. */ inline symtab_node * diff --git a/gcc/cgraphunit.c b/gcc/cgraphunit.c index b401f0817a3..1c74cee69ac 100644 --- a/gcc/cgraphunit.c +++ b/gcc/cgraphunit.c @@ -859,8 +859,11 @@ process_function_and_variable_attributes (cgraph_node *first, if (node->alias && lookup_attribute ("flatten", DECL_ATTRIBUTES (decl))) { - warning_at (DECL_SOURCE_LOCATION (node->decl), OPT_Wattributes, - "% attribute is ignored on aliases"); + tree tdecl = node->get_alias_target_tree (); + if (!tdecl || !DECL_P (tdecl) + || !lookup_attribute ("flatten", DECL_ATTRIBUTES (tdecl))) + warning_at (DECL_SOURCE_LOCATION (decl), OPT_Wattributes, + "% attribute is ignored on aliases"); } if (DECL_PRESERVE_P (decl)) node->mark_force_output (); diff --git a/gcc/testsuite/g++.dg/ext/attr-flatten1.C b/gcc/testsuite/g++.dg/ext/attr-flatten1.C new file mode 100644 index 00000000000..5bcbfb6f4aa --- /dev/null +++ b/gcc/testsuite/g++.dg/ext/attr-flatten1.C @@ -0,0 +1,9 @@ +// PR c++/96078 +// { dg-do compile { target c++11 } } + +struct A { + [[gnu::flatten]] A() {} + [[gnu::flatten]] ~A() {} +}; + +A a;