2018-03-08 Richard Biener <rguenther@suse.de>
[official-gcc.git] / gcc / testsuite / gcc.target / powerpc / spe-unwind-1.c
blob84d4bf28803f2da00b4b37e088dfcfaca6c30cc0
1 /* Verify that unwinding can find SPE registers in signal frames. */
2 /* Origin: Joseph Myers <joseph@codesourcery.com> */
3 /* { dg-do run { target { powerpc*-*-linux* && powerpc_spe } } } */
4 /* { dg-options "-fexceptions -fnon-call-exceptions -O2" } */
6 #include <unwind.h>
7 #include <stdlib.h>
8 #include <signal.h>
9 #include <string.h>
11 int count;
12 char *null;
13 int found_reg;
15 typedef int v2si __attribute__((__vector_size__(8)));
17 v2si v1 = { 123, 234 };
18 v2si v2 = { 345, 456 };
20 static _Unwind_Reason_Code
21 force_unwind_stop (int version, _Unwind_Action actions,
22 _Unwind_Exception_Class exc_class,
23 struct _Unwind_Exception *exc_obj,
24 struct _Unwind_Context *context,
25 void *stop_parameter)
27 unsigned int reg;
28 if (actions & _UA_END_OF_STACK)
29 abort ();
30 if (_Unwind_GetGR (context, 1215) == 123)
31 found_reg = 1;
32 return _URC_NO_REASON;
35 static void force_unwind ()
37 struct _Unwind_Exception *exc = malloc (sizeof (*exc));
38 memset (&exc->exception_class, 0, sizeof (exc->exception_class));
39 exc->exception_cleanup = 0;
41 #ifndef __USING_SJLJ_EXCEPTIONS__
42 _Unwind_ForcedUnwind (exc, force_unwind_stop, 0);
43 #else
44 _Unwind_SjLj_ForcedUnwind (exc, force_unwind_stop, 0);
45 #endif
47 abort ();
50 static void counter (void *p __attribute__((unused)))
52 ++count;
55 static void handler (void *p __attribute__((unused)))
57 if (count != 2)
58 abort ();
59 if (!found_reg)
60 abort ();
61 exit (0);
64 static int __attribute__((noinline)) fn5 ()
66 char dummy __attribute__((cleanup (counter)));
67 force_unwind ();
68 return 0;
71 static void fn4 (int sig)
73 char dummy __attribute__((cleanup (counter)));
74 /* Clobber high part without compiler's knowledge so the only saved
75 copy is from the signal frame. */
76 asm volatile ("evmergelo 15,15,15");
77 fn5 ();
78 null = NULL;
81 static void fn3 ()
83 abort ();
86 static int __attribute__((noinline)) fn2 ()
88 register v2si r15 asm("r15");
89 r15 = v1;
90 asm volatile ("" : "+r" (r15));
91 *null = 0;
92 fn3 ();
93 return 0;
96 static int __attribute__((noinline)) fn1 ()
98 signal (SIGSEGV, fn4);
99 signal (SIGBUS, fn4);
100 fn2 ();
101 return 0;
104 static int __attribute__((noinline)) fn0 ()
106 char dummy __attribute__((cleanup (handler)));
107 fn1 ();
108 null = 0;
109 return 0;
112 int main()
114 fn0 ();
115 abort ();