aboutsummaryrefslogtreecommitdiff
path: root/gcc
diff options
context:
space:
mode:
authorJakub Jelinek <jakub@redhat.com>2023-12-09 10:20:05 +0100
committerJakub Jelinek <jakub@redhat.com>2023-12-09 10:20:05 +0100
commit388ab03975c5c6b3c434ebb95c56c07ea8932486 (patch)
tree901288c35d4c0f9eef39f4cc3c0cbd1739a6b2d8 /gcc
parent9363d91956931bb28735bed97035b9ec965c850f (diff)
downloadgcc-388ab03975c5c6b3c434ebb95c56c07ea8932486.zip
gcc-388ab03975c5c6b3c434ebb95c56c07ea8932486.tar.gz
gcc-388ab03975c5c6b3c434ebb95c56c07ea8932486.tar.bz2
c++: Don't diagnose ignoring of attributes if all ignored attributes are attribute_ignored_p
There is another thing I wonder about: with -Wno-attributes= we are supposed to ignore the attributes altogether, but we are actually still warning about them when we emit these generic warnings about ignoring all attributes which appertain to this and that (perhaps with some exceptions we first remove from the attribute chain), like: void foo () { [[foo::bar]]; } with -Wattributes -Wno-attributes=foo::bar Shouldn't we call some helper function in cases like this and warn not when std_attrs (or how the attribute chain var is called) is non-NULL, but if it is non-NULL and contains at least one non-attribute_ignored_p attribute? I've kept warnings for cases where the C++ standard says explicitly any attributes aren't ok - "If an attribute-specifier-seq appertains to a friend declaration, that declaration shall be a definition." or https://eel.is/c++draft/dcl.type.elab#3 or https://eel.is/c++draft/temp.spec#temp.explicit-3 For some changes I haven't figured out how could I cover it in the testsuite. Note, C uses a different strategy, it has c_warn_unused_attributes function which warns about all the attributes one by one unless they are ignored (or allowed in certain position). Though that is just a single diagnostic wording, while C++ FE just warns that there are some ignored attributes and doesn't name them individually (except for namespace and using namespace) and uses different wordings in different spots. 2023-12-09 Jakub Jelinek <jakub@redhat.com> gcc/ * attribs.h (any_nonignored_attribute_p): Declare. * attribs.cc (any_nonignored_attribute_p): New function. gcc/cp/ * parser.cc (cp_parser_statement, cp_parser_expression_statement, cp_parser_declaration, cp_parser_asm_definition): Don't diagnose ignored attributes if !any_nonignored_attribute_p. * decl.cc (grokdeclarator): Likewise. * name-lookup.cc (handle_namespace_attrs, finish_using_directive): Don't diagnose ignoring of attr_ignored_p attributes. gcc/testsuite/ * g++.dg/warn/Wno-attributes-1.C: New test.
Diffstat (limited to 'gcc')
-rw-r--r--gcc/attribs.cc13
-rw-r--r--gcc/attribs.h1
-rw-r--r--gcc/cp/decl.cc3
-rw-r--r--gcc/cp/name-lookup.cc4
-rw-r--r--gcc/cp/parser.cc11
-rw-r--r--gcc/testsuite/g++.dg/warn/Wno-attributes-1.C52
6 files changed, 75 insertions, 9 deletions
diff --git a/gcc/attribs.cc b/gcc/attribs.cc
index ff4b638..776655d 100644
--- a/gcc/attribs.cc
+++ b/gcc/attribs.cc
@@ -584,6 +584,19 @@ attribute_ignored_p (const attribute_spec *const as)
return as->max_length == -2;
}
+/* Return true if the ATTRS chain contains at least one attribute which
+ is not ignored. */
+
+bool
+any_nonignored_attribute_p (tree attrs)
+{
+ for (tree attr = attrs; attr; attr = TREE_CHAIN (attr))
+ if (!attribute_ignored_p (attr))
+ return true;
+
+ return false;
+}
+
/* See whether LIST contains at least one instance of attribute ATTR
(possibly with different arguments). Return the first such attribute
if so, otherwise return null. */
diff --git a/gcc/attribs.h b/gcc/attribs.h
index fdeebff..2c615a4 100644
--- a/gcc/attribs.h
+++ b/gcc/attribs.h
@@ -48,6 +48,7 @@ extern void apply_tm_attr (tree, tree);
extern tree make_attribute (const char *, const char *, tree);
extern bool attribute_ignored_p (tree);
extern bool attribute_ignored_p (const attribute_spec *const);
+extern bool any_nonignored_attribute_p (tree);
extern struct scoped_attributes *
register_scoped_attributes (const scoped_attribute_specs &, bool = false);
diff --git a/gcc/cp/decl.cc b/gcc/cp/decl.cc
index 4b68527..b1ada1d 100644
--- a/gcc/cp/decl.cc
+++ b/gcc/cp/decl.cc
@@ -13058,7 +13058,8 @@ grokdeclarator (const cp_declarator *declarator,
&& !diagnose_misapplied_contracts (declspecs->std_attributes))
{
location_t attr_loc = declspecs->locations[ds_std_attribute];
- if (warning_at (attr_loc, OPT_Wattributes, "attribute ignored"))
+ if (any_nonignored_attribute_p (declspecs->std_attributes)
+ && warning_at (attr_loc, OPT_Wattributes, "attribute ignored"))
inform (attr_loc, "an attribute that appertains to a type-specifier "
"is ignored");
}
diff --git a/gcc/cp/name-lookup.cc b/gcc/cp/name-lookup.cc
index 76f1d44..09dc6ef 100644
--- a/gcc/cp/name-lookup.cc
+++ b/gcc/cp/name-lookup.cc
@@ -6356,7 +6356,7 @@ handle_namespace_attrs (tree ns, tree attributes)
DECL_ATTRIBUTES (ns) = tree_cons (name, args,
DECL_ATTRIBUTES (ns));
}
- else
+ else if (!attribute_ignored_p (d))
{
warning (OPT_Wattributes, "%qD attribute directive ignored",
name);
@@ -8703,7 +8703,7 @@ finish_using_directive (tree target, tree attribs)
diagnosed = true;
}
}
- else
+ else if (!attribute_ignored_p (a))
warning (OPT_Wattributes, "%qD attribute directive ignored", name);
}
}
diff --git a/gcc/cp/parser.cc b/gcc/cp/parser.cc
index ca91a50..de7af15 100644
--- a/gcc/cp/parser.cc
+++ b/gcc/cp/parser.cc
@@ -12778,9 +12778,8 @@ cp_parser_statement (cp_parser* parser, tree in_statement_expr,
SET_EXPR_LOCATION (statement, statement_location);
/* Allow "[[fallthrough]];" or "[[assume(cond)]];", but warn otherwise. */
- if (std_attrs != NULL_TREE)
- warning_at (attrs_loc,
- OPT_Wattributes,
+ if (std_attrs != NULL_TREE && any_nonignored_attribute_p (std_attrs))
+ warning_at (attrs_loc, OPT_Wattributes,
"attributes at the beginning of statement are ignored");
}
@@ -12989,7 +12988,7 @@ cp_parser_expression_statement (cp_parser* parser, tree in_statement_expr)
}
/* Allow "[[fallthrough]];", but warn otherwise. */
- if (attr != NULL_TREE)
+ if (attr != NULL_TREE && any_nonignored_attribute_p (attr))
warning_at (loc, OPT_Wattributes,
"attributes at the beginning of statement are ignored");
@@ -15194,7 +15193,7 @@ cp_parser_declaration (cp_parser* parser, tree prefix_attrs)
}
}
- if (std_attrs != NULL_TREE && !attribute_ignored_p (std_attrs))
+ if (std_attrs != NULL_TREE && any_nonignored_attribute_p (std_attrs))
warning_at (make_location (attrs_loc, attrs_loc, parser->lexer),
OPT_Wattributes, "attribute ignored");
if (cp_lexer_next_token_is (parser->lexer, CPP_SEMICOLON))
@@ -22675,7 +22674,7 @@ cp_parser_asm_definition (cp_parser* parser)
symtab->finalize_toplevel_asm (string);
}
- if (std_attrs)
+ if (std_attrs && any_nonignored_attribute_p (std_attrs))
warning_at (asm_loc, OPT_Wattributes,
"attributes ignored on %<asm%> declaration");
}
diff --git a/gcc/testsuite/g++.dg/warn/Wno-attributes-1.C b/gcc/testsuite/g++.dg/warn/Wno-attributes-1.C
new file mode 100644
index 0000000..863ca5c
--- /dev/null
+++ b/gcc/testsuite/g++.dg/warn/Wno-attributes-1.C
@@ -0,0 +1,52 @@
+// { dg-do compile { target c++11 } }
+// { dg-options "-Wno-attributes=bar:: -Wno-attributes=baz::qux" }
+
+[[foo::bar]]; // { dg-warning "attribute ignored" }
+[[bar::foo, foo::bar, baz::qux]]; // { dg-warning "attribute ignored" }
+[[bar::foo, bar::bar, baz::qux]]; // { dg-bogus "attribute ignored" }
+
+namespace [[foo::bar]] N { // { dg-warning "'bar' attribute directive ignored" }
+ int n;
+}
+namespace [[bar::foo, foo::bar, baz::qux]] O { // { dg-warning "'bar' attribute directive ignored" }
+ int o;
+}
+namespace [[bar::foo, bar::bar, baz::qux]] P { // { dg-bogus "attribute directive ignored" }
+ int p;
+}
+
+void
+foo ()
+{
+ int i = 0;
+ [[foo::bar]]; // { dg-warning "attributes at the beginning of statement are ignored" }
+ [[bar::foo, foo::bar, baz::qux]]; // { dg-warning "attributes at the beginning of statement are ignored" }
+ [[bar::foo, bar::bar, baz::qux]]; // { dg-bogus "attributes at the beginning of statement are ignored" }
+ [[foo::bar]] ++i; // { dg-warning "attributes at the beginning of statement are ignored" }
+ [[bar::foo, foo::bar, baz::qux]] ++i; // { dg-warning "attributes at the beginning of statement are ignored" }
+ [[bar::foo, bar::bar, baz::qux]] ++i; // { dg-bogus "attributes at the beginning of statement are ignored" }
+ [[foo::bar]] asm (""); // { dg-warning "attributes ignored on 'asm' declaration" }
+ [[bar::foo, foo::bar, baz::qux]] asm (""); // { dg-warning "attributes ignored on 'asm' declaration" }
+ [[bar::foo, bar::bar, baz::qux]] asm (""); // { dg-bogus "attributes ignored on 'asm' declaration" }
+ [[foo::bar]] using namespace N; // { dg-warning "'bar' attribute directive ignored" }
+ [[bar::foo, foo::bar, baz::qux]] using namespace O; // { dg-warning "'bar' attribute directive ignored" }
+ [[bar::foo, bar::bar, baz::qux]] using namespace P; // { dg-bogus "attribute directive ignored" }
+}
+
+class S
+{
+ [[foo::bar]] friend int bar (S &); // { dg-warning "attribute ignored" }
+ // { dsg-message "an attribute that appertains to a friend declaration that is not a definition is ignored" "" { target *-*-* } .-1 }
+ [[bar::foo, foo::bar, baz::qux]] friend int baz (S &); // { dg-warning "attribute ignored" }
+ // { dsg-message "an attribute that appertains to a friend declaration that is not a definition is ignored" "" { target *-*-* } .-1 }
+ [[bar::foo, bar::bar, baz::qux]] friend int qux (S &); // { dg-warning "attribute ignored" }
+ // { dsg-message "an attribute that appertains to a friend declaration that is not a definition is ignored" "" { target *-*-* } .-1 }
+public:
+ int s;
+};
+
+int [[foo::bar]] i; // { dg-warning "attribute ignored" }
+ // { dg-message "an attribute that appertains to a type-specifier is ignored" "" { target *-*-* } .-1 }
+int [[bar::foo, foo::bar, baz::qux]] j; // { dg-warning "attribute ignored" }
+ // { dg-message "an attribute that appertains to a type-specifier is ignored" "" { target *-*-* } .-1 }
+int [[bar::foo, bar::bar, baz::qux]] k; // { dg-bogus "attribute ignored" }