diff options
Diffstat (limited to 'libc/sysdeps/linux/mips/sigaction.c')
| -rw-r--r-- | libc/sysdeps/linux/mips/sigaction.c | 177 | 
1 files changed, 76 insertions, 101 deletions
| diff --git a/libc/sysdeps/linux/mips/sigaction.c b/libc/sysdeps/linux/mips/sigaction.c index bd3382f63..0135481ae 100644 --- a/libc/sysdeps/linux/mips/sigaction.c +++ b/libc/sysdeps/linux/mips/sigaction.c @@ -1,154 +1,129 @@ -/* Copyright (C) 1997,1998,1999,2000,2002,2003, 2004 -   Free Software Foundation, Inc. +/* Copyright (C) 1997, 1998, 1999, 2000 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. +   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 -   Lesser General Public License for more details. +   Library 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, write to the Free -   Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA -   02111-1307 USA.  */ +   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., 59 Temple Place - Suite 330, +   Boston, MA 02111-1307, USA.   + +   Totally hacked up for uClibc by Erik Andersen <andersen@codepoet.org> +   */  #include <errno.h>  #include <signal.h>  #include <string.h> -#include <sysdep.h>  #include <sys/syscall.h> - -#include "kernel-features.h" - -/* The difference here is that the sigaction structure used in the -   kernel is not the same as we use in the libc.  Therefore we must -   translate it here.  */  #include <bits/kernel_sigaction.h> -#ifndef LIBC_SIGACTION +#define SA_RESTORER	0x04000000 +  extern __typeof(sigaction) __libc_sigaction; -#endif -#if __ASSUME_REALTIME_SIGNALS == 0 -/* The variable is shared between all wrappers around signal handling -   functions which have RT equivalents.	 This is the definition.  */ -int __libc_missing_rt_sigs; +#ifdef __NR_rt_sigaction -#endif +/* Experimentally off - libc_hidden_proto(memcpy) */  #if _MIPS_SIM != _ABIO32  # ifdef __NR_rt_sigreturn -static void restore_rt (void) asm ("__restore_rt"); +static void restore_rt (void) __asm__ ("__restore_rt");  # endif  # ifdef __NR_sigreturn -static void restore (void) asm ("__restore"); +static void restore (void) __asm__ ("__restore");  # endif  #endif  /* If ACT is not NULL, change the action for SIG to *ACT.     If OACT is not NULL, put the old action for SIG in *OACT.  */ -int -__libc_sigaction (int sig, const struct sigaction *act, struct sigaction *oact) +int __libc_sigaction (int sig, const struct sigaction *act, struct sigaction *oact)  { -#if __ASSUME_REALTIME_SIGNALS == 0 -  struct old_kernel_sigaction k_sigact, k_osigact; -#endif -  int result; - -#if defined __NR_rt_sigaction || __ASSUME_REALTIME_SIGNALS > 0 -  /* First try the RT signals.	*/ -# if __ASSUME_REALTIME_SIGNALS == 0 -  if (!__libc_missing_rt_sigs) -# endif -    { -      struct kernel_sigaction kact, koact; -      /* Save the current error value for later.  We need not do this -	 if we are guaranteed to have realtime signals.	 */ -# if __ASSUME_REALTIME_SIGNALS == 0 -      int saved_errno = errno; -# endif +    int result; +    struct kernel_sigaction kact, koact; -      if (act) -	{ -	  kact.k_sa_handler = act->sa_handler; -	  memcpy (&kact.sa_mask, &act->sa_mask, sizeof (kernel_sigset_t)); -	  kact.sa_flags = act->sa_flags; +    if (act) { +	kact.k_sa_handler = act->sa_handler; +	memcpy (&kact.sa_mask, &act->sa_mask, sizeof (kact.sa_mask)); +	kact.sa_flags = act->sa_flags;  # ifdef HAVE_SA_RESTORER  #  if _MIPS_SIM == _ABIO32 -	  kact.sa_restorer = act->sa_restorer; +	kact.sa_restorer = act->sa_restorer;  #  else -	  kact.sa_restorer = &restore_rt; +	kact.sa_restorer = &restore_rt;  #  endif  # endif -	} +    } -      /* XXX The size argument hopefully will have to be changed to the -	 real size of the user-level sigset_t.	*/ -      result = INLINE_SYSCALL (rt_sigaction, 4, sig, -			       act ? __ptrvalue (&kact) : NULL, -			       oact ? __ptrvalue (&koact) : NULL, -			       sizeof (kernel_sigset_t)); +    /* XXX The size argument hopefully will have to be changed to the +       real size of the user-level sigset_t.  */ +    result = __syscall_rt_sigaction(sig, act ? __ptrvalue (&kact) : NULL, +	    oact ? __ptrvalue (&koact) : NULL, _NSIG / 8); -# if __ASSUME_REALTIME_SIGNALS == 0 -      if (result >= 0 || errno != ENOSYS) -# endif -	{ -	  if (oact && result >= 0) -	    { -	      oact->sa_handler = koact.k_sa_handler; -	      memcpy (&oact->sa_mask, &koact.sa_mask, -				sizeof (kernel_sigset_t)); -	      oact->sa_flags = koact.sa_flags; +    if (oact && result >= 0) { +	oact->sa_handler = koact.k_sa_handler; +	memcpy (&oact->sa_mask, &koact.sa_mask, sizeof (oact->sa_mask)); +	oact->sa_flags = koact.sa_flags;  # ifdef HAVE_SA_RESTORER -	      oact->sa_restorer = koact.sa_restorer; -# endif -	    } -	  return result; -	} - -# if __ASSUME_REALTIME_SIGNALS == 0 -      __set_errno (saved_errno); -      __libc_missing_rt_sigs = 1; +	oact->sa_restorer = koact.sa_restorer;  # endif      } -#endif +    return result; +} -#if __ASSUME_REALTIME_SIGNALS == 0 -  if (act) -    { -      k_sigact.k_sa_handler = act->sa_handler; -      k_sigact.sa_mask = act->sa_mask.__val[0]; -      k_sigact.sa_flags = act->sa_flags; -# ifdef HAVE_SA_RESTORER -      k_sigact.sa_restorer = act->sa_restorer; -# endif -    } -  result = INLINE_SYSCALL (sigaction, 3, sig, -			   act ? __ptrvalue (&k_sigact) : NULL, -			   oact ? __ptrvalue (&k_osigact) : NULL); -  if (oact && result >= 0) -    { -      oact->sa_handler = k_osigact.k_sa_handler; -      oact->sa_mask.__val[0] = k_osigact.sa_mask; -      oact->sa_flags = k_osigact.sa_flags; + +#else +extern void restore (void) __asm__ ("__restore") attribute_hidden; + +/* If ACT is not NULL, change the action for SIG to *ACT. +   If OACT is not NULL, put the old action for SIG in *OACT.  */ +int __libc_sigaction (int sig, const struct sigaction *act, struct sigaction *oact) +{ +    int result; +    struct old_kernel_sigaction kact, koact; + +    if (act) { +	kact.k_sa_handler = act->sa_handler; +	kact.sa_mask = act->sa_mask.__val[0]; +	kact.sa_flags = act->sa_flags;  # ifdef HAVE_SA_RESTORER  #  if _MIPS_SIM == _ABIO32 -      oact->sa_restorer = k_osigact.sa_restorer; +	kact.sa_restorer = act->sa_restorer;  #  else -      oact->sa_restorer = &restore; +	kact.sa_restorer = &restore_rt;  #  endif  # endif      } -  return result; -#endif + +    result = __syscall_sigaction(sig, act ? __ptrvalue (&kact) : NULL, +	    oact ? __ptrvalue (&koact) : NULL); + +    if (result < 0) { +	__set_errno(-result); +	return -1; +    } + +    if (oact) { +	oact->sa_handler = koact.k_sa_handler; +	oact->sa_mask.__val[0] = koact.sa_mask; +	oact->sa_flags = koact.sa_flags; +# ifdef HAVE_SA_RESTORER +	oact->sa_restorer = koact.sa_restorer; +# endif +    } +    return result;  } +#endif +  #ifndef LIBC_SIGACTION  libc_hidden_proto(sigaction)  weak_alias(__libc_sigaction,sigaction) | 
