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