[pr84733] Fix ICE popping local scope
authorNathan Sidwell <nathan@acm.org>
Fri, 27 Mar 2020 20:09:12 +0000 (13:09 -0700)
committerNathan Sidwell <nathan@acm.org>
Fri, 27 Mar 2020 20:09:12 +0000 (13:09 -0700)
PR c++/84733
* name-lookup.c (do_pushdecl): Look through cleanp levels.

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

index 2a2e6eac7bf681ccb766afa434248d6e15f52648..885f29d8c86fe04e234fa86b3e245593ea2577ba 100644 (file)
@@ -1,3 +1,8 @@
+2020-03-27  Nathan Sidwell  <nathan@acm.org>
+
+       PR c++/84733
+       * name-lookup.c (do_pushdecl): Look through cleanp levels.
+
 2020-03-27  Martin Sebor  <msebor@redhat.com>
 
        PR c++/94078
index e6dfb9cc723575e7b313a005ede57da7f424c570..8dd0b0d723e76d8ced4f949e29d81c6e716fdd1c 100644 (file)
@@ -2998,7 +2998,8 @@ do_pushdecl (tree decl, bool is_friend)
   /* The binding level we will be pushing into.  During local class
      pushing, we want to push to the containing scope.  */
   cp_binding_level *level = current_binding_level;
-  while (level->kind == sk_class)
+  while (level->kind == sk_class
+        || level->kind == sk_cleanup)
     level = level->level_chain;
 
   /* An anonymous namespace has a NULL DECL_NAME, but we still want to
diff --git a/gcc/testsuite/g++.dg/lookup/pr84733.C b/gcc/testsuite/g++.dg/lookup/pr84733.C
new file mode 100644 (file)
index 0000000..d0394ea
--- /dev/null
@@ -0,0 +1,21 @@
+// { dg-do compile { target c++11 } }
+// PR c++/84733 ICE popping local binding after cleanup region
+
+struct c {
+  ~c();
+} b;
+
+void f() {
+#ifndef OK
+  try {
+  d:
+    ;
+  } catch (int) {
+  }
+#endif
+  decltype(b) a;
+  int e;
+  struct e { } f;
+  e = 5;
+  struct e j;
+}
diff --git a/gcc/testsuite/g++.dg/lookup/pr94257.C b/gcc/testsuite/g++.dg/lookup/pr94257.C
new file mode 100644 (file)
index 0000000..d92e292
--- /dev/null
@@ -0,0 +1,16 @@
+// { dg-do compile { target c++11 } }
+// PR94257 ICE with silly inline nest.
+
+inline namespace B { // { dg-message "namespace B { }" }
+namespace B { }  // { dg-message "namespace B::B { }" }
+}
+
+namespace B { // { dg-error ".namespace B. is ambiguous" }
+}
+
+// But this is fine
+namespace D {
+inline namespace D { }
+}
+namespace D {
+}