entered into RCS
authorroland <roland>
Sat, 9 Jul 1994 07:18:33 +0000 (07:18 +0000)
committerroland <roland>
Sat, 9 Jul 1994 07:18:33 +0000 (07:18 +0000)
sysdeps/mach/hurd/mips/trampoline.c [new file with mode: 0644]

diff --git a/sysdeps/mach/hurd/mips/trampoline.c b/sysdeps/mach/hurd/mips/trampoline.c
new file mode 100644 (file)
index 0000000..a6c9cec
--- /dev/null
@@ -0,0 +1,96 @@
+/* Set thread_state for sighandler, and sigcontext to recover.  MIPS version.
+Copyright (C) 1994 Free Software Foundation, Inc.
+This file is part of the GNU C Library.
+
+The GNU C Library is free software; you can redistribute it and/or
+modify it under the terms of the GNU Library General Public License as
+published by the Free Software Foundation; either version 2 of the
+License, or (at your option) any later version.
+
+The GNU C Library is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
+Library General Public License for more details.
+
+You should have received a copy of the GNU Library General Public
+License along with the GNU C Library; see the file COPYING.LIB.  If
+not, write to the Free Software Foundation, Inc., 675 Mass Ave,
+Cambridge, MA 02139, USA.  */
+
+#include <hurd/signal.h>
+#include <mach/thread_status.h>
+
+static void
+trampoline (void (*handler) (int signo, int sigcode, struct sigcontext *scp),
+           int signo, int sigcode, struct sigcontext *scp)
+{
+  (*handler) (signo, sigcode, scp);
+  (void) __sigreturn (scp);    /* Does not return.  */
+  while (1)
+    LOSE;                      /* Firewall.  */
+}
+
+struct sigcontext *
+_hurd_setup_sighandler (int flags,
+                       __sighandler_t handler,
+                       struct sigaltstack *sigaltstack,
+                       int signo, int sigcode,
+                       void *state)
+{
+  struct mips_thread_state *ts;
+  void *sigsp;
+  struct sigcontext *scp;
+  struct 
+    {
+      void *retaddr;           /* Never used.  */
+      __sighandler_t handler;
+      int signo;
+      int sigcode;
+      struct sigcontext *scp;  /* Points to ctx, below.  */
+      struct sigcontext ctx;
+    } *stackframe;
+
+  ts = state;
+
+  if ((flags & SA_ONSTACK) &&
+      !(sigaltstack->ss_flags & (SA_DISABLE|SA_ONSTACK)))
+    {
+      sigsp = sigaltstack->ss_sp + sigaltstack->ss_size;
+      sigaltstack->ss_flags |= SA_ONSTACK;
+    }
+  else
+    sigsp = (char *) ts->r29;
+
+  /* Push the arguments to call `trampoline' on the stack.  */
+  sigsp -= sizeof (*stackframe);
+  stackframe = sigsp;
+  stackframe->handler = handler;
+  stackframe->signo = signo;
+  stackframe->sigcode = sigcode;
+  stackframe->scp = scp = &stackframe->ctx;
+
+  /* Set up the sigcontext from the current state of the thread.  */
+
+  scp->sc_onstack = sigaltstack->ss_flags & SA_ONSTACK ? 1 : 0;
+
+  scp->sc_gpr[16] = ts->r16;
+  scp->sc_gpr[17] = ts->r17;
+  scp->sc_gpr[18] = ts->r18;
+  scp->sc_gpr[19] = ts->r19;
+  scp->sc_gpr[20] = ts->r20;
+  scp->sc_gpr[21] = ts->r21;
+  scp->sc_gpr[22] = ts->r22;
+  scp->sc_gpr[23] = ts->r23;
+  scp->sc_gpr[28] = ts->r28;
+  scp->sc_gpr[31] = ts->r31;
+  
+  scp->sc_pc = ts->pc;
+  scp->sc_sp = ts->r29;
+  scp->sc_fp = ts->r30;
+
+  /* Modify the thread state to call `trampoline' on the new stack.  */
+  ts->r29 = (int) sigsp;
+  ts->pc = (int) &trampoline;
+
+  return scp;
+}