OSDN Git Service

* gcc.dg/cleanup-10.c, gcc.dg/cleanup-11.c, gcc.dg/cleanup-8.c,
[pf3gnuchains/gcc-fork.git] / gcc / testsuite / gcc.dg / cleanup-11.c
1 /* { dg-do run { target *-*-linux* powerpc*-*-darwin* } } */
2 /* { dg-options "-fexceptions -fnon-call-exceptions -O2" } */
3 /* Verify that cleanups work with exception handling through realtime signal
4    frames on alternate stack.  */
5
6 #include <unwind.h>
7 #include <stdlib.h>
8 #include <signal.h>
9 #include <unistd.h>
10
11 static _Unwind_Reason_Code
12 force_unwind_stop (int version, _Unwind_Action actions,
13                    _Unwind_Exception_Class exc_class,
14                    struct _Unwind_Exception *exc_obj,
15                    struct _Unwind_Context *context,
16                    void *stop_parameter)
17 {
18   if (actions & _UA_END_OF_STACK)
19     abort ();
20   return _URC_NO_REASON;
21 }
22
23 static void force_unwind ()
24 {
25   struct _Unwind_Exception *exc = malloc (sizeof (*exc));
26   exc->exception_class = 0;
27   exc->exception_cleanup = 0;
28                    
29 #ifndef __USING_SJLJ_EXCEPTIONS__
30   _Unwind_ForcedUnwind (exc, force_unwind_stop, 0);
31 #else
32   _Unwind_SjLj_ForcedUnwind (exc, force_unwind_stop, 0);
33 #endif
34                    
35   abort ();
36 }
37
38 int count;
39 char *null;
40
41 static void counter (void *p __attribute__((unused)))
42 {
43   ++count;
44 }
45
46 static void handler (void *p __attribute__((unused)))
47 {
48   if (count != 2)
49     abort ();
50   exit (0);
51 }
52
53 static int __attribute__((noinline)) fn5 ()
54 {
55   char dummy __attribute__((cleanup (counter)));
56   force_unwind ();
57   return 0;
58 }
59
60 static void fn4 (int sig, siginfo_t *info, void *ctx)
61 {
62   char dummy __attribute__((cleanup (counter)));
63   fn5 ();
64   null = NULL;
65 }
66
67 static void fn3 ()
68 {
69   abort ();
70 }
71
72 static int __attribute__((noinline)) fn2 ()
73 {
74   *null = 0;
75   fn3 ();
76   return 0;
77 }
78
79 static int __attribute__((noinline)) fn1 ()
80 {
81   stack_t ss;
82   struct sigaction s;
83
84   ss.ss_size = 4 * sysconf (_SC_PAGESIZE);
85   if (ss.ss_size < SIGSTKSZ)
86     ss.ss_size = SIGSTKSZ;
87   ss.ss_sp = malloc (ss.ss_size);
88   if (ss.ss_sp == NULL)
89     exit (1);
90   ss.ss_flags = 0;
91   if (sigaltstack (&ss, NULL) < 0)
92     exit (1);
93
94   sigemptyset (&s.sa_mask);
95   s.sa_sigaction = fn4;
96   s.sa_flags = SA_RESETHAND | SA_ONSTACK | SA_SIGINFO;
97   sigaction (SIGSEGV, &s, NULL);
98   sigaction (SIGBUS, &s, NULL);
99   fn2 ();
100   return 0;
101 }
102
103 static int __attribute__((noinline)) fn0 ()
104 {
105   char dummy __attribute__((cleanup (handler)));
106   fn1 ();
107   null = 0;
108   return 0;
109 }
110
111 int main()
112
113   fn0 ();
114   abort ();
115 }