ppc32: retaddr
authorChanho Park <chanho61.park@samsung.com>
Fri, 18 Jul 2014 09:05:08 +0000 (18:05 +0900)
committerChanho Park <chanho61.park@samsung.com>
Mon, 21 Jul 2014 08:31:27 +0000 (17:31 +0900)
Signed-off-by: Chanho Park <chanho61.park@samsung.com>
gcc/config/rs6000/rs6000.c
gcc/testsuite/gcc.dg/20051128-1.c [new file with mode: 0644]

index bf67e72..c788663 100644 (file)
@@ -21499,18 +21499,22 @@ rs6000_return_addr (int count, rtx frame)
   if (count != 0
       || ((DEFAULT_ABI == ABI_V4 || DEFAULT_ABI == ABI_DARWIN) && flag_pic))
     {
+      rtx x;
       cfun->machine->ra_needs_full_frame = 1;
 
-      return
-       gen_rtx_MEM
-         (Pmode,
-          memory_address
-          (Pmode,
-           plus_constant (Pmode,
-                          copy_to_reg
-                          (gen_rtx_MEM (Pmode,
-                                        memory_address (Pmode, frame))),
-                          RETURN_ADDRESS_OFFSET)));
+      if (count == 0)
+       {
+         gcc_assert (frame == frame_pointer_rtx);
+         x = arg_pointer_rtx;
+       }
+      else
+       {
+         x = memory_address (Pmode, frame);
+         x = copy_to_reg (gen_rtx_MEM (Pmode, x));
+       }
+
+      x = plus_constant (Pmode, x, RETURN_ADDRESS_OFFSET);
+      return gen_rtx_MEM (Pmode, memory_address (Pmode, x));
     }
 
   cfun->machine->ra_need_lr = 1;
diff --git a/gcc/testsuite/gcc.dg/20051128-1.c b/gcc/testsuite/gcc.dg/20051128-1.c
new file mode 100644 (file)
index 0000000..1b234ee
--- /dev/null
@@ -0,0 +1,41 @@
+/* { dg-do run } */
+/* { dg-options "-O2 -fpic" } */
+
+extern void exit (int);
+extern void abort (void);
+
+int b;
+
+struct A
+{
+  void *pad[147];
+  void *ra, *h;
+  long o;
+};
+
+void
+__attribute__((noinline))
+foo (struct A *a, void *x)
+{
+  __builtin_memset (a, 0, sizeof (a));
+  if (!b)
+    exit (0);
+}
+
+void
+__attribute__((noinline))
+bar (void)
+{
+  struct A a;
+
+  __builtin_unwind_init ();
+  foo (&a, __builtin_return_address (0));
+}
+
+int
+main (void)
+{
+  bar ();
+  abort ();
+  return 0;
+}