Return EPERM if TASK is not right.
[kopensolaris-gnu/glibc.git] / hurd / catch-exc.c
1 /* Copyright (C) 1994 Free Software Foundation, Inc.
2 This file is part of the GNU C Library.
3
4 The GNU C Library is free software; you can redistribute it and/or
5 modify it under the terms of the GNU Library General Public License as
6 published by the Free Software Foundation; either version 2 of the
7 License, or (at your option) any later version.
8
9 The GNU C Library is distributed in the hope that it will be useful,
10 but WITHOUT ANY WARRANTY; without even the implied warranty of
11 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
12 Library General Public License for more details.
13
14 You should have received a copy of the GNU Library General Public
15 License along with the GNU C Library; see the file COPYING.LIB.  If
16 not, write to the Free Software Foundation, Inc., 675 Mass Ave,
17 Cambridge, MA 02139, USA.  */
18
19 #include <mach/exc_server.h>
20 #include <hurd/signal.h>
21
22 /* Called by the microkernel when a thread gets an exception.  */
23
24 kern_return_t
25 _S_catch_exception_raise (mach_port_t port,
26                           thread_t thread,
27                           task_t task,
28                           int exception,
29                           int code,
30                           int subcode)
31 {
32   int signo, sigcode, error;
33   struct hurd_sigstate *ss;
34
35   if (task != __mach_task_self ())
36     /* The sender wasn't the kernel.  */
37     return EPERM;
38
39   /* Call the machine-dependent function to translate the Mach exception
40      codes into a signal number and subcode.  */
41   _hurd_exception2signal (exception, code, subcode,
42                           &signo, &sigcode, &error);
43
44   /* Find the sigstate structure for the faulting thread.  */
45   __mutex_lock (&_hurd_siglock);
46   for (ss = _hurd_sigstates; ss != NULL; ss = ss->next)
47     if (ss->thread == thread)
48       break;
49   __mutex_unlock (&_hurd_siglock);
50   if (ss == NULL)
51     ss = _hurd_thread_sigstate (thread); /* Allocate a fresh one.  */
52
53   if (__spin_lock_locked (&ss->lock.held))
54     /* Oops.  The thread faulted with its sigstate lock held.
55        Bad scene.  What to do?  */
56     ;                           /* XXX */
57   else
58     __mutex_lock (&ss->lock);
59
60   /* Post the signal.  */
61   _hurd_internal_post_signal (ss, signo, sigcode, error,
62                               MACH_PORT_NULL, MACH_MSG_TYPE_PORT_SEND);
63
64   return KERN_SUCCESS;
65 }