]> git.ipfire.org Git - thirdparty/gcc.git/commitdiff
re PR middle-end/52939 (ice in gimple_get_virt_method_for_binfo with -O3)
authorMartin Jambor <mjambor@suse.cz>
Fri, 13 Apr 2012 17:57:21 +0000 (19:57 +0200)
committerMartin Jambor <jamborm@gcc.gnu.org>
Fri, 13 Apr 2012 17:57:21 +0000 (19:57 +0200)
2012-04-13  Martin Jambor  <mjambor@suse.cz>

PR middle-end/52939
* gimple-fold.c (gimple_get_virt_method_for_binfo): Bail out if
fold_ctor_reference returns a zero constant.

* testsuite/g++.dg/ipa/pr52939.C: New test.

From-SVN: r186428

gcc/ChangeLog
gcc/gimple-fold.c
gcc/testsuite/ChangeLog
gcc/testsuite/g++.dg/ipa/pr52939.C [new file with mode: 0644]

index e3a691e6dc55c285e8101f08a385420c27abffac..e04d1be979ceabbe1a3ed408efc955dff47c7f37 100644 (file)
@@ -1,3 +1,9 @@
+2012-04-13  Martin Jambor  <mjambor@suse.cz>
+
+       PR middle-end/52939
+       * gimple-fold.c (gimple_get_virt_method_for_binfo): Bail out if
+       fold_ctor_reference returns a zero constant.
+
 2012-04-13  Enkovich Ilya  <ilya.enkovich@intel.com>
 
        * config.gcc: Add i386/gnu-user-common.h before all
index 0ea5d56543fe3c8ee88b502acd3dc97530c733ef..049da5782afe6f9c56422459ce20a691af9ee9f7 100644 (file)
@@ -3087,7 +3087,7 @@ gimple_get_virt_method_for_binfo (HOST_WIDE_INT token, tree known_binfo)
   offset += token * size;
   fn = fold_ctor_reference (TREE_TYPE (TREE_TYPE (v)), DECL_INITIAL (v),
                            offset, size);
-  if (!fn)
+  if (!fn || integer_zerop (fn))
     return NULL_TREE;
   gcc_assert (TREE_CODE (fn) == ADDR_EXPR
              || TREE_CODE (fn) == FDESC_EXPR);
index b39e623f89f89a642168f86478151a8a3ec1ec66..556d2f6db86878bafa88fd34b9cc3479f504a3f3 100644 (file)
@@ -1,3 +1,8 @@
+2012-04-13  Martin Jambor  <mjambor@suse.cz>
+
+       PR middle-end/52939
+       * g++.dg/ipa/pr52939.C: New test.
+
 2012-04-13  Tom de Vries  <tom@codesourcery.com>
 
        * gcc.dg/pr52734.c: New test.
diff --git a/gcc/testsuite/g++.dg/ipa/pr52939.C b/gcc/testsuite/g++.dg/ipa/pr52939.C
new file mode 100644 (file)
index 0000000..e120827
--- /dev/null
@@ -0,0 +1,58 @@
+/* Verify that we do not ICE on invalid devirtualizations (which might
+   be OK at run-time because never executed).  */
+/* { dg-do run } */
+/* { dg-options "-O3 -fno-early-inlining -fno-inline"  } */
+
+extern "C" void abort (void);
+
+class A
+{
+public:
+  int data;
+  virtual int foo (int i);
+};
+
+class B : public A
+{
+public:
+  virtual int foo (int i);
+  virtual int bar (int i);
+};
+
+int A::foo (int i)
+{
+  return i + 1;
+}
+
+int B::foo (int i)
+{
+  return i + 2;
+}
+
+int B::bar (int i)
+{
+  return i + 3;
+}
+
+static int middleman (class A *obj, int i)
+{
+  class B *b = (class B *) obj;
+
+  if (i != 1)
+    return b->bar (i);
+  else
+    return i;
+}
+
+int __attribute__ ((noinline,noclone)) get_input(void)
+{
+  return 1;
+}
+
+int main (int argc, char *argv[])
+{
+  class A o;
+  if (middleman (&o, get_input ()) != 1)
+    abort ();
+  return 0;
+}