OMAP: DSS2: fix irq-stats compilation
[linux/fpc-iii.git] / arch / sh / kernel / signal_64.c
blobce76dbdef2940fa8e920dcbc8f4bc265a0607b5f
1 /*
2 * arch/sh/kernel/signal_64.c
4 * Copyright (C) 2000, 2001 Paolo Alberelli
5 * Copyright (C) 2003 - 2008 Paul Mundt
6 * Copyright (C) 2004 Richard Curnow
8 * This file is subject to the terms and conditions of the GNU General Public
9 * License. See the file "COPYING" in the main directory of this archive
10 * for more details.
12 #include <linux/rwsem.h>
13 #include <linux/sched.h>
14 #include <linux/mm.h>
15 #include <linux/smp.h>
16 #include <linux/kernel.h>
17 #include <linux/signal.h>
18 #include <linux/errno.h>
19 #include <linux/wait.h>
20 #include <linux/personality.h>
21 #include <linux/freezer.h>
22 #include <linux/ptrace.h>
23 #include <linux/unistd.h>
24 #include <linux/stddef.h>
25 #include <linux/tracehook.h>
26 #include <asm/ucontext.h>
27 #include <asm/uaccess.h>
28 #include <asm/pgtable.h>
29 #include <asm/cacheflush.h>
30 #include <asm/fpu.h>
32 #define REG_RET 9
33 #define REG_ARG1 2
34 #define REG_ARG2 3
35 #define REG_ARG3 4
36 #define REG_SP 15
37 #define REG_PR 18
38 #define REF_REG_RET regs->regs[REG_RET]
39 #define REF_REG_SP regs->regs[REG_SP]
40 #define DEREF_REG_PR regs->regs[REG_PR]
42 #define DEBUG_SIG 0
44 #define _BLOCKABLE (~(sigmask(SIGKILL) | sigmask(SIGSTOP)))
46 static int
47 handle_signal(unsigned long sig, siginfo_t *info, struct k_sigaction *ka,
48 sigset_t *oldset, struct pt_regs * regs);
50 static inline void
51 handle_syscall_restart(struct pt_regs *regs, struct sigaction *sa)
53 /* If we're not from a syscall, bail out */
54 if (regs->syscall_nr < 0)
55 return;
57 /* check for system call restart.. */
58 switch (regs->regs[REG_RET]) {
59 case -ERESTART_RESTARTBLOCK:
60 case -ERESTARTNOHAND:
61 no_system_call_restart:
62 regs->regs[REG_RET] = -EINTR;
63 break;
65 case -ERESTARTSYS:
66 if (!(sa->sa_flags & SA_RESTART))
67 goto no_system_call_restart;
68 /* fallthrough */
69 case -ERESTARTNOINTR:
70 /* Decode syscall # */
71 regs->regs[REG_RET] = regs->syscall_nr;
72 regs->pc -= 4;
73 break;
78 * Note that 'init' is a special process: it doesn't get signals it doesn't
79 * want to handle. Thus you cannot kill init even with a SIGKILL even by
80 * mistake.
82 * Note that we go through the signals twice: once to check the signals that
83 * the kernel can handle, and then we build all the user-level signal handling
84 * stack-frames in one go after that.
86 static int do_signal(struct pt_regs *regs, sigset_t *oldset)
88 siginfo_t info;
89 int signr;
90 struct k_sigaction ka;
93 * We want the common case to go fast, which
94 * is why we may in certain cases get here from
95 * kernel mode. Just return without doing anything
96 * if so.
98 if (!user_mode(regs))
99 return 1;
101 if (try_to_freeze())
102 goto no_signal;
104 if (current_thread_info()->status & TS_RESTORE_SIGMASK)
105 oldset = &current->saved_sigmask;
106 else if (!oldset)
107 oldset = &current->blocked;
109 signr = get_signal_to_deliver(&info, &ka, regs, 0);
110 if (signr > 0) {
111 handle_syscall_restart(regs, &ka.sa);
113 /* Whee! Actually deliver the signal. */
114 if (handle_signal(signr, &info, &ka, oldset, regs) == 0) {
116 * If a signal was successfully delivered, the
117 * saved sigmask is in its frame, and we can
118 * clear the TS_RESTORE_SIGMASK flag.
120 current_thread_info()->status &= ~TS_RESTORE_SIGMASK;
121 tracehook_signal_handler(signr, &info, &ka, regs, 0);
122 return 1;
126 no_signal:
127 /* Did we come from a system call? */
128 if (regs->syscall_nr >= 0) {
129 /* Restart the system call - no handlers present */
130 switch (regs->regs[REG_RET]) {
131 case -ERESTARTNOHAND:
132 case -ERESTARTSYS:
133 case -ERESTARTNOINTR:
134 /* Decode Syscall # */
135 regs->regs[REG_RET] = regs->syscall_nr;
136 regs->pc -= 4;
137 break;
139 case -ERESTART_RESTARTBLOCK:
140 regs->regs[REG_RET] = __NR_restart_syscall;
141 regs->pc -= 4;
142 break;
146 /* No signal to deliver -- put the saved sigmask back */
147 if (current_thread_info()->status & TS_RESTORE_SIGMASK) {
148 current_thread_info()->status &= ~TS_RESTORE_SIGMASK;
149 sigprocmask(SIG_SETMASK, &current->saved_sigmask, NULL);
152 return 0;
156 * Atomically swap in the new signal mask, and wait for a signal.
158 asmlinkage int
159 sys_sigsuspend(old_sigset_t mask,
160 unsigned long r3, unsigned long r4, unsigned long r5,
161 unsigned long r6, unsigned long r7,
162 struct pt_regs * regs)
164 sigset_t saveset;
166 mask &= _BLOCKABLE;
167 spin_lock_irq(&current->sighand->siglock);
168 saveset = current->blocked;
169 siginitset(&current->blocked, mask);
170 recalc_sigpending();
171 spin_unlock_irq(&current->sighand->siglock);
173 REF_REG_RET = -EINTR;
174 while (1) {
175 current->state = TASK_INTERRUPTIBLE;
176 schedule();
177 set_restore_sigmask();
178 regs->pc += 4; /* because sys_sigreturn decrements the pc */
179 if (do_signal(regs, &saveset)) {
180 /* pc now points at signal handler. Need to decrement
181 it because entry.S will increment it. */
182 regs->pc -= 4;
183 return -EINTR;
188 asmlinkage int
189 sys_rt_sigsuspend(sigset_t *unewset, size_t sigsetsize,
190 unsigned long r4, unsigned long r5, unsigned long r6,
191 unsigned long r7,
192 struct pt_regs * regs)
194 sigset_t saveset, newset;
196 /* XXX: Don't preclude handling different sized sigset_t's. */
197 if (sigsetsize != sizeof(sigset_t))
198 return -EINVAL;
200 if (copy_from_user(&newset, unewset, sizeof(newset)))
201 return -EFAULT;
202 sigdelsetmask(&newset, ~_BLOCKABLE);
203 spin_lock_irq(&current->sighand->siglock);
204 saveset = current->blocked;
205 current->blocked = newset;
206 recalc_sigpending();
207 spin_unlock_irq(&current->sighand->siglock);
209 REF_REG_RET = -EINTR;
210 while (1) {
211 current->state = TASK_INTERRUPTIBLE;
212 schedule();
213 regs->pc += 4; /* because sys_sigreturn decrements the pc */
214 if (do_signal(regs, &saveset)) {
215 /* pc now points at signal handler. Need to decrement
216 it because entry.S will increment it. */
217 regs->pc -= 4;
218 return -EINTR;
223 asmlinkage int
224 sys_sigaction(int sig, const struct old_sigaction __user *act,
225 struct old_sigaction __user *oact)
227 struct k_sigaction new_ka, old_ka;
228 int ret;
230 if (act) {
231 old_sigset_t mask;
232 if (!access_ok(VERIFY_READ, act, sizeof(*act)) ||
233 __get_user(new_ka.sa.sa_handler, &act->sa_handler) ||
234 __get_user(new_ka.sa.sa_restorer, &act->sa_restorer))
235 return -EFAULT;
236 __get_user(new_ka.sa.sa_flags, &act->sa_flags);
237 __get_user(mask, &act->sa_mask);
238 siginitset(&new_ka.sa.sa_mask, mask);
241 ret = do_sigaction(sig, act ? &new_ka : NULL, oact ? &old_ka : NULL);
243 if (!ret && oact) {
244 if (!access_ok(VERIFY_WRITE, oact, sizeof(*oact)) ||
245 __put_user(old_ka.sa.sa_handler, &oact->sa_handler) ||
246 __put_user(old_ka.sa.sa_restorer, &oact->sa_restorer))
247 return -EFAULT;
248 __put_user(old_ka.sa.sa_flags, &oact->sa_flags);
249 __put_user(old_ka.sa.sa_mask.sig[0], &oact->sa_mask);
252 return ret;
255 asmlinkage int
256 sys_sigaltstack(const stack_t __user *uss, stack_t __user *uoss,
257 unsigned long r4, unsigned long r5, unsigned long r6,
258 unsigned long r7,
259 struct pt_regs * regs)
261 return do_sigaltstack(uss, uoss, REF_REG_SP);
265 * Do a signal return; undo the signal stack.
267 struct sigframe {
268 struct sigcontext sc;
269 unsigned long extramask[_NSIG_WORDS-1];
270 long long retcode[2];
273 struct rt_sigframe {
274 struct siginfo __user *pinfo;
275 void *puc;
276 struct siginfo info;
277 struct ucontext uc;
278 long long retcode[2];
281 #ifdef CONFIG_SH_FPU
282 static inline int
283 restore_sigcontext_fpu(struct pt_regs *regs, struct sigcontext __user *sc)
285 int err = 0;
286 int fpvalid;
288 err |= __get_user (fpvalid, &sc->sc_fpvalid);
289 conditional_used_math(fpvalid);
290 if (! fpvalid)
291 return err;
293 if (current == last_task_used_math) {
294 last_task_used_math = NULL;
295 regs->sr |= SR_FD;
298 err |= __copy_from_user(&current->thread.fpu.hard, &sc->sc_fpregs[0],
299 (sizeof(long long) * 32) + (sizeof(int) * 1));
301 return err;
304 static inline int
305 setup_sigcontext_fpu(struct pt_regs *regs, struct sigcontext __user *sc)
307 int err = 0;
308 int fpvalid;
310 fpvalid = !!used_math();
311 err |= __put_user(fpvalid, &sc->sc_fpvalid);
312 if (! fpvalid)
313 return err;
315 if (current == last_task_used_math) {
316 enable_fpu();
317 save_fpu(current);
318 disable_fpu();
319 last_task_used_math = NULL;
320 regs->sr |= SR_FD;
323 err |= __copy_to_user(&sc->sc_fpregs[0], &current->thread.fpu.hard,
324 (sizeof(long long) * 32) + (sizeof(int) * 1));
325 clear_used_math();
327 return err;
329 #else
330 static inline int
331 restore_sigcontext_fpu(struct pt_regs *regs, struct sigcontext __user *sc)
333 return 0;
335 static inline int
336 setup_sigcontext_fpu(struct pt_regs *regs, struct sigcontext __user *sc)
338 return 0;
340 #endif
342 static int
343 restore_sigcontext(struct pt_regs *regs, struct sigcontext __user *sc, long long *r2_p)
345 unsigned int err = 0;
346 unsigned long long current_sr, new_sr;
347 #define SR_MASK 0xffff8cfd
349 #define COPY(x) err |= __get_user(regs->x, &sc->sc_##x)
351 COPY(regs[0]); COPY(regs[1]); COPY(regs[2]); COPY(regs[3]);
352 COPY(regs[4]); COPY(regs[5]); COPY(regs[6]); COPY(regs[7]);
353 COPY(regs[8]); COPY(regs[9]); COPY(regs[10]); COPY(regs[11]);
354 COPY(regs[12]); COPY(regs[13]); COPY(regs[14]); COPY(regs[15]);
355 COPY(regs[16]); COPY(regs[17]); COPY(regs[18]); COPY(regs[19]);
356 COPY(regs[20]); COPY(regs[21]); COPY(regs[22]); COPY(regs[23]);
357 COPY(regs[24]); COPY(regs[25]); COPY(regs[26]); COPY(regs[27]);
358 COPY(regs[28]); COPY(regs[29]); COPY(regs[30]); COPY(regs[31]);
359 COPY(regs[32]); COPY(regs[33]); COPY(regs[34]); COPY(regs[35]);
360 COPY(regs[36]); COPY(regs[37]); COPY(regs[38]); COPY(regs[39]);
361 COPY(regs[40]); COPY(regs[41]); COPY(regs[42]); COPY(regs[43]);
362 COPY(regs[44]); COPY(regs[45]); COPY(regs[46]); COPY(regs[47]);
363 COPY(regs[48]); COPY(regs[49]); COPY(regs[50]); COPY(regs[51]);
364 COPY(regs[52]); COPY(regs[53]); COPY(regs[54]); COPY(regs[55]);
365 COPY(regs[56]); COPY(regs[57]); COPY(regs[58]); COPY(regs[59]);
366 COPY(regs[60]); COPY(regs[61]); COPY(regs[62]);
367 COPY(tregs[0]); COPY(tregs[1]); COPY(tregs[2]); COPY(tregs[3]);
368 COPY(tregs[4]); COPY(tregs[5]); COPY(tregs[6]); COPY(tregs[7]);
370 /* Prevent the signal handler manipulating SR in a way that can
371 crash the kernel. i.e. only allow S, Q, M, PR, SZ, FR to be
372 modified */
373 current_sr = regs->sr;
374 err |= __get_user(new_sr, &sc->sc_sr);
375 regs->sr &= SR_MASK;
376 regs->sr |= (new_sr & ~SR_MASK);
378 COPY(pc);
380 #undef COPY
382 /* Must do this last in case it sets regs->sr.fd (i.e. after rest of sr
383 * has been restored above.) */
384 err |= restore_sigcontext_fpu(regs, sc);
386 regs->syscall_nr = -1; /* disable syscall checks */
387 err |= __get_user(*r2_p, &sc->sc_regs[REG_RET]);
388 return err;
391 asmlinkage int sys_sigreturn(unsigned long r2, unsigned long r3,
392 unsigned long r4, unsigned long r5,
393 unsigned long r6, unsigned long r7,
394 struct pt_regs * regs)
396 struct sigframe __user *frame = (struct sigframe __user *) (long) REF_REG_SP;
397 sigset_t set;
398 long long ret;
400 /* Always make any pending restarted system calls return -EINTR */
401 current_thread_info()->restart_block.fn = do_no_restart_syscall;
403 if (!access_ok(VERIFY_READ, frame, sizeof(*frame)))
404 goto badframe;
406 if (__get_user(set.sig[0], &frame->sc.oldmask)
407 || (_NSIG_WORDS > 1
408 && __copy_from_user(&set.sig[1], &frame->extramask,
409 sizeof(frame->extramask))))
410 goto badframe;
412 sigdelsetmask(&set, ~_BLOCKABLE);
414 spin_lock_irq(&current->sighand->siglock);
415 current->blocked = set;
416 recalc_sigpending();
417 spin_unlock_irq(&current->sighand->siglock);
419 if (restore_sigcontext(regs, &frame->sc, &ret))
420 goto badframe;
421 regs->pc -= 4;
423 return (int) ret;
425 badframe:
426 force_sig(SIGSEGV, current);
427 return 0;
430 asmlinkage int sys_rt_sigreturn(unsigned long r2, unsigned long r3,
431 unsigned long r4, unsigned long r5,
432 unsigned long r6, unsigned long r7,
433 struct pt_regs * regs)
435 struct rt_sigframe __user *frame = (struct rt_sigframe __user *) (long) REF_REG_SP;
436 sigset_t set;
437 stack_t __user st;
438 long long ret;
440 /* Always make any pending restarted system calls return -EINTR */
441 current_thread_info()->restart_block.fn = do_no_restart_syscall;
443 if (!access_ok(VERIFY_READ, frame, sizeof(*frame)))
444 goto badframe;
446 if (__copy_from_user(&set, &frame->uc.uc_sigmask, sizeof(set)))
447 goto badframe;
449 sigdelsetmask(&set, ~_BLOCKABLE);
450 spin_lock_irq(&current->sighand->siglock);
451 current->blocked = set;
452 recalc_sigpending();
453 spin_unlock_irq(&current->sighand->siglock);
455 if (restore_sigcontext(regs, &frame->uc.uc_mcontext, &ret))
456 goto badframe;
457 regs->pc -= 4;
459 if (__copy_from_user(&st, &frame->uc.uc_stack, sizeof(st)))
460 goto badframe;
461 /* It is more difficult to avoid calling this function than to
462 call it and ignore errors. */
463 do_sigaltstack(&st, NULL, REF_REG_SP);
465 return (int) ret;
467 badframe:
468 force_sig(SIGSEGV, current);
469 return 0;
473 * Set up a signal frame.
475 static int
476 setup_sigcontext(struct sigcontext __user *sc, struct pt_regs *regs,
477 unsigned long mask)
479 int err = 0;
481 /* Do this first, otherwise is this sets sr->fd, that value isn't preserved. */
482 err |= setup_sigcontext_fpu(regs, sc);
484 #define COPY(x) err |= __put_user(regs->x, &sc->sc_##x)
486 COPY(regs[0]); COPY(regs[1]); COPY(regs[2]); COPY(regs[3]);
487 COPY(regs[4]); COPY(regs[5]); COPY(regs[6]); COPY(regs[7]);
488 COPY(regs[8]); COPY(regs[9]); COPY(regs[10]); COPY(regs[11]);
489 COPY(regs[12]); COPY(regs[13]); COPY(regs[14]); COPY(regs[15]);
490 COPY(regs[16]); COPY(regs[17]); COPY(regs[18]); COPY(regs[19]);
491 COPY(regs[20]); COPY(regs[21]); COPY(regs[22]); COPY(regs[23]);
492 COPY(regs[24]); COPY(regs[25]); COPY(regs[26]); COPY(regs[27]);
493 COPY(regs[28]); COPY(regs[29]); COPY(regs[30]); COPY(regs[31]);
494 COPY(regs[32]); COPY(regs[33]); COPY(regs[34]); COPY(regs[35]);
495 COPY(regs[36]); COPY(regs[37]); COPY(regs[38]); COPY(regs[39]);
496 COPY(regs[40]); COPY(regs[41]); COPY(regs[42]); COPY(regs[43]);
497 COPY(regs[44]); COPY(regs[45]); COPY(regs[46]); COPY(regs[47]);
498 COPY(regs[48]); COPY(regs[49]); COPY(regs[50]); COPY(regs[51]);
499 COPY(regs[52]); COPY(regs[53]); COPY(regs[54]); COPY(regs[55]);
500 COPY(regs[56]); COPY(regs[57]); COPY(regs[58]); COPY(regs[59]);
501 COPY(regs[60]); COPY(regs[61]); COPY(regs[62]);
502 COPY(tregs[0]); COPY(tregs[1]); COPY(tregs[2]); COPY(tregs[3]);
503 COPY(tregs[4]); COPY(tregs[5]); COPY(tregs[6]); COPY(tregs[7]);
504 COPY(sr); COPY(pc);
506 #undef COPY
508 err |= __put_user(mask, &sc->oldmask);
510 return err;
514 * Determine which stack to use..
516 static inline void __user *
517 get_sigframe(struct k_sigaction *ka, unsigned long sp, size_t frame_size)
519 if ((ka->sa.sa_flags & SA_ONSTACK) != 0 && ! sas_ss_flags(sp))
520 sp = current->sas_ss_sp + current->sas_ss_size;
522 return (void __user *)((sp - frame_size) & -8ul);
525 void sa_default_restorer(void); /* See comments below */
526 void sa_default_rt_restorer(void); /* See comments below */
528 static int setup_frame(int sig, struct k_sigaction *ka,
529 sigset_t *set, struct pt_regs *regs)
531 struct sigframe __user *frame;
532 int err = 0;
533 int signal;
535 frame = get_sigframe(ka, regs->regs[REG_SP], sizeof(*frame));
537 if (!access_ok(VERIFY_WRITE, frame, sizeof(*frame)))
538 goto give_sigsegv;
540 signal = current_thread_info()->exec_domain
541 && current_thread_info()->exec_domain->signal_invmap
542 && sig < 32
543 ? current_thread_info()->exec_domain->signal_invmap[sig]
544 : sig;
546 err |= setup_sigcontext(&frame->sc, regs, set->sig[0]);
548 /* Give up earlier as i386, in case */
549 if (err)
550 goto give_sigsegv;
552 if (_NSIG_WORDS > 1) {
553 err |= __copy_to_user(frame->extramask, &set->sig[1],
554 sizeof(frame->extramask)); }
556 /* Give up earlier as i386, in case */
557 if (err)
558 goto give_sigsegv;
560 /* Set up to return from userspace. If provided, use a stub
561 already in userspace. */
562 if (ka->sa.sa_flags & SA_RESTORER) {
564 * On SH5 all edited pointers are subject to NEFF
566 DEREF_REG_PR = neff_sign_extend((unsigned long)
567 ka->sa.sa_restorer | 0x1);
568 } else {
570 * Different approach on SH5.
571 * . Endianness independent asm code gets placed in entry.S .
572 * This is limited to four ASM instructions corresponding
573 * to two long longs in size.
574 * . err checking is done on the else branch only
575 * . flush_icache_range() is called upon __put_user() only
576 * . all edited pointers are subject to NEFF
577 * . being code, linker turns ShMedia bit on, always
578 * dereference index -1.
580 DEREF_REG_PR = neff_sign_extend((unsigned long)
581 frame->retcode | 0x01);
583 if (__copy_to_user(frame->retcode,
584 (void *)((unsigned long)sa_default_restorer & (~1)), 16) != 0)
585 goto give_sigsegv;
587 /* Cohere the trampoline with the I-cache. */
588 flush_cache_sigtramp(DEREF_REG_PR-1);
592 * Set up registers for signal handler.
593 * All edited pointers are subject to NEFF.
595 regs->regs[REG_SP] = neff_sign_extend((unsigned long)frame);
596 regs->regs[REG_ARG1] = signal; /* Arg for signal handler */
598 /* FIXME:
599 The glibc profiling support for SH-5 needs to be passed a sigcontext
600 so it can retrieve the PC. At some point during 2003 the glibc
601 support was changed to receive the sigcontext through the 2nd
602 argument, but there are still versions of libc.so in use that use
603 the 3rd argument. Until libc.so is stabilised, pass the sigcontext
604 through both 2nd and 3rd arguments.
607 regs->regs[REG_ARG2] = (unsigned long long)(unsigned long)(signed long)&frame->sc;
608 regs->regs[REG_ARG3] = (unsigned long long)(unsigned long)(signed long)&frame->sc;
610 regs->pc = neff_sign_extend((unsigned long)ka->sa.sa_handler);
612 set_fs(USER_DS);
614 /* Broken %016Lx */
615 pr_debug("SIG deliver (#%d,%s:%d): sp=%p pc=%08Lx%08Lx link=%08Lx%08Lx\n",
616 signal, current->comm, current->pid, frame,
617 regs->pc >> 32, regs->pc & 0xffffffff,
618 DEREF_REG_PR >> 32, DEREF_REG_PR & 0xffffffff);
620 return 0;
622 give_sigsegv:
623 force_sigsegv(sig, current);
624 return -EFAULT;
627 static int setup_rt_frame(int sig, struct k_sigaction *ka, siginfo_t *info,
628 sigset_t *set, struct pt_regs *regs)
630 struct rt_sigframe __user *frame;
631 int err = 0;
632 int signal;
634 frame = get_sigframe(ka, regs->regs[REG_SP], sizeof(*frame));
636 if (!access_ok(VERIFY_WRITE, frame, sizeof(*frame)))
637 goto give_sigsegv;
639 signal = current_thread_info()->exec_domain
640 && current_thread_info()->exec_domain->signal_invmap
641 && sig < 32
642 ? current_thread_info()->exec_domain->signal_invmap[sig]
643 : sig;
645 err |= __put_user(&frame->info, &frame->pinfo);
646 err |= __put_user(&frame->uc, &frame->puc);
647 err |= copy_siginfo_to_user(&frame->info, info);
649 /* Give up earlier as i386, in case */
650 if (err)
651 goto give_sigsegv;
653 /* Create the ucontext. */
654 err |= __put_user(0, &frame->uc.uc_flags);
655 err |= __put_user(0, &frame->uc.uc_link);
656 err |= __put_user((void *)current->sas_ss_sp,
657 &frame->uc.uc_stack.ss_sp);
658 err |= __put_user(sas_ss_flags(regs->regs[REG_SP]),
659 &frame->uc.uc_stack.ss_flags);
660 err |= __put_user(current->sas_ss_size, &frame->uc.uc_stack.ss_size);
661 err |= setup_sigcontext(&frame->uc.uc_mcontext,
662 regs, set->sig[0]);
663 err |= __copy_to_user(&frame->uc.uc_sigmask, set, sizeof(*set));
665 /* Give up earlier as i386, in case */
666 if (err)
667 goto give_sigsegv;
669 /* Set up to return from userspace. If provided, use a stub
670 already in userspace. */
671 if (ka->sa.sa_flags & SA_RESTORER) {
673 * On SH5 all edited pointers are subject to NEFF
675 DEREF_REG_PR = neff_sign_extend((unsigned long)
676 ka->sa.sa_restorer | 0x1);
677 } else {
679 * Different approach on SH5.
680 * . Endianness independent asm code gets placed in entry.S .
681 * This is limited to four ASM instructions corresponding
682 * to two long longs in size.
683 * . err checking is done on the else branch only
684 * . flush_icache_range() is called upon __put_user() only
685 * . all edited pointers are subject to NEFF
686 * . being code, linker turns ShMedia bit on, always
687 * dereference index -1.
689 DEREF_REG_PR = neff_sign_extend((unsigned long)
690 frame->retcode | 0x01);
692 if (__copy_to_user(frame->retcode,
693 (void *)((unsigned long)sa_default_rt_restorer & (~1)), 16) != 0)
694 goto give_sigsegv;
696 /* Cohere the trampoline with the I-cache. */
697 flush_icache_range(DEREF_REG_PR-1, DEREF_REG_PR-1+15);
701 * Set up registers for signal handler.
702 * All edited pointers are subject to NEFF.
704 regs->regs[REG_SP] = neff_sign_extend((unsigned long)frame);
705 regs->regs[REG_ARG1] = signal; /* Arg for signal handler */
706 regs->regs[REG_ARG2] = (unsigned long long)(unsigned long)(signed long)&frame->info;
707 regs->regs[REG_ARG3] = (unsigned long long)(unsigned long)(signed long)&frame->uc.uc_mcontext;
708 regs->pc = neff_sign_extend((unsigned long)ka->sa.sa_handler);
710 set_fs(USER_DS);
712 pr_debug("SIG deliver (#%d,%s:%d): sp=%p pc=%08Lx%08Lx link=%08Lx%08Lx\n",
713 signal, current->comm, current->pid, frame,
714 regs->pc >> 32, regs->pc & 0xffffffff,
715 DEREF_REG_PR >> 32, DEREF_REG_PR & 0xffffffff);
717 return 0;
719 give_sigsegv:
720 force_sigsegv(sig, current);
721 return -EFAULT;
725 * OK, we're invoking a handler
727 static int
728 handle_signal(unsigned long sig, siginfo_t *info, struct k_sigaction *ka,
729 sigset_t *oldset, struct pt_regs * regs)
731 int ret;
733 /* Set up the stack frame */
734 if (ka->sa.sa_flags & SA_SIGINFO)
735 ret = setup_rt_frame(sig, ka, info, oldset, regs);
736 else
737 ret = setup_frame(sig, ka, oldset, regs);
739 if (ka->sa.sa_flags & SA_ONESHOT)
740 ka->sa.sa_handler = SIG_DFL;
742 if (ret == 0) {
743 spin_lock_irq(&current->sighand->siglock);
744 sigorsets(&current->blocked,&current->blocked,&ka->sa.sa_mask);
745 if (!(ka->sa.sa_flags & SA_NODEFER))
746 sigaddset(&current->blocked,sig);
747 recalc_sigpending();
748 spin_unlock_irq(&current->sighand->siglock);
751 return ret;
754 asmlinkage void do_notify_resume(struct pt_regs *regs, unsigned long thread_info_flags)
756 if (thread_info_flags & _TIF_SIGPENDING)
757 do_signal(regs, 0);
759 if (thread_info_flags & _TIF_NOTIFY_RESUME) {
760 clear_thread_flag(TIF_NOTIFY_RESUME);
761 tracehook_notify_resume(regs);
762 if (current->replacement_session_keyring)
763 key_replace_session_keyring();