[PR c++/84962] ICE with anon-struct member
authorNathan Sidwell <nathan@acm.org>
Tue, 20 Mar 2018 16:01:08 +0000 (16:01 +0000)
committerNathan Sidwell <nathan@gcc.gnu.org>
Tue, 20 Mar 2018 16:01:08 +0000 (16:01 +0000)
https://gcc.gnu.org/ml/gcc-patches/2018-03/msg00961.html
PR c++/84962
* name-lookup.c (pushdecl_class_level): Push anon-struct's
member_vec, if there is one.

PR c++/84962
* g++.dg/lookup/pr84962.C: New.

From-SVN: r258686

gcc/cp/ChangeLog
gcc/cp/name-lookup.c
gcc/testsuite/ChangeLog
gcc/testsuite/g++.dg/lookup/pr84962.C [new file with mode: 0644]

index 7ce79d3862b8c5c9527b1bd28130a3a2fa5d0a39..5d7b9c09983c613f52fcbb6ef607b5b0455f64ea 100644 (file)
@@ -1,5 +1,9 @@
 2018-03-20  Nathan Sidwell  <nathan@acm.org>
 
+       PR c++/84962
+       * name-lookup.c (pushdecl_class_level): Push anon-struct's
+       member_vec, if there is one.
+
        PR c++/84970
        * cp-tree.h (lookup_list_keep): Declare.
        * tree.c (lookup_list_keep): New, broken out of ...
index cc8bb2f81c68e960dca11a81f1e15795f7fef171..411a79696a19ee8352f7d9e923c3c2679f6774a5 100644 (file)
@@ -4490,16 +4490,30 @@ pushdecl_class_level (tree x)
       /* If X is an anonymous aggregate, all of its members are
         treated as if they were members of the class containing the
         aggregate, for naming purposes.  */
-      tree f;
-
-      for (f = TYPE_FIELDS (TREE_TYPE (x)); f; f = DECL_CHAIN (f))
-       {
-         location_t save_location = input_location;
-         input_location = DECL_SOURCE_LOCATION (f);
-         if (!pushdecl_class_level (f))
-           is_valid = false;
-         input_location = save_location;
+      location_t save_location = input_location;
+      tree anon = TREE_TYPE (x);
+      if (vec<tree, va_gc> *member_vec = CLASSTYPE_MEMBER_VEC (anon))
+       for (unsigned ix = member_vec->length (); ix--;)
+         {
+           tree binding = (*member_vec)[ix];
+           if (STAT_HACK_P (binding))
+             {
+               if (!pushdecl_class_level (STAT_TYPE (binding)))
+                 is_valid = false;
+               binding = STAT_DECL (binding);
+             }
+           if (!pushdecl_class_level (binding))
+             is_valid = false;
        }
+      else
+       for (tree f = TYPE_FIELDS (anon); f; f = DECL_CHAIN (f))
+         if (TREE_CODE (f) == FIELD_DECL)
+           {
+             input_location = DECL_SOURCE_LOCATION (f);
+             if (!pushdecl_class_level (f))
+               is_valid = false;
+           }
+      input_location = save_location;
     }
   timevar_cond_stop (TV_NAME_LOOKUP, subtime);
   return is_valid;
index 14c775ba191988d2959f407f1985cf04a43c8af7..21a16df4884c4725c9f781a23f20c8f1f5b866b0 100644 (file)
@@ -1,5 +1,8 @@
 2018-03-20  Nathan Sidwell  <nathan@acm.org>
 
+       PR c++/84962
+       * g++.dg/lookup/pr84962.C: New.
+
        PR c++/84970
        * g++.dg/lookup/pr84970.C: New.
 
diff --git a/gcc/testsuite/g++.dg/lookup/pr84962.C b/gcc/testsuite/g++.dg/lookup/pr84962.C
new file mode 100644 (file)
index 0000000..ee801c2
--- /dev/null
@@ -0,0 +1,14 @@
+// PR c++/84952 ICE with anon-struct having member fns
+// { dg-do compile { target c++11 } }
+// { dg-additional-options -Wno-pedantic }
+
+struct X {
+  struct 
+  {
+    template <typename> int a ();
+    // { dg-error "can only have" "" { target *-*-* } .-1 }
+  };
+
+  int  : a; // { dg-error "non-integral" }
+};
+