blob: 0fc5d8563e19e9895d22ac151dc6fcc6dde8758c [file] [log] [blame]
Linus Torvalds1da177e2005-04-16 15:20:36 -07001/*
2 * linux/arch/x86_64/ia32/ia32_signal.c
3 *
4 * Copyright (C) 1991, 1992 Linus Torvalds
5 *
6 * 1997-11-28 Modified for POSIX.1b signals by Richard Henderson
7 * 2000-06-20 Pentium III FXSR, SSE support by Gareth Hughes
8 * 2000-12-* x86-64 compatibility mode signal handling by Andi Kleen
Linus Torvalds1da177e2005-04-16 15:20:36 -07009 */
10
11#include <linux/sched.h>
12#include <linux/mm.h>
13#include <linux/smp.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070014#include <linux/kernel.h>
15#include <linux/signal.h>
16#include <linux/errno.h>
17#include <linux/wait.h>
18#include <linux/ptrace.h>
19#include <linux/unistd.h>
20#include <linux/stddef.h>
21#include <linux/personality.h>
22#include <linux/compat.h>
Andi Kleen9fbbd4d2007-02-13 13:26:26 +010023#include <linux/binfmts.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070024#include <asm/ucontext.h>
25#include <asm/uaccess.h>
26#include <asm/i387.h>
27#include <asm/ia32.h>
28#include <asm/ptrace.h>
29#include <asm/ia32_unistd.h>
30#include <asm/user32.h>
31#include <asm/sigcontext32.h>
32#include <asm/fpu32.h>
33#include <asm/proto.h>
34#include <asm/vsyscall32.h>
35
36#define DEBUG_SIG 0
37
38#define _BLOCKABLE (~(sigmask(SIGKILL) | sigmask(SIGSTOP)))
39
40asmlinkage int do_signal(struct pt_regs *regs, sigset_t *oldset);
41void signal_fault(struct pt_regs *regs, void __user *frame, char *where);
42
43int copy_siginfo_to_user32(compat_siginfo_t __user *to, siginfo_t *from)
44{
45 int err;
Thomas Gleixner99b9cdf2008-01-30 13:30:07 +010046
47 if (!access_ok(VERIFY_WRITE, to, sizeof(compat_siginfo_t)))
Linus Torvalds1da177e2005-04-16 15:20:36 -070048 return -EFAULT;
49
50 /* If you change siginfo_t structure, please make sure that
51 this code is fixed accordingly.
52 It should never copy any pad contained in the structure
53 to avoid security leaks, but must copy the generic
54 3 ints plus the relevant union member. */
55 err = __put_user(from->si_signo, &to->si_signo);
56 err |= __put_user(from->si_errno, &to->si_errno);
Thomas Gleixner99b9cdf2008-01-30 13:30:07 +010057 err |= __put_user((short)from->si_code, &to->si_code);
Linus Torvalds1da177e2005-04-16 15:20:36 -070058
59 if (from->si_code < 0) {
60 err |= __put_user(from->si_pid, &to->si_pid);
Thomas Gleixner99b9cdf2008-01-30 13:30:07 +010061 err |= __put_user(from->si_uid, &to->si_uid);
62 err |= __put_user(ptr_to_compat(from->si_ptr), &to->si_ptr);
Linus Torvalds1da177e2005-04-16 15:20:36 -070063 } else {
Thomas Gleixner99b9cdf2008-01-30 13:30:07 +010064 /*
65 * First 32bits of unions are always present:
66 * si_pid === si_band === si_tid === si_addr(LS half)
67 */
68 err |= __put_user(from->_sifields._pad[0],
69 &to->_sifields._pad[0]);
Linus Torvalds1da177e2005-04-16 15:20:36 -070070 switch (from->si_code >> 16) {
71 case __SI_FAULT >> 16:
72 break;
73 case __SI_CHLD >> 16:
74 err |= __put_user(from->si_utime, &to->si_utime);
75 err |= __put_user(from->si_stime, &to->si_stime);
76 err |= __put_user(from->si_status, &to->si_status);
77 /* FALL THROUGH */
78 default:
79 case __SI_KILL >> 16:
80 err |= __put_user(from->si_uid, &to->si_uid);
81 break;
82 case __SI_POLL >> 16:
Thomas Gleixner99b9cdf2008-01-30 13:30:07 +010083 err |= __put_user(from->si_fd, &to->si_fd);
Linus Torvalds1da177e2005-04-16 15:20:36 -070084 break;
85 case __SI_TIMER >> 16:
Thomas Gleixner99b9cdf2008-01-30 13:30:07 +010086 err |= __put_user(from->si_overrun, &to->si_overrun);
Linus Torvalds1da177e2005-04-16 15:20:36 -070087 err |= __put_user(ptr_to_compat(from->si_ptr),
Thomas Gleixner99b9cdf2008-01-30 13:30:07 +010088 &to->si_ptr);
Linus Torvalds1da177e2005-04-16 15:20:36 -070089 break;
Thomas Gleixner99b9cdf2008-01-30 13:30:07 +010090 /* This is not generated by the kernel as of now. */
91 case __SI_RT >> 16:
Linus Torvalds1da177e2005-04-16 15:20:36 -070092 case __SI_MESGQ >> 16:
93 err |= __put_user(from->si_uid, &to->si_uid);
94 err |= __put_user(from->si_int, &to->si_int);
95 break;
96 }
97 }
98 return err;
99}
100
101int copy_siginfo_from_user32(siginfo_t *to, compat_siginfo_t __user *from)
102{
103 int err;
104 u32 ptr32;
Thomas Gleixner99b9cdf2008-01-30 13:30:07 +0100105
106 if (!access_ok(VERIFY_READ, from, sizeof(compat_siginfo_t)))
Linus Torvalds1da177e2005-04-16 15:20:36 -0700107 return -EFAULT;
108
109 err = __get_user(to->si_signo, &from->si_signo);
110 err |= __get_user(to->si_errno, &from->si_errno);
111 err |= __get_user(to->si_code, &from->si_code);
112
113 err |= __get_user(to->si_pid, &from->si_pid);
114 err |= __get_user(to->si_uid, &from->si_uid);
115 err |= __get_user(ptr32, &from->si_ptr);
116 to->si_ptr = compat_ptr(ptr32);
117
118 return err;
119}
120
Thomas Gleixner99b9cdf2008-01-30 13:30:07 +0100121asmlinkage long sys32_sigsuspend(int history0, int history1, old_sigset_t mask)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700122{
Linus Torvalds1da177e2005-04-16 15:20:36 -0700123 mask &= _BLOCKABLE;
124 spin_lock_irq(&current->sighand->siglock);
Andi Kleen1d001df2006-09-26 10:52:26 +0200125 current->saved_sigmask = current->blocked;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700126 siginitset(&current->blocked, mask);
127 recalc_sigpending();
128 spin_unlock_irq(&current->sighand->siglock);
129
Andi Kleen1d001df2006-09-26 10:52:26 +0200130 current->state = TASK_INTERRUPTIBLE;
131 schedule();
132 set_thread_flag(TIF_RESTORE_SIGMASK);
133 return -ERESTARTNOHAND;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700134}
135
Thomas Gleixner99b9cdf2008-01-30 13:30:07 +0100136asmlinkage long sys32_sigaltstack(const stack_ia32_t __user *uss_ptr,
137 stack_ia32_t __user *uoss_ptr,
138 struct pt_regs *regs)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700139{
Thomas Gleixner99b9cdf2008-01-30 13:30:07 +0100140 stack_t uss, uoss;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700141 int ret;
Thomas Gleixner99b9cdf2008-01-30 13:30:07 +0100142 mm_segment_t seg;
143
144 if (uss_ptr) {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700145 u32 ptr;
Thomas Gleixner99b9cdf2008-01-30 13:30:07 +0100146
147 memset(&uss, 0, sizeof(stack_t));
148 if (!access_ok(VERIFY_READ, uss_ptr, sizeof(stack_ia32_t)) ||
Linus Torvalds1da177e2005-04-16 15:20:36 -0700149 __get_user(ptr, &uss_ptr->ss_sp) ||
150 __get_user(uss.ss_flags, &uss_ptr->ss_flags) ||
151 __get_user(uss.ss_size, &uss_ptr->ss_size))
152 return -EFAULT;
153 uss.ss_sp = compat_ptr(ptr);
154 }
Thomas Gleixner99b9cdf2008-01-30 13:30:07 +0100155 seg = get_fs();
156 set_fs(KERNEL_DS);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700157 ret = do_sigaltstack(uss_ptr ? &uss : NULL, &uoss, regs->rsp);
Thomas Gleixner99b9cdf2008-01-30 13:30:07 +0100158 set_fs(seg);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700159 if (ret >= 0 && uoss_ptr) {
Thomas Gleixner99b9cdf2008-01-30 13:30:07 +0100160 if (!access_ok(VERIFY_WRITE, uoss_ptr, sizeof(stack_ia32_t)) ||
Linus Torvalds1da177e2005-04-16 15:20:36 -0700161 __put_user(ptr_to_compat(uoss.ss_sp), &uoss_ptr->ss_sp) ||
162 __put_user(uoss.ss_flags, &uoss_ptr->ss_flags) ||
163 __put_user(uoss.ss_size, &uoss_ptr->ss_size))
164 ret = -EFAULT;
Thomas Gleixner99b9cdf2008-01-30 13:30:07 +0100165 }
166 return ret;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700167}
168
169/*
170 * Do a signal return; undo the signal stack.
171 */
172
173struct sigframe
174{
175 u32 pretcode;
176 int sig;
177 struct sigcontext_ia32 sc;
178 struct _fpstate_ia32 fpstate;
179 unsigned int extramask[_COMPAT_NSIG_WORDS-1];
180 char retcode[8];
181};
182
183struct rt_sigframe
184{
185 u32 pretcode;
186 int sig;
187 u32 pinfo;
188 u32 puc;
189 compat_siginfo_t info;
190 struct ucontext_ia32 uc;
191 struct _fpstate_ia32 fpstate;
192 char retcode[8];
193};
194
Linus Torvalds1da177e2005-04-16 15:20:36 -0700195#define COPY(x) { \
196 unsigned int reg; \
197 err |= __get_user(reg, &sc->e ##x); \
198 regs->r ## x = reg; \
199}
200
201#define RELOAD_SEG(seg,mask) \
Thomas Gleixner99b9cdf2008-01-30 13:30:07 +0100202 { unsigned int cur; \
Linus Torvalds1da177e2005-04-16 15:20:36 -0700203 unsigned short pre; \
204 err |= __get_user(pre, &sc->seg); \
Thomas Gleixner99b9cdf2008-01-30 13:30:07 +0100205 asm volatile("movl %%" #seg ",%0" : "=r" (cur)); \
206 pre |= mask; \
207 if (pre != cur) loadsegment(seg, pre); }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700208
Thomas Gleixner99b9cdf2008-01-30 13:30:07 +0100209static int ia32_restore_sigcontext(struct pt_regs *regs,
210 struct sigcontext_ia32 __user *sc,
211 unsigned int *peax)
212{
213 unsigned int tmpflags, gs, oldgs, err = 0;
214 struct _fpstate_ia32 __user *buf;
215 u32 tmp;
216
217 /* Always make any pending restarted system calls return -EINTR */
218 current_thread_info()->restart_block.fn = do_no_restart_syscall;
219
220#if DEBUG_SIG
221 printk(KERN_DEBUG "SIG restore_sigcontext: "
222 "sc=%p err(%x) eip(%x) cs(%x) flg(%x)\n",
223 sc, sc->err, sc->eip, sc->cs, sc->eflags);
224#endif
225
226 /*
227 * Reload fs and gs if they have changed in the signal
228 * handler. This does not handle long fs/gs base changes in
229 * the handler, but does not clobber them at least in the
230 * normal case.
231 */
232 err |= __get_user(gs, &sc->gs);
233 gs |= 3;
234 asm("movl %%gs,%0" : "=r" (oldgs));
235 if (gs != oldgs)
236 load_gs_index(gs);
237
238 RELOAD_SEG(fs, 3);
239 RELOAD_SEG(ds, 3);
240 RELOAD_SEG(es, 3);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700241
242 COPY(di); COPY(si); COPY(bp); COPY(sp); COPY(bx);
243 COPY(dx); COPY(cx); COPY(ip);
Thomas Gleixner99b9cdf2008-01-30 13:30:07 +0100244 /* Don't touch extended registers */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700245
Thomas Gleixner99b9cdf2008-01-30 13:30:07 +0100246 err |= __get_user(regs->cs, &sc->cs);
247 regs->cs |= 3;
248 err |= __get_user(regs->ss, &sc->ss);
249 regs->ss |= 3;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700250
Thomas Gleixner99b9cdf2008-01-30 13:30:07 +0100251 err |= __get_user(tmpflags, &sc->eflags);
252 regs->eflags = (regs->eflags & ~0x40DD5) | (tmpflags & 0x40DD5);
253 /* disable syscall checks */
254 regs->orig_rax = -1;
255
256 err |= __get_user(tmp, &sc->fpstate);
257 buf = compat_ptr(tmp);
258 if (buf) {
259 if (!access_ok(VERIFY_READ, buf, sizeof(*buf)))
260 goto badframe;
261 err |= restore_i387_ia32(current, buf, 0);
262 } else {
263 struct task_struct *me = current;
264
265 if (used_math()) {
266 clear_fpu(me);
267 clear_used_math();
Linus Torvalds1da177e2005-04-16 15:20:36 -0700268 }
269 }
270
Thomas Gleixner99b9cdf2008-01-30 13:30:07 +0100271 err |= __get_user(tmp, &sc->eax);
272 *peax = tmp;
273
Linus Torvalds1da177e2005-04-16 15:20:36 -0700274 return err;
275
276badframe:
277 return 1;
278}
279
280asmlinkage long sys32_sigreturn(struct pt_regs *regs)
281{
282 struct sigframe __user *frame = (struct sigframe __user *)(regs->rsp-8);
283 sigset_t set;
284 unsigned int eax;
285
286 if (!access_ok(VERIFY_READ, frame, sizeof(*frame)))
287 goto badframe;
288 if (__get_user(set.sig[0], &frame->sc.oldmask)
289 || (_COMPAT_NSIG_WORDS > 1
Thomas Gleixner99b9cdf2008-01-30 13:30:07 +0100290 && __copy_from_user((((char *) &set.sig) + 4),
291 &frame->extramask,
Linus Torvalds1da177e2005-04-16 15:20:36 -0700292 sizeof(frame->extramask))))
293 goto badframe;
294
295 sigdelsetmask(&set, ~_BLOCKABLE);
296 spin_lock_irq(&current->sighand->siglock);
297 current->blocked = set;
298 recalc_sigpending();
299 spin_unlock_irq(&current->sighand->siglock);
Thomas Gleixner99b9cdf2008-01-30 13:30:07 +0100300
Linus Torvalds1da177e2005-04-16 15:20:36 -0700301 if (ia32_restore_sigcontext(regs, &frame->sc, &eax))
302 goto badframe;
303 return eax;
304
305badframe:
306 signal_fault(regs, frame, "32bit sigreturn");
307 return 0;
Thomas Gleixner99b9cdf2008-01-30 13:30:07 +0100308}
Linus Torvalds1da177e2005-04-16 15:20:36 -0700309
310asmlinkage long sys32_rt_sigreturn(struct pt_regs *regs)
311{
312 struct rt_sigframe __user *frame;
313 sigset_t set;
314 unsigned int eax;
315 struct pt_regs tregs;
316
317 frame = (struct rt_sigframe __user *)(regs->rsp - 4);
318
319 if (!access_ok(VERIFY_READ, frame, sizeof(*frame)))
320 goto badframe;
321 if (__copy_from_user(&set, &frame->uc.uc_sigmask, sizeof(set)))
322 goto badframe;
323
324 sigdelsetmask(&set, ~_BLOCKABLE);
325 spin_lock_irq(&current->sighand->siglock);
326 current->blocked = set;
327 recalc_sigpending();
328 spin_unlock_irq(&current->sighand->siglock);
Thomas Gleixner99b9cdf2008-01-30 13:30:07 +0100329
Linus Torvalds1da177e2005-04-16 15:20:36 -0700330 if (ia32_restore_sigcontext(regs, &frame->uc.uc_mcontext, &eax))
331 goto badframe;
332
333 tregs = *regs;
334 if (sys32_sigaltstack(&frame->uc.uc_stack, NULL, &tregs) == -EFAULT)
335 goto badframe;
336
337 return eax;
338
339badframe:
Thomas Gleixner99b9cdf2008-01-30 13:30:07 +0100340 signal_fault(regs, frame, "32bit rt sigreturn");
Linus Torvalds1da177e2005-04-16 15:20:36 -0700341 return 0;
Thomas Gleixner99b9cdf2008-01-30 13:30:07 +0100342}
Linus Torvalds1da177e2005-04-16 15:20:36 -0700343
344/*
345 * Set up a signal frame.
346 */
347
Thomas Gleixner99b9cdf2008-01-30 13:30:07 +0100348static int ia32_setup_sigcontext(struct sigcontext_ia32 __user *sc,
349 struct _fpstate_ia32 __user *fpstate,
350 struct pt_regs *regs, unsigned int mask)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700351{
352 int tmp, err = 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700353
354 tmp = 0;
355 __asm__("movl %%gs,%0" : "=r"(tmp): "0"(tmp));
356 err |= __put_user(tmp, (unsigned int __user *)&sc->gs);
357 __asm__("movl %%fs,%0" : "=r"(tmp): "0"(tmp));
358 err |= __put_user(tmp, (unsigned int __user *)&sc->fs);
359 __asm__("movl %%ds,%0" : "=r"(tmp): "0"(tmp));
360 err |= __put_user(tmp, (unsigned int __user *)&sc->ds);
361 __asm__("movl %%es,%0" : "=r"(tmp): "0"(tmp));
362 err |= __put_user(tmp, (unsigned int __user *)&sc->es);
363
364 err |= __put_user((u32)regs->rdi, &sc->edi);
365 err |= __put_user((u32)regs->rsi, &sc->esi);
366 err |= __put_user((u32)regs->rbp, &sc->ebp);
367 err |= __put_user((u32)regs->rsp, &sc->esp);
368 err |= __put_user((u32)regs->rbx, &sc->ebx);
369 err |= __put_user((u32)regs->rdx, &sc->edx);
370 err |= __put_user((u32)regs->rcx, &sc->ecx);
371 err |= __put_user((u32)regs->rax, &sc->eax);
372 err |= __put_user((u32)regs->cs, &sc->cs);
373 err |= __put_user((u32)regs->ss, &sc->ss);
374 err |= __put_user(current->thread.trap_no, &sc->trapno);
375 err |= __put_user(current->thread.error_code, &sc->err);
376 err |= __put_user((u32)regs->rip, &sc->eip);
Peter Beutner4724e3e2006-01-11 22:43:18 +0100377 err |= __put_user((u32)regs->eflags, &sc->eflags);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700378 err |= __put_user((u32)regs->rsp, &sc->esp_at_signal);
379
380 tmp = save_i387_ia32(current, fpstate, regs, 0);
381 if (tmp < 0)
382 err = -EFAULT;
Thomas Gleixner99b9cdf2008-01-30 13:30:07 +0100383 else {
Linus Torvalds1da177e2005-04-16 15:20:36 -0700384 clear_used_math();
385 stts();
386 err |= __put_user(ptr_to_compat(tmp ? fpstate : NULL),
387 &sc->fpstate);
388 }
389
390 /* non-iBCS2 extensions.. */
391 err |= __put_user(mask, &sc->oldmask);
392 err |= __put_user(current->thread.cr2, &sc->cr2);
393
394 return err;
395}
396
397/*
398 * Determine which stack to use..
399 */
Thomas Gleixner99b9cdf2008-01-30 13:30:07 +0100400static void __user *get_sigframe(struct k_sigaction *ka, struct pt_regs *regs,
401 size_t frame_size)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700402{
403 unsigned long rsp;
404
405 /* Default to using normal stack */
406 rsp = regs->rsp;
407
408 /* This is the X/Open sanctioned signal stack switching. */
409 if (ka->sa.sa_flags & SA_ONSTACK) {
410 if (sas_ss_flags(rsp) == 0)
411 rsp = current->sas_ss_sp + current->sas_ss_size;
412 }
413
414 /* This is the legacy signal stack switching. */
415 else if ((regs->ss & 0xffff) != __USER_DS &&
416 !(ka->sa.sa_flags & SA_RESTORER) &&
Thomas Gleixner99b9cdf2008-01-30 13:30:07 +0100417 ka->sa.sa_restorer)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700418 rsp = (unsigned long) ka->sa.sa_restorer;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700419
Markus F.X.J. Oberhumerd347f372005-10-09 18:54:23 +0200420 rsp -= frame_size;
421 /* Align the stack pointer according to the i386 ABI,
422 * i.e. so that on function entry ((sp + 4) & 15) == 0. */
423 rsp = ((rsp + 4) & -16ul) - 4;
424 return (void __user *) rsp;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700425}
426
Roland McGrath0928d6e2005-06-23 00:08:37 -0700427int ia32_setup_frame(int sig, struct k_sigaction *ka,
Thomas Gleixner99b9cdf2008-01-30 13:30:07 +0100428 compat_sigset_t *set, struct pt_regs *regs)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700429{
430 struct sigframe __user *frame;
Thomas Gleixner99b9cdf2008-01-30 13:30:07 +0100431 void __user *restorer;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700432 int err = 0;
433
Thomas Gleixner99b9cdf2008-01-30 13:30:07 +0100434 /* copy_to_user optimizes that into a single 8 byte store */
435 static const struct {
436 u16 poplmovl;
437 u32 val;
438 u16 int80;
439 u16 pad;
440 } __attribute__((packed)) code = {
441 0xb858, /* popl %eax ; movl $...,%eax */
442 __NR_ia32_sigreturn,
443 0x80cd, /* int $0x80 */
444 0,
445 };
446
Linus Torvalds1da177e2005-04-16 15:20:36 -0700447 frame = get_sigframe(ka, regs, sizeof(*frame));
448
449 if (!access_ok(VERIFY_WRITE, frame, sizeof(*frame)))
450 goto give_sigsegv;
451
Andi Kleendd54a112006-09-26 10:52:41 +0200452 err |= __put_user(sig, &frame->sig);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700453 if (err)
454 goto give_sigsegv;
455
456 err |= ia32_setup_sigcontext(&frame->sc, &frame->fpstate, regs,
457 set->sig[0]);
458 if (err)
459 goto give_sigsegv;
460
461 if (_COMPAT_NSIG_WORDS > 1) {
462 err |= __copy_to_user(frame->extramask, &set->sig[1],
463 sizeof(frame->extramask));
Thomas Gleixner99b9cdf2008-01-30 13:30:07 +0100464 if (err)
465 goto give_sigsegv;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700466 }
Linus Torvalds1da177e2005-04-16 15:20:36 -0700467
468 /* Return stub is in 32bit vsyscall page */
Thomas Gleixner99b9cdf2008-01-30 13:30:07 +0100469 if (current->binfmt->hasvdso)
470 restorer = VSYSCALL32_SIGRETURN;
471 else
472 restorer = (void *)&frame->retcode;
473 if (ka->sa.sa_flags & SA_RESTORER)
474 restorer = ka->sa.sa_restorer;
475 err |= __put_user(ptr_to_compat(restorer), &frame->pretcode);
476
477 /*
478 * These are actually not used anymore, but left because some
479 * gdb versions depend on them as a marker.
480 */
481 err |= __copy_to_user(frame->retcode, &code, 8);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700482 if (err)
483 goto give_sigsegv;
484
485 /* Set up registers for signal handler */
486 regs->rsp = (unsigned long) frame;
487 regs->rip = (unsigned long) ka->sa.sa_handler;
488
Andi Kleen536e3ee2006-09-26 10:52:41 +0200489 /* Make -mregparm=3 work */
490 regs->rax = sig;
491 regs->rdx = 0;
492 regs->rcx = 0;
493
Thomas Gleixner99b9cdf2008-01-30 13:30:07 +0100494 asm volatile("movl %0,%%ds" :: "r" (__USER32_DS));
495 asm volatile("movl %0,%%es" :: "r" (__USER32_DS));
Linus Torvalds1da177e2005-04-16 15:20:36 -0700496
Thomas Gleixner99b9cdf2008-01-30 13:30:07 +0100497 regs->cs = __USER32_CS;
498 regs->ss = __USER32_DS;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700499
500 set_fs(USER_DS);
Andi Kleenab2e0b42006-09-26 10:52:41 +0200501 regs->eflags &= ~TF_MASK;
502 if (test_thread_flag(TIF_SINGLESTEP))
503 ptrace_notify(SIGTRAP);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700504
505#if DEBUG_SIG
Thomas Gleixner99b9cdf2008-01-30 13:30:07 +0100506 printk(KERN_DEBUG "SIG deliver (%s:%d): sp=%p pc=%lx ra=%u\n",
507 current->comm, current->pid, frame, regs->rip, frame->pretcode);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700508#endif
509
Andi Kleen1d001df2006-09-26 10:52:26 +0200510 return 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700511
512give_sigsegv:
513 force_sigsegv(sig, current);
Andi Kleen1d001df2006-09-26 10:52:26 +0200514 return -EFAULT;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700515}
516
Roland McGrath0928d6e2005-06-23 00:08:37 -0700517int ia32_setup_rt_frame(int sig, struct k_sigaction *ka, siginfo_t *info,
Thomas Gleixner99b9cdf2008-01-30 13:30:07 +0100518 compat_sigset_t *set, struct pt_regs *regs)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700519{
520 struct rt_sigframe __user *frame;
Thomas Gleixner99b9cdf2008-01-30 13:30:07 +0100521 struct exec_domain *ed = current_thread_info()->exec_domain;
522 void __user *restorer = VSYSCALL32_RTSIGRETURN;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700523 int err = 0;
524
Thomas Gleixner99b9cdf2008-01-30 13:30:07 +0100525 /* __copy_to_user optimizes that into a single 8 byte store */
526 static const struct {
527 u8 movl;
528 u32 val;
529 u16 int80;
530 u16 pad;
531 u8 pad2;
532 } __attribute__((packed)) code = {
533 0xb8,
534 __NR_ia32_rt_sigreturn,
535 0x80cd,
536 0,
537 };
538
Linus Torvalds1da177e2005-04-16 15:20:36 -0700539 frame = get_sigframe(ka, regs, sizeof(*frame));
540
541 if (!access_ok(VERIFY_WRITE, frame, sizeof(*frame)))
542 goto give_sigsegv;
543
Thomas Gleixner99b9cdf2008-01-30 13:30:07 +0100544 err |= __put_user((ed && ed->signal_invmap && sig < 32
545 ? ed->signal_invmap[sig] : sig), &frame->sig);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700546 err |= __put_user(ptr_to_compat(&frame->info), &frame->pinfo);
547 err |= __put_user(ptr_to_compat(&frame->uc), &frame->puc);
548 err |= copy_siginfo_to_user32(&frame->info, info);
549 if (err)
550 goto give_sigsegv;
551
552 /* Create the ucontext. */
553 err |= __put_user(0, &frame->uc.uc_flags);
554 err |= __put_user(0, &frame->uc.uc_link);
555 err |= __put_user(current->sas_ss_sp, &frame->uc.uc_stack.ss_sp);
556 err |= __put_user(sas_ss_flags(regs->rsp),
557 &frame->uc.uc_stack.ss_flags);
558 err |= __put_user(current->sas_ss_size, &frame->uc.uc_stack.ss_size);
559 err |= ia32_setup_sigcontext(&frame->uc.uc_mcontext, &frame->fpstate,
Thomas Gleixner99b9cdf2008-01-30 13:30:07 +0100560 regs, set->sig[0]);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700561 err |= __copy_to_user(&frame->uc.uc_sigmask, set, sizeof(*set));
562 if (err)
563 goto give_sigsegv;
564
Thomas Gleixner99b9cdf2008-01-30 13:30:07 +0100565 if (ka->sa.sa_flags & SA_RESTORER)
566 restorer = ka->sa.sa_restorer;
567 err |= __put_user(ptr_to_compat(restorer), &frame->pretcode);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700568
Thomas Gleixner99b9cdf2008-01-30 13:30:07 +0100569 /*
570 * Not actually used anymore, but left because some gdb
571 * versions need it.
572 */
573 err |= __copy_to_user(frame->retcode, &code, 8);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700574 if (err)
575 goto give_sigsegv;
576
577 /* Set up registers for signal handler */
578 regs->rsp = (unsigned long) frame;
579 regs->rip = (unsigned long) ka->sa.sa_handler;
580
Albert Cahalana7aacdf2006-10-29 22:26:17 -0500581 /* Make -mregparm=3 work */
582 regs->rax = sig;
583 regs->rdx = (unsigned long) &frame->info;
584 regs->rcx = (unsigned long) &frame->uc;
585
Albert Cahalan8e3de532006-12-07 02:14:06 +0100586 /* Make -mregparm=3 work */
587 regs->rax = sig;
588 regs->rdx = (unsigned long) &frame->info;
589 regs->rcx = (unsigned long) &frame->uc;
590
Thomas Gleixner99b9cdf2008-01-30 13:30:07 +0100591 asm volatile("movl %0,%%ds" :: "r" (__USER32_DS));
592 asm volatile("movl %0,%%es" :: "r" (__USER32_DS));
593
594 regs->cs = __USER32_CS;
595 regs->ss = __USER32_DS;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700596
597 set_fs(USER_DS);
Andi Kleen1d001df2006-09-26 10:52:26 +0200598 regs->eflags &= ~TF_MASK;
Andi Kleenab2e0b42006-09-26 10:52:41 +0200599 if (test_thread_flag(TIF_SINGLESTEP))
600 ptrace_notify(SIGTRAP);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700601
602#if DEBUG_SIG
Thomas Gleixner99b9cdf2008-01-30 13:30:07 +0100603 printk(KERN_DEBUG "SIG deliver (%s:%d): sp=%p pc=%lx ra=%u\n",
604 current->comm, current->pid, frame, regs->rip, frame->pretcode);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700605#endif
606
Andi Kleen1d001df2006-09-26 10:52:26 +0200607 return 0;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700608
609give_sigsegv:
610 force_sigsegv(sig, current);
Andi Kleen1d001df2006-09-26 10:52:26 +0200611 return -EFAULT;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700612}