https://gcc.gnu.org/g:92216cbc59b3a4566d49de5dfa059b70b03d639a

commit r15-6385-g92216cbc59b3a4566d49de5dfa059b70b03d639a
Author: Jakub Jelinek <ja...@redhat.com>
Date:   Fri Dec 20 10:17:56 2024 +0100

    c++: Fix up maybe_unused attribute handling [PR110345]
    
    When adding test coverage for maybe_unused attribute, I've run into
    several things:
    1) similarly to deprecated attribute, the attribute shouldn't pedantically
       appertain to types other than class/enumeration definitions
    2) similarly to deprecated attribute, the attribute shouldn't pedantically
       appertain to unnamed bit-fields
    3) the standard says that it can appertain to identifier labels, but
       we handled it silently also on case and default labels
    4) I've run into a weird spurious error on
       int f [[maybe_unused]];
       int & [[maybe_unused]] i = f;
       int && [[maybe_unused]] j = 0;
       The problem was that we create an attribute variant for the int &
       type, then create an attribute variant for the int && type, and
       the type_canon_hash hashing just thought those 2 are the same,
       so used int & [[maybe_unused]] type for j rather than
       int && [[maybe_unused]].  As TYPE_REF_IS_RVALUE is a flag in the
       generic code, it was easily possible to hash that flag and compare
       it
    
    2024-12-19  Jakub Jelinek  <ja...@redhat.com>
    
            PR c++/110345
    gcc/
            * tree.cc (type_hash_canon_hash): Hash TYPE_REF_IS_RVALUE for
            REFERENCE_TYPE.
            (type_cache_hasher::equal): Compare TYPE_REF_IS_RVALUE for
            REFERENCE_TYPE.
    gcc/cp/
            * tree.cc (handle_maybe_unused_attribute): New function.
            (std_attributes): Use handle_maybe_unused_attribute instead
            of handle_unused_attribute for maybe_unused attribute.
    gcc/testsuite/
            * g++.dg/cpp0x/attr-maybe_unused1.C: New test.
            * g++.dg/cpp0x/alignas21.C: Add test for int && alignas (int).

Diff:
---
 gcc/cp/tree.cc                                  |  23 +++-
 gcc/testsuite/g++.dg/cpp0x/alignas21.C          |   1 +
 gcc/testsuite/g++.dg/cpp0x/attr-maybe_unused1.C | 148 ++++++++++++++++++++++++
 gcc/tree.cc                                     |   8 +-
 4 files changed, 178 insertions(+), 2 deletions(-)

diff --git a/gcc/cp/tree.cc b/gcc/cp/tree.cc
index 12d0831e4c4c..9b181be6237f 100644
--- a/gcc/cp/tree.cc
+++ b/gcc/cp/tree.cc
@@ -5162,6 +5162,27 @@ handle_std_deprecated_attribute (tree *node, tree name, 
tree args, int flags,
   return ret;
 }
 
