/* Handle faults in the signal thread. Copyright (C) 1994-2014 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 Lesser General Public License as published by the Free Software Foundation; either version 2.1 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 Lesser General Public License for more details. You should have received a copy of the GNU Lesser General Public License along with the GNU C Library; if not, see <http://www.gnu.org/licenses/>. */ #include <hurd.h> #include <hurd/signal.h> #include "hurdfault.h" #include <errno.h> #include <string.h> #include <setjmp.h> #include <stdio.h> #include <thread_state.h> #include "faultexc_server.h" /* mig-generated header for our exc server. */ #include <assert.h> jmp_buf _hurdsig_fault_env; struct hurd_signal_preemptor _hurdsig_fault_preemptor = {0}; /* XXX temporary to deal with spelling fix */ weak_alias (_hurdsig_fault_preemptor, _hurdsig_fault_preempter) static mach_port_t forward_sigexc; kern_return_t _hurdsig_fault_catch_exception_raise (mach_port_t port, thread_t thread, task_t task, #ifdef EXC_MASK_ALL /* New interface flavor. */ exception_type_t exception, exception_data_t code, mach_msg_type_number_t codeCnt #else /* Vanilla Mach 3.0 interface. */ integer_t exception, integer_t code, integer_t subcode #endif ) { int signo; struct hurd_signal_detail d; if (port != forward_sigexc || thread != _hurd_msgport_thread || task != __mach_task_self ()) return EPERM; /* Strange bogosity. */ d.exc = exception; #ifdef EXC_MASK_ALL assert (codeCnt >= 2); d.exc_code = code[0]; d.exc_subcode = code[1]; #else d.exc_code = code; d.exc_subcode = subcode; #endif /* Call the machine-dependent function to translate the Mach exception codes into a signal number and subcode. */ _hurd_exception2signal (&d, &signo); return HURD_PREEMPT_SIGNAL_P (&_hurdsig_fault_preemptor, signo, d.code) ? 0 : EGREGIOUS; } #ifdef EXC_MASK_ALL /* XXX New interface flavor has additional RPCs that we could be using instead. These RPCs roll a thread_get_state/thread_set_state into the message, so the signal thread ought to use these to save some calls. */ kern_return_t _hurdsig_fault_catch_exception_raise_state (mach_port_t port, exception_type_t exception, exception_data_t code, mach_msg_type_number_t codeCnt, int *flavor, thread_state_t old_state, mach_msg_type_number_t old_stateCnt, thread_state_t new_state, mach_msg_type_number_t *new_stateCnt) { abort (); return KERN_FAILURE; } kern_return_t _hurdsig_fault_catch_exception_raise_state_identity (mach_port_t exception_port, thread_t thread, task_t task, exception_type_t exception, exception_data_t code, mach_msg_type_number_t codeCnt, int *flavor, thread_state_t old_state, mach_msg_type_number_t old_stateCnt, thread_state_t new_state, mach_msg_type_number_t *new_stateCnt) { abort (); return KERN_FAILURE; } #endif #ifdef NDR_CHAR_ASCII /* OSF Mach flavors have different names. */ # define mig_reply_header_t mig_reply_error_t #endif static void faulted (void) { struct { mach_msg_header_t head; char buf[64]; } request; mig_reply_header_t reply; extern int _hurdsig_fault_exc_server (mach_msg_header_t *, mach_msg_header_t *); /* Wait for the exception_raise message forwarded by the proc server. */ if (__mach_msg (&request.head, MACH_RCV_MSG, 0, sizeof request, forward_sigexc, MACH_MSG_TIMEOUT_NONE, MACH_PORT_NULL) != MACH_MSG_SUCCESS) __libc_fatal ("msg receive failed on signal thread exc\n"); /* Run the exc demuxer which should call the server function above. That function returns 0 if the exception was expected. */ _hurdsig_fault_exc_server (&request.head, &reply.Head); if (reply.Head.msgh_remote_port != MACH_PORT_NULL) __mach_msg (&reply.Head, MACH_SEND_MSG, reply.Head.msgh_size, 0, MACH_PORT_NULL, MACH_MSG_TIMEOUT_NONE, MACH_PORT_NULL); if (reply.RetCode == MIG_BAD_ID) __mach_msg_destroy (&request.head); if (reply.RetCode) __libc_fatal ("BUG: unexpected fault in signal thread\n"); _hurdsig_fault_preemptor.signals = 0; longjmp (_hurdsig_fault_env, 1); } static char faultstack[1024]; /* Send exceptions for the signal thread to the proc server. It will forward the message on to our message port, and then restore the thread's state to code which does `longjmp (_hurd_sigthread_fault_env, 1)'. */ void _hurdsig_fault_init (void) { error_t err; struct machine_thread_state state; mach_port_t sigexc; /* Allocate a port to receive signal thread exceptions. We will move this receive right to the proc server. */ err = __mach_port_allocate (__mach_task_self (), MACH_PORT_RIGHT_RECEIVE, &sigexc); assert_perror (err); err = __mach_port_allocate (__mach_task_self (), MACH_PORT_RIGHT_RECEIVE, &forward_sigexc); assert_perror (err); /* Allocate a port to receive the exception msgs forwarded from the proc server. */ err = __mach_port_insert_right (__mach_task_self (), sigexc, sigexc, MACH_MSG_TYPE_MAKE_SEND); assert_perror (err); /* Set the queue limit for this port to just one. The proc server will notice if we ever get a second exception while one remains queued and unreceived, and decide we are hopelessly buggy. */ #ifdef MACH_PORT_RECEIVE_STATUS_COUNT { const mach_port_limits_t lim = { mpl_qlimit: 1 }; assert (MACH_PORT_RECEIVE_STATUS_COUNT == sizeof lim / sizeof (natural_t)); err = __mach_port_set_attributes (__mach_task_self (), forward_sigexc, MACH_PORT_RECEIVE_STATUS, (mach_port_info_t) &lim, MACH_PORT_RECEIVE_STATUS_COUNT); } #else err = __mach_port_set_qlimit (__mach_task_self (), forward_sigexc, 1); #endif assert_perror (err); /* This state will be restored when we fault. It runs the function above. */ memset (&state, 0, sizeof state); MACHINE_THREAD_STATE_SET_PC (&state, faulted); MACHINE_THREAD_STATE_SET_SP (&state, faultstack, sizeof faultstack); err = __USEPORT (PROC, __proc_handle_exceptions (port, sigexc, forward_sigexc, MACH_MSG_TYPE_MAKE_SEND, MACHINE_THREAD_STATE_FLAVOR, (natural_t *) &state, MACHINE_THREAD_STATE_COUNT)); assert_perror (err); /* Direct signal thread exceptions to the proc server. */ #ifdef THREAD_EXCEPTION_PORT err = __thread_set_special_port (_hurd_msgport_thread, THREAD_EXCEPTION_PORT, sigexc); #elif defined (EXC_MASK_ALL) __thread_set_exception_ports (_hurd_msgport_thread, EXC_MASK_ALL & ~(EXC_MASK_SYSCALL | EXC_MASK_MACH_SYSCALL | EXC_MASK_RPC_ALERT), sigexc, EXCEPTION_STATE_IDENTITY, MACHINE_THREAD_STATE); #else # error thread_set_exception_ports? #endif __mach_port_deallocate (__mach_task_self (), sigexc); assert_perror (err); }