Blame gcc44-pr33763.patch

9a2ea4c
2007-11-06  Jakub Jelinek  <jakub@redhat.com>
9a2ea4c
9a2ea4c
	PR tree-optimization/33763
9a2ea4c
	* gcc.dg/pr33763.c: New test.
9a2ea4c
	* g++.dg/opt/inline13.C: New test.
9a2ea4c
9a2ea4c
2007-11-06  Jan Hubicka  <jh@suse.cz>
9a2ea4c
9a2ea4c
	PR tree-optimization/33763
9a2ea4c
	* tree-inline.c (expand_call_inline): Silently ignore always_inline
9a2ea4c
	attribute for redefined extern inline functions.
9a2ea4c
9a2ea4c
--- gcc/tree-inline.c.jj	2007-11-06 09:29:04.000000000 +0100
9a2ea4c
+++ gcc/tree-inline.c	2007-11-06 16:19:12.000000000 +0100
9a2ea4c
@@ -3157,6 +3157,12 @@ expand_call_inline (basic_block bb, gimp
9a2ea4c
 	goto egress;
9a2ea4c
 
9a2ea4c
       if (lookup_attribute ("always_inline", DECL_ATTRIBUTES (fn))
9a2ea4c
+	/* For extern inline functions that get redefined we always
9a2ea4c
+	   silently ignored alway_inline flag. Better behaviour would
9a2ea4c
+	   be to be able to keep both bodies and use extern inline body
9a2ea4c
+	   for inlining, but we can't do that because frontends overwrite
9a2ea4c
+	   the body.  */
9a2ea4c
+	  && !cg_edge->callee->local.redefined_extern_inline
9a2ea4c
 	  /* Avoid warnings during early inline pass. */
9a2ea4c
 	  && cgraph_global_info_ready)
9a2ea4c
 	{
9a2ea4c
--- gcc/testsuite/gcc.dg/pr33763.c.jj	2007-11-06 16:19:12.000000000 +0100
9a2ea4c
+++ gcc/testsuite/gcc.dg/pr33763.c	2007-11-06 16:19:12.000000000 +0100
9a2ea4c
@@ -0,0 +1,60 @@
9a2ea4c
+/* PR tree-optimization/33763 */
9a2ea4c
+/* { dg-do compile } */
9a2ea4c
+/* { dg-options "-O2" } */
9a2ea4c
+
9a2ea4c
+typedef struct
9a2ea4c
+{
9a2ea4c
+  void *a;
9a2ea4c
+  void *b;
9a2ea4c
+} T;
9a2ea4c
+extern void *foo (const char *, const char *);
9a2ea4c
+extern void *bar (void *, const char *, T);
9a2ea4c
+extern int baz (const char *, int);
9a2ea4c
+
9a2ea4c
+extern inline __attribute__ ((always_inline, gnu_inline)) int
9a2ea4c
+baz (const char *x, int y)
9a2ea4c
+{
9a2ea4c
+  return 2;
9a2ea4c
+}
9a2ea4c
+
9a2ea4c
+int
9a2ea4c
+baz (const char *x, int y)
9a2ea4c
+{
9a2ea4c
+  return 1;
9a2ea4c
+}
9a2ea4c
+
9a2ea4c
+int xa, xb;
9a2ea4c
+
9a2ea4c
+static void *
9a2ea4c
+inl (const char *x, const char *y)
9a2ea4c
+{
9a2ea4c
+  T t = { &xa, &xb };
9a2ea4c
+  int *f = (int *) __builtin_malloc (sizeof (int));
9a2ea4c
+  const char *z;
9a2ea4c
+  int o = 0;
9a2ea4c
+  void *r = 0;
9a2ea4c
+
9a2ea4c
+  for (z = y; *z; z++)
9a2ea4c
+    {
9a2ea4c
+      if (*z == 'r')
9a2ea4c
+	o |= 1;
9a2ea4c
+      if (*z == 'w')
9a2ea4c
+	o |= 2;
9a2ea4c
+    }
9a2ea4c
+  if (o == 1)
9a2ea4c
+    *f = baz (x, 0);
9a2ea4c
+  if (o == 2)
9a2ea4c
+    *f = baz (x, 1);
9a2ea4c
+  if (o == 3)
9a2ea4c
+    *f = baz (x, 2);
9a2ea4c
+
9a2ea4c
+  if (o && *f > 0)
9a2ea4c
+    r = bar (f, "w", t);
9a2ea4c
+  return r;
9a2ea4c
+}
9a2ea4c
+
9a2ea4c
+void *
9a2ea4c
+foo (const char *x, const char *y)
9a2ea4c
+{
9a2ea4c
+  return inl (x, y);
9a2ea4c
+}
9a2ea4c
--- gcc/testsuite/g++.dg/opt/inline13.C.jj	2007-11-06 16:20:20.000000000 +0100
9a2ea4c
+++ gcc/testsuite/g++.dg/opt/inline13.C	2007-11-06 16:21:30.000000000 +0100
9a2ea4c
@@ -0,0 +1,60 @@
9a2ea4c
+// PR tree-optimization/33763
9a2ea4c
+// { dg-do compile }
9a2ea4c
+// { dg-options "-O2" }
9a2ea4c
+
9a2ea4c
+typedef struct
9a2ea4c
+{
9a2ea4c
+  void *a;
9a2ea4c
+  void *b;
9a2ea4c
+} T;
9a2ea4c
+extern void *foo (const char *, const char *);
9a2ea4c
+extern void *bar (void *, const char *, T);
9a2ea4c
+extern int baz (const char *, int);
9a2ea4c
+
9a2ea4c
+extern inline __attribute__ ((always_inline, gnu_inline)) int
9a2ea4c
+baz (const char *x, int y)
9a2ea4c
+{
9a2ea4c
+  return 2;
9a2ea4c
+}
9a2ea4c
+
9a2ea4c
+int
9a2ea4c
+baz (const char *x, int y)
9a2ea4c
+{
9a2ea4c
+  return 1;
9a2ea4c
+}
9a2ea4c
+
9a2ea4c
+int xa, xb;
9a2ea4c
+
9a2ea4c
+static void *
9a2ea4c
+inl (const char *x, const char *y)
9a2ea4c
+{
9a2ea4c
+  T t = { &xa, &xb };
9a2ea4c
+  int *f = (int *) __builtin_malloc (sizeof (int));
9a2ea4c
+  const char *z;
9a2ea4c
+  int o = 0;
9a2ea4c
+  void *r = 0;
9a2ea4c
+
9a2ea4c
+  for (z = y; *z; z++)
9a2ea4c
+    {
9a2ea4c
+      if (*z == 'r')
9a2ea4c
+	o |= 1;
9a2ea4c
+      if (*z == 'w')
9a2ea4c
+	o |= 2;
9a2ea4c
+    }
9a2ea4c
+  if (o == 1)
9a2ea4c
+    *f = baz (x, 0);
9a2ea4c
+  if (o == 2)
9a2ea4c
+    *f = baz (x, 1);
9a2ea4c
+  if (o == 3)
9a2ea4c
+    *f = baz (x, 2);
9a2ea4c
+
9a2ea4c
+  if (o && *f > 0)
9a2ea4c
+    r = bar (f, "w", t);
9a2ea4c
+  return r;
9a2ea4c
+}
9a2ea4c
+
9a2ea4c
+void *
9a2ea4c
+foo (const char *x, const char *y)
9a2ea4c
+{
9a2ea4c
+  return inl (x, y);
9a2ea4c
+}