+/* The C++17 [[maybe_unused]] attribute mostly maps to the GNU unused
+   attribute.  */
+
+static tree
+handle_maybe_unused_attribute (tree *node, tree name, tree args, int flags,
+                              bool *no_add_attrs)
+{
+  tree t = *node;
+  tree ret = handle_unused_attribute (node, name, args, flags, no_add_attrs);
+  if (TYPE_P (*node) && t != *node)
+    pedwarn (input_location, OPT_Wattributes,
+            "%qE on a type other than class or enumeration definition", name);
+  else if (TREE_CODE (*node) == FIELD_DECL && DECL_UNNAMED_BIT_FIELD (*node))
+    pedwarn (input_location, OPT_Wattributes, "%qE on unnamed bit-field",
+            name);
+  else if (TREE_CODE (*node) == LABEL_DECL && DECL_NAME (*node) == NULL_TREE)
+    pedwarn (input_location, OPT_Wattributes,
+            "%qE on %<case%> or %<default%> label", name);
+  return ret;
+}
+
 /* Table of valid C++ attributes.  */
 static const attribute_spec cxx_gnu_attributes[] =
 {
@@ -5188,7 +5209,7 @@ static const attribute_spec std_attributes[] =
   { "deprecated", 0, 1, false, false, false, false,
     handle_std_deprecated_attribute, NULL },
   { "maybe_unused", 0, 0, false, false, false, false,
-    handle_unused_attribute, NULL },
+    handle_maybe_unused_attribute, NULL },
   { "nodiscard", 0, 1, false, false, false, false,
     handle_nodiscard_attribute, NULL },
   { "no_unique_address", 0, 0, true, false, false, false,
diff --git a/gcc/testsuite/g++.dg/cpp0x/alignas21.C 
b/gcc/testsuite/g++.dg/cpp0x/alignas21.C
index b7be7c5ce926..3d3059a4e02c 100644
--- a/gcc/testsuite/g++.dg/cpp0x/alignas21.C
+++ b/gcc/testsuite/g++.dg/cpp0x/alignas21.C
@@ -81,6 +81,7 @@ int g2 alignas (int) [2];
 int corge () alignas (int);                    // { dg-error "'alignas' on a 
type other than class" }
 int *alignas (int) h;                          // { dg-error "'alignas' on a 
type other than class" }
 int & alignas (int) i = f;                     // { dg-error "'alignas' on a 
type other than class" }
+int && alignas (int) j = 0;                    // { dg-error "'alignas' on a 
type other than class" }
 int S::* alignas (int) k;                      // { dg-error "'alignas' on a 
type other than class" }
 auto l = sizeof (int [2] alignas (int));       // { dg-error "'alignas' on a 
type other than class" }
 int freddy (alignas (int) int a,               // { dg-error "alignment may 
not be specified for 'a'" }
diff --git a/gcc/testsuite/g++.dg/cpp0x/attr-maybe_unused1.C 
b/gcc/testsuite/g++.dg/cpp0x/attr-maybe_unused1.C
new file mode 100644
index 000000000000..c2c2d36b5797
--- /dev/null
+++ b/gcc/testsuite/g++.dg/cpp0x/attr-maybe_unused1.C
@@ -0,0 +1,148 @@
+// C++ 26 P2552R3 - On the ignorability of standard attributes
+// { dg-do compile { target c++11 } }
+
+int arr[2];
+struct S { int a, b; };
+S arr2[2];
+
+void
+foo (int n)
+{
+  [[maybe_unused]] int x1;
+  [[maybe_unused ("foobar")]] int x2;          // { dg-error "'maybe_unused' 
attribute does not take any arguments" }
+                                               // { dg-error "expected 
primary-expression before 'int'" "" { target *-*-* } .-1 }
+  [[maybe_unused (0)]] int x3;                 // { dg-error "'maybe_unused' 
attribute does not take any arguments" }
+                                               // { dg-error "expected 
primary-expression before 'int'" "" { target *-*-* } .-1 }
+
+  auto a = [] [[maybe_unused]] () {};
+  auto b = [] constexpr [[maybe_unused]] {};   // { dg-error "'maybe_unused' 
on a type other than class or enumeration definition" }
+                                               // { dg-error "parameter 
declaration before lambda declaration specifiers only optional with" "" { 
target c++20_down } .-1 }
+                                               // { dg-error "'constexpr' 
lambda only available with" "" { target c++14_down } .-2 }
+  auto c = [] noexcept [[maybe_unused]] {};    // { dg-error "'maybe_unused' 
on a type other than class or enumeration definition" }
+                                               // { dg-error "parameter 
declaration before lambda exception specification only optional with" "" { 
target c++20_down } .-1 }
+  auto d = [] () [[maybe_unused]] {};          // { dg-error "'maybe_unused' 
on a type other than class or enumeration definition" }
+  auto e = new int [n] [[maybe_unused]];       // { dg-warning "attributes 
ignored on outermost array type in new expression" }
+  auto e2 = new int [n] [[maybe_unused]] [42]; // { dg-warning "attributes 
ignored on outermost array type in new expression" }
+  auto f = new int [n][42] [[maybe_unused]];   // { dg-error "'maybe_unused' 
on a type other than class or enumeration definition" }
+  [[maybe_unused]];                            // { dg-warning "attributes at 
the beginning of statement are ignored" }
+  [[maybe_unused]] {}                          // { dg-warning "attributes at 
the beginning of statement are ignored" }
+  [[maybe_unused]] if (true) {}                        // { dg-warning 
"attributes at the beginning of statement are ignored" }
+  [[maybe_unused]] while (false) {}            // { dg-warning "attributes at 
the beginning of statement are ignored" }
+  [[maybe_unused]] goto lab;                   // { dg-warning "attributes at 
the beginning of statement are ignored" }
+  [[maybe_unused]] lab:;
+  [[maybe_unused]] try {} catch (int) {}       // { dg-warning "attributes at 
the beginning of statement are ignored" }
+  if ([[maybe_unused]] int x = 0) {}
+  switch (n)
+    {
+    [[maybe_unused]] case 1:                   // { dg-error "'maybe_unused' 
on 'case' or 'default' label" }
+    [[maybe_unused]] break;                    // { dg-warning "attributes at 
the beginning of statement are ignored" }
+    [[maybe_unused]] default:                  // { dg-error "'maybe_unused' 
on 'case' or 'default' label" }
+        break;
+    }
+  for ([[maybe_unused]] auto a : arr) {}
+  for ([[maybe_unused]] auto [a, b] : arr2) {} // { dg-error "structured 
bindings only available with" "" { target c++14_down } }
+  [[maybe_unused]] asm ("");                   // { dg-warning "attributes 
ignored on 'asm' declaration" }
+  try {} catch ([[maybe_unused]] int x) {}
+  try {} catch ([[maybe_unused]] int) {}
+  try {} catch (int [[maybe_unused]] x) {}     // { dg-warning "attribute 
ignored" }
+  try {} catch (int [[maybe_unused]]) {}       // { dg-warning "attribute 
ignored" }
+  try {} catch (int x [[maybe_unused]]) {}
+}
+
+[[maybe_unused]] int bar ();
+using foobar [[maybe_unused]] = int;
+[[maybe_unused]] int a;
+[[maybe_unused]] auto [b, c] = arr;            // { dg-error "structured 
bindings only available with" "" { target c++14_down } }
+[[maybe_unused]];                              // { dg-warning "attribute 
ignored" }
+inline [[maybe_unused]] void baz () {}         // { dg-warning "attribute 
ignored" }
+                                               // { dg-error "standard 
attributes in middle of decl-specifiers" "" { target *-*-* } .-1 }
+constexpr [[maybe_unused]] int qux () { return 0; }    // { dg-warning 
"attribute ignored" }
+                                               // { dg-error "standard 
attributes in middle of decl-specifiers" "" { target *-*-* } .-1 }
+int [[maybe_unused]] d;                                // { dg-warning 
"attribute ignored" }
+int const [[maybe_unused]] e = 1;              // { dg-warning "attribute 
ignored" }
+struct A {} [[maybe_unused]];                  // { dg-warning "attribute 
ignored in declaration of 'struct A'" }
+struct A [[maybe_unused]];                     // { dg-warning "attribute 
ignored" }
+struct A [[maybe_unused]] a1;                  // { dg-warning "attribute 
ignored" }
+A [[maybe_unused]] a2;                         // { dg-warning "attribute 
ignored" }
+enum B { B0 } [[maybe_unused]];                        // { dg-warning 
"attribute ignored in declaration of 'enum B'" }
+enum B [[maybe_unused]];                       // { dg-warning "attribute 
ignored" }
+enum B [[maybe_unused]] b1;                    // { dg-warning "attribute 
ignored" }
+B [[maybe_unused]] b2;                         // { dg-warning "attribute 
ignored" }
+struct [[maybe_unused]] C {};
+int f [[maybe_unused]];
+int g[2] [[maybe_unused]];                     // { dg-error "'maybe_unused' 
on a type other than class or enumeration definition" }
+int g2 [[maybe_unused]] [2];
+int corge () [[maybe_unused]];                 // { dg-error "'maybe_unused' 
on a type other than class or enumeration definition" }
+int *[[maybe_unused]] h;                       // { dg-error "'maybe_unused' 
on a type other than class or enumeration definition" }
+int & [[maybe_unused]] i = f;                  // { dg-error "'maybe_unused' 
on a type other than class or enumeration definition" }
+int && [[maybe_unused]] j = 0;                 // { dg-error "'maybe_unused' 
on a type other than class or enumeration definition" }
+int S::* [[maybe_unused]] k;                   // { dg-error "'maybe_unused' 
on a type other than class or enumeration definition" }
+auto l = sizeof (int [2] [[maybe_unused]]);    // { dg-error "'maybe_unused' 
on a type other than class or enumeration definition" }
+int freddy ([[maybe_unused]] int a,
+           [[maybe_unused]] int,
+           [[maybe_unused]] int c = 0,
+           [[maybe_unused]] int = 0);
+void
+corge ([[maybe_unused]] int a,
+       [[maybe_unused]] int,
+       [[maybe_unused]] int c = 0,
+       [[maybe_unused]] int = 0)
+{
+}
+[[maybe_unused]] void
+garply ()
+{
+}
+int grault (int [[maybe_unused]] a,            // { dg-warning "attribute 
ignored" }
+           int [[maybe_unused]],               // { dg-warning "attribute 
ignored" }
+           int [[maybe_unused]] c = 0,         // { dg-warning "attribute 
ignored" }
+           int [[maybe_unused]] = 0);          // { dg-warning "attribute 
ignored" }
+void
+waldo (int [[maybe_unused]] a,                 // { dg-warning "attribute 
ignored" }
+       int [[maybe_unused]],                   // { dg-warning "attribute 
ignored" }
+       int [[maybe_unused]] c = 0,             // { dg-warning "attribute 
ignored" }
+       int [[maybe_unused]] = 0)               // { dg-warning "attribute 
ignored" }
+{
+}
+int plugh (int a [[maybe_unused]],
+           int b [[maybe_unused]] = 0);
+void
+thud (int a [[maybe_unused]],
+      int b [[maybe_unused]] = 0)
+{
+}
+enum [[maybe_unused]] D { D0 };
+enum class [[maybe_unused]] E { E0 };
+enum F {};
+enum [[maybe_unused]] F;                       // { dg-warning "type 
attributes ignored after type is already defined" }
+enum G {
+  G0 [[maybe_unused]],
+  G1 [[maybe_unused]] = 2
+};
+namespace [[maybe_unused]] H { using H0 = int; }// { dg-warning 
"'maybe_unused' attribute directive ignored" }
+namespace [[maybe_unused]] {}                  // { dg-warning "'maybe_unused' 
attribute directive ignored" }
+[[maybe_unused]] using namespace H;            // { dg-warning "'maybe_unused' 
attribute directive ignored" }
+struct [[maybe_unused]] I
+{
+  [[maybe_unused]];                            // { dg-error "declaration does 
not declare anything" }
+  [[maybe_unused]] int i;
+  [[maybe_unused]] int foo ();
+  [[maybe_unused]] int bar () { return 1; }
+  [[maybe_unused]] int : 0;                    // { dg-error "'maybe_unused' 
on unnamed bit-field" }
+  [[maybe_unused]] int i2 : 5;
+  [[maybe_unused]] static int i3;
+  static int i4;
+};
+[[maybe_unused]] int I::i4 = 0;
+struct J : [[maybe_unused]] C {};              // { dg-warning "attributes on 
base specifiers are ignored" }
+#if __cpp_concepts >= 201907L
+template <typename T>
+concept K [[maybe_unused]] = requires { true; };// { dg-warning 
"'maybe_unused' attribute ignored" "" { target c++20 } }
+#endif
+typedef int L [[maybe_unused]];
+template <typename T>
+struct M {};
+template <>
+struct [[maybe_unused]] M<int> { int m; };
+typedef int N[2] [[maybe_unused]];             // { dg-error "'maybe_unused' 
on a type other than class or enumeration definition" }
+typedef int O [[maybe_unused]] [2];
diff --git a/gcc/tree.cc b/gcc/tree.cc
index 322451139162..e3b5e56b828c 100644
--- a/gcc/tree.cc
+++ b/gcc/tree.cc
@@ -6120,6 +6120,10 @@ type_hash_canon_hash (tree type)
       hstate.add_poly_int (TYPE_VECTOR_SUBPARTS (type));
       break;
 
+    case REFERENCE_TYPE:
+      hstate.add_flag (TYPE_REF_IS_RVALUE (type));
+      break;
+
     default:
       break;
     }
@@ -6162,7 +6166,6 @@ type_cache_hasher::equal (type_hash *a, type_hash *b)
     case OPAQUE_TYPE:
     case COMPLEX_TYPE:
     case POINTER_TYPE:
-    case REFERENCE_TYPE:
     case NULLPTR_TYPE:
       return true;
 
@@ -6252,6 +6255,9 @@ type_cache_hasher::equal (type_hash *a, type_hash *b)
        break;
       return false;
 
+    case REFERENCE_TYPE:
+      return TYPE_REF_IS_RVALUE (a->type) == TYPE_REF_IS_RVALUE (b->type);
+
     default:
       return false;
     }

Reply via email to