856cce6d0b24be863a3783cf2a411e4bbd317c7b
[kopensolaris-gnu/glibc.git] / sysdeps / mach / hurd / mips / trampoline.c
1 /* Set thread_state for sighandler, and sigcontext to recover.  MIPS version.
2 Copyright (C) 1994 Free Software Foundation, Inc.
3 This file is part of the GNU C Library.
4
5 The GNU C Library is free software; you can redistribute it and/or
6 modify it under the terms of the GNU Library General Public License as
7 published by the Free Software Foundation; either version 2 of the
8 License, or (at your option) any later version.
9
10 The GNU C Library is distributed in the hope that it will be useful,
11 but WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
13 Library General Public License for more details.
14
15 You should have received a copy of the GNU Library General Public
16 License along with the GNU C Library; see the file COPYING.LIB.  If
17 not, write to the Free Software Foundation, Inc., 675 Mass Ave,
18 Cambridge, MA 02139, USA.  */
19
20 #include <hurd/signal.h>
21 #include <mach/thread_status.h>
22
23 static void
24 trampoline (void (*handler) (int signo, int sigcode, struct sigcontext *scp),
25             int signo, int sigcode, struct sigcontext *scp)
26 {
27   (*handler) (signo, sigcode, scp);
28   (void) __sigreturn (scp);     /* Does not return.  */
29   while (1)
30     LOSE;                       /* Firewall.  */
31 }
32
33 struct sigcontext *
34 _hurd_setup_sighandler (int flags,
35                         __sighandler_t handler,
36                         struct sigaltstack *sigaltstack,
37                         int signo, int sigcode,
38                         void *state)
39 {
40   struct mips_thread_state *ts;
41   void *sigsp;
42   struct sigcontext *scp;
43
44   ts = state;
45
46   if ((flags & SA_ONSTACK) &&
47       !(sigaltstack->ss_flags & (SA_DISABLE|SA_ONSTACK)))
48     {
49       sigsp = sigaltstack->ss_sp + sigaltstack->ss_size;
50       sigaltstack->ss_flags |= SA_ONSTACK;
51     }
52   else
53     sigsp = (char *) ts->r29;
54
55   /* Set up the sigcontext structure on the stack.  This is all the stack
56      needs, since the args are passed in registers (below).  */
57   sigsp -= sizeof (*scp);
58   scp = sigsp;
59
60   /* Set up the sigcontext from the current state of the thread.  */
61
62   scp->sc_onstack = sigaltstack->ss_flags & SA_ONSTACK ? 1 : 0;
63
64   scp->sc_gpr[2] = ts->r2;
65   scp->sc_gpr[16] = ts->r16;
66   scp->sc_gpr[17] = ts->r17;
67   scp->sc_gpr[18] = ts->r18;
68   scp->sc_gpr[19] = ts->r19;
69   scp->sc_gpr[20] = ts->r20;
70   scp->sc_gpr[21] = ts->r21;
71   scp->sc_gpr[22] = ts->r22;
72   scp->sc_gpr[23] = ts->r23;
73   scp->sc_gpr[28] = ts->r28;
74   scp->sc_gpr[31] = ts->r31;
75   
76   scp->sc_pc = ts->pc;
77   scp->sc_sp = ts->r29;
78   scp->sc_fp = ts->r30;
79
80   /* Modify the thread state to call `trampoline' on the new stack.  */
81
82   /* These registers are used for passing the first four arguments to a
83      function (the rest go on the stack).  Fortunately `trampoline' takes
84      just four arguments, so they all fit in registers.  */
85   ts->r4 = (int) handler;
86   ts->r5 = signo;
87   ts->r6 = sigcode;
88   ts->r7 = (int) scp;
89
90   ts->r29 = (int) sigsp;        /* r29 is the stack pointer register.  */
91   ts->pc = (int) &trampoline;
92
93   return scp;
94 }