1 /*
2 * Copyright (c) 2009 Cisco Systems, Inc. All Rights Reserved.
3 * Copyright (c) 2009 FUJITSU LIMITED. All Rights Reserved.
4 *
5 * This program is free software; you can redistribute it and/or modify it
6 * under the terms of version 2 of the GNU General Public License as
7 * published by the Free Software Foundation.
8 *
9 * This program is distributed in the hope that it would be useful, but
10 * WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
12 *
13 * Further, this software is distributed without any warranty that it is
14 * free of the rightful claim of any third person regarding infringement
15 * or the like. Any license provided herein, whether implied or
16 * otherwise, applies only to this software file. Patent licenses, if
17 * any, provided herein do not apply to combinations of this program with
18 * other software, or any other product whatsoever.
19 *
20 * You should have received a copy of the GNU General Public License along
21 * with this program; if not, write the Free Software Foundation, Inc.,
22 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
23 *
24 * Author: Liu Bo <liubo2009@cn.fujitsu.com>
25 * Author: Ngie Cooper <yaneurabeya@gmail.com>
26 *
27 */
28
29 #ifndef LTP_RT_SIGACTION_H
30 #define LTP_RT_SIGACTION_H
31
32 #include "ltp_signal.h"
33
34 #define INVAL_SA_PTR ((void *)-1)
35
36 #if defined(__mips__)
37 struct kernel_sigaction {
38 unsigned int sa_flags;
39 __sighandler_t k_sa_handler;
40 sigset_t sa_mask;
41 };
42 #else
43 struct kernel_sigaction {
44 __sighandler_t k_sa_handler;
45 unsigned long sa_flags;
46 void (*sa_restorer) (void);
47 sigset_t sa_mask;
48 };
49 #endif
50
51 /* This macro marks if (struct sigaction) has .sa_restorer member */
52 #if !defined(__ia64__) && !defined(__alpha__) && !defined(__hppa__) && !defined(__mips__)
53 # define HAVE_SA_RESTORER
54 #endif
55
56 #ifdef __x86_64__
57
58 /*
59 * From asm/signal.h -- this value isn't exported anywhere outside of glibc and
60 * asm/signal.h and is only required for the rt_sig* function family because
61 * sigaction(2), et all, appends this if necessary to
62 * (struct sigaction).sa_flags. HEH.
63 *
64 * I do #undef though, just in case...
65 *
66 * Also, from .../arch/x86/kernel/signal.c:448 for v2.6.30 (something or
67 * other):
68 *
69 * x86-64 should always use SA_RESTORER.
70 *
71 * -- thus SA_RESTORER must always be defined along with
72 * (struct sigaction).sa_restorer for this architecture.
73 */
74 #undef SA_RESTORER
75 #define SA_RESTORER 0x04000000
76
77 void (*restore_rt)(void);
78
handler_h(int signal)79 static void handler_h(int signal)
80 {
81 return;
82 }
83
84 /* Setup an initial signal handler for signal number = sig for x86_64. */
sig_initial(int sig)85 static inline int sig_initial(int sig)
86 {
87 int ret_code = -1;
88 struct sigaction act, oact;
89
90 act.sa_handler = handler_h;
91 act.sa_flags = 0;
92 /* Clear out the signal set. */
93 if (sigemptyset(&act.sa_mask) < 0) {
94 /* Add the signal to the mask set. */
95 } else if (sigaddset(&act.sa_mask, sig) < 0) {
96 /* Set act.sa_restorer via syscall(2) */
97 } else if (sigaction(sig, &act, &oact) < 0) {
98 /* Copy oact.sa_restorer via syscall(2) */
99 } else if (sigaction(sig, &act, &oact) < 0) {
100 /* And voila -- we just tricked the kernel into giving us our
101 * restorer function! */
102 } else {
103 restore_rt = oact.sa_restorer;
104 ret_code = 0;
105 }
106
107 return ret_code;
108 }
109
110 #endif /* __x86_64__ */
111
112 #ifdef __sparc__
113 # if defined __arch64__ || defined __sparcv9
114
115 /*
116 * Based on glibc/sysdeps/unix/sysv/linux/sparc/sparc64/sigaction.c
117 */
118
119 extern char *__rt_sig_stub;
120
__rt_sigreturn_stub(void)121 static void __attribute__((used)) __rt_sigreturn_stub(void)
122 {
123 __asm__ ("__rt_sig_stub: mov %0, %%g1\n\t"
124 "ta 0x6d\n\t"
125 : /* no outputs */
126 : "i" (__NR_rt_sigreturn));
127 }
128
129 # else /* sparc32 */
130
131 /*
132 * Based on glibc/sysdeps/unix/sysv/linux/sparc/sparc32/sigaction.c
133 */
134
135 extern char *__rt_sig_stub, *__sig_stub;
136
__rt_sigreturn_stub(void)137 static void __attribute__((used)) __rt_sigreturn_stub(void)
138 {
139 __asm__ ("__rt_sig_stub: mov %0, %%g1\n\t"
140 "ta 0x10\n\t"
141 : /* no outputs */
142 : "i" (__NR_rt_sigreturn));
143 }
144
__sigreturn_stub(void)145 static void __attribute__((used)) __sigreturn_stub(void)
146 {
147 __asm__ ("__sig_stub: mov %0, %%g1\n\t"
148 "ta 0x10\n\t"
149 : /* no outputs */
150 : "i" (__NR_sigreturn));
151 }
152
153 # endif
154 #endif /* __sparc__ */
155
156 #ifdef __arc__
157
158 #undef SA_RESTORER
159 #define SA_RESTORER 0x04000000
160
161 /*
162 * based on uClibc/libc/sysdeps/linux/arc/sigaction.c
163 */
164 static void
restore_rt(void)165 __attribute__ ((optimize("Os"))) __attribute__((used)) restore_rt(void)
166 {
167 __asm__ (
168 "mov r8, %0 \n\t"
169 #ifdef __ARCHS__
170 "trap_s 0 \n\t"
171 #else
172 "trap0 \n\t"
173 #endif
174 : /* no outputs */
175 : "i" (__NR_rt_sigreturn)
176 : "r8");
177 }
178 #endif
179
180 #ifdef TST_TEST_H__
181 # define TST_SYSCALL tst_syscall
182 #else
183 # define TST_SYSCALL ltp_syscall
184 #endif
185
186 /* This is a wrapper for __NR_rt_sigaction syscall.
187 * act/oact values of INVAL_SA_PTR is used to pass
188 * an invalid pointer to syscall(__NR_rt_sigaction)
189 *
190 * Based on glibc/sysdeps/unix/sysv/linux/{...}/sigaction.c
191 */
192
ltp_rt_sigaction(int signum,const struct sigaction * act,struct sigaction * oact,size_t sigsetsize)193 static int ltp_rt_sigaction(int signum, const struct sigaction *act,
194 struct sigaction *oact, size_t sigsetsize)
195 {
196 int ret;
197 struct kernel_sigaction kact, koact;
198 struct kernel_sigaction *kact_p = NULL;
199 struct kernel_sigaction *koact_p = NULL;
200
201 if (act == INVAL_SA_PTR) {
202 kact_p = INVAL_SA_PTR;
203 } else if (act) {
204 kact.k_sa_handler = act->sa_handler;
205 memcpy(&kact.sa_mask, &act->sa_mask, sizeof(sigset_t));
206 kact.sa_flags = act->sa_flags;
207 #ifndef __mips__
208 kact.sa_restorer = NULL;
209 #endif
210 kact_p = &kact;
211 }
212
213 if (oact == INVAL_SA_PTR)
214 koact_p = INVAL_SA_PTR;
215 else if (oact)
216 koact_p = &koact;
217
218 #ifdef __x86_64__
219 sig_initial(signum);
220 #endif
221
222 #if defined __x86_64__ || defined __arc__
223 kact.sa_flags |= SA_RESTORER;
224 kact.sa_restorer = restore_rt;
225 #endif
226
227 #ifdef __sparc__
228 unsigned long stub = 0;
229 # if defined __arch64__ || defined __sparcv9
230 stub = ((unsigned long) &__rt_sig_stub) - 8;
231 # else /* sparc32 */
232 if ((kact.sa_flags & SA_SIGINFO) != 0)
233 stub = ((unsigned long) &__rt_sig_stub) - 8;
234 else
235 stub = ((unsigned long) &__sig_stub) - 8;
236 # endif
237 #endif
238
239
240 #ifdef __sparc__
241 ret = TST_SYSCALL(__NR_rt_sigaction, signum,
242 kact_p, koact_p,
243 stub, sigsetsize);
244 #else
245 ret = TST_SYSCALL(__NR_rt_sigaction, signum,
246 kact_p, koact_p,
247 sigsetsize);
248 #endif
249
250 if (ret >= 0) {
251 if (oact && (oact != INVAL_SA_PTR)) {
252 oact->sa_handler = koact.k_sa_handler;
253 memcpy(&oact->sa_mask, &koact.sa_mask,
254 sizeof(sigset_t));
255 oact->sa_flags = koact.sa_flags;
256 #ifdef HAVE_SA_RESTORER
257 oact->sa_restorer = koact.sa_restorer;
258 #endif
259 }
260 }
261
262 return ret;
263 }
264
265 #endif /* LTP_RT_SIGACTION_H */
266