PR C++/88114 Gen destructor of an abstract class
authorTobias Burnus <burnus@net-b.de>
Fri, 11 Jan 2019 19:40:13 +0000 (20:40 +0100)
committerTobias Burnus <burnus@gcc.gnu.org>
Fri, 11 Jan 2019 19:40:13 +0000 (20:40 +0100)
        PR C++/8811
        * decl2.c (maybe_emit_vtables): If needed, generate code for
        the destructor of an abstract class.
        (mark_used): Update comment for older function-name change.

        PR C++/88114
        * g++.dg/cpp0x/defaulted61.C: New.
        * g++.dg/cpp0x/defaulted62.C: New.

From-SVN: r267855

gcc/cp/ChangeLog
gcc/cp/decl2.c
gcc/testsuite/ChangeLog
gcc/testsuite/g++.dg/cpp0x/defaulted61.C [new file with mode: 0644]
gcc/testsuite/g++.dg/cpp0x/defaulted62.C [new file with mode: 0644]

index 4b9fa821d74e22d90d63276555b68a172b07eaa5..5a5256f13fc5b5d57cbe4eb7ce8dfe1a96e274c3 100644 (file)
@@ -1,3 +1,10 @@
+2019-01-11  Tobias Burnus  <burnus@net-b.de>
+
+       PR C++/88114
+       * decl2.c (maybe_emit_vtables): If needed, generate code for
+       the destructor of an abstract class.
+       (mark_used): Update comment for older function-name change.
+
 2019-01-11  Paolo Carlini  <paolo.carlini@oracle.com>
 
        * decl.c (start_decl): Improve error location.
index dbab95fbc965c275858fad673dd399817e1994b1..591cc21be20458dea51db798747a1a80f094f08f 100644 (file)
@@ -2220,6 +2220,16 @@ maybe_emit_vtables (tree ctype)
        }
     }
 
+  /* For abstract classes, the destructor has been removed from the
+     vtable (in class.c's build_vtbl_initializer).  For a compiler-
+     generated destructor, it hence might not have been generated in
+     this translation unit - and with '#pragma interface' it might
+     never get generated.  */
+  if (CLASSTYPE_PURE_VIRTUALS (ctype)
+      && TYPE_HAS_NONTRIVIAL_DESTRUCTOR (ctype)
+      && DECL_DEFAULTED_IN_CLASS_P(CLASSTYPE_DESTRUCTOR(ctype)))
+    note_vague_linkage_fn (CLASSTYPE_DESTRUCTOR(ctype));
+
   /* Since we're writing out the vtable here, also write the debug
      info.  */
   note_debug_info_needed (ctype);
@@ -5497,7 +5507,7 @@ mark_used (tree decl, tsubst_flags_t complain)
         within the body of a function so as to avoid collecting live data
         on the stack (such as overload resolution candidates).
 
-         We could just let cp_write_global_declarations handle synthesizing
+         We could just let c_parse_final_cleanups handle synthesizing
          this function by adding it to deferred_fns, but doing
          it at the use site produces better error messages.  */
       ++function_depth;
index 3b1de48f99020f07b1ba7b62350e149d229cd728..8a031e1d9f8645e37094d939019418a3da3f1155 100644 (file)
@@ -1,3 +1,9 @@
+2019-01-11  Tobias Burnus  <burnus@net-b.de>
+
+       PR C++/88114
+       * g++.dg/cpp0x/defaulted61.C: New       
+       * g++.dg/cpp0x/defaulted62.C: New.
+
 2019-01-11  Jakub Jelinek  <jakub@redhat.com>
 
        PR tree-optimization/88693
diff --git a/gcc/testsuite/g++.dg/cpp0x/defaulted61.C b/gcc/testsuite/g++.dg/cpp0x/defaulted61.C
new file mode 100644 (file)
index 0000000..e7e0a48
--- /dev/null
@@ -0,0 +1,22 @@
+// { dg-do compile { target c++11 } }
+// { dg-final { scan-assembler "_ZN3OneD0Ev" } }
+
+// PR C++/88114
+// Destructor of an abstract class was never generated
+// when compiling the class - nor later due to the
+// '#pragma interface'
+
+#pragma implementation
+#pragma interface
+
+class One
+{
+ public:
+  virtual ~One() = default;
+  void some_fn();
+  virtual void later() = 0;
+ private:
+  int m_int;
+};
+
+void One::some_fn() { }
diff --git a/gcc/testsuite/g++.dg/cpp0x/defaulted62.C b/gcc/testsuite/g++.dg/cpp0x/defaulted62.C
new file mode 100644 (file)
index 0000000..d8dab60
--- /dev/null
@@ -0,0 +1,25 @@
+// { dg-do compile { target c++11 } }
+// { dg-final { scan-assembler-not "_ZN3OneD0Ev" } }
+
+// PR C++/88114
+// Destructor of an abstract class was never generated
+// when compiling the class - nor later due to the
+// '#pragma interface'
+// -> g++.dg/cpp0x/defaulted61.C
+
+// HERE, in g++.dg/cpp0x/defaulted62.C:
+// As we have commented the pragmas, it should NOT be created
+// #pragma implementation
+// #pragma interface
+
+class One
+{
+ public:
+  virtual ~One() = default;
+  void some_fn();
+  virtual void later() = 0;
+ private:
+  int m_int;
+};
+
+void One::some_fn() { }