Blame sysdeps/mach/hurd/sigaction.c

Packit 6c4009
/* Copyright (C) 1991-2018 Free Software Foundation, Inc.
Packit 6c4009
Packit 6c4009
   This file is part of the GNU C Library.
Packit 6c4009
Packit 6c4009
   The GNU C Library is free software; you can redistribute it and/or
Packit 6c4009
   modify it under the terms of the GNU Lesser General Public
Packit 6c4009
   License as published by the Free Software Foundation; either
Packit 6c4009
   version 2.1 of the License, or (at your option) any later version.
Packit 6c4009
Packit 6c4009
   The GNU C Library is distributed in the hope that it will be useful,
Packit 6c4009
   but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 6c4009
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit 6c4009
   Lesser General Public License for more details.
Packit 6c4009
Packit 6c4009
   You should have received a copy of the GNU Lesser General Public
Packit 6c4009
   License along with the GNU C Library; if not, see
Packit 6c4009
   <http://www.gnu.org/licenses/>.  */
Packit 6c4009
Packit 6c4009
#include <errno.h>
Packit 6c4009
#include <signal.h>
Packit 6c4009
#include <hurd.h>
Packit 6c4009
#include <hurd/signal.h>
Packit 6c4009
Packit 6c4009
/* If ACT is not NULL, change the action for SIG to *ACT.
Packit 6c4009
   If OACT is not NULL, put the old action for SIG in *OACT.  */
Packit 6c4009
int
Packit 6c4009
__sigaction (int sig, const struct sigaction *act, struct sigaction *oact)
Packit 6c4009
{
Packit 6c4009
  struct hurd_sigstate *ss;
Packit 6c4009
  struct sigaction a, old;
Packit 6c4009
  sigset_t pending;
Packit 6c4009
Packit 6c4009
  if (sig <= 0 || sig >= NSIG ||
Packit 6c4009
      (act != NULL && act->sa_handler != SIG_DFL &&
Packit 6c4009
       ((__sigmask (sig) & _SIG_CANT_MASK) ||
Packit 6c4009
	act->sa_handler == SIG_ERR)))
Packit 6c4009
    {
Packit 6c4009
      errno = EINVAL;
Packit 6c4009
      return -1;
Packit 6c4009
    }
Packit 6c4009
Packit 6c4009
  /* Copy so we fault before taking locks.  */
Packit 6c4009
  if (act != NULL)
Packit 6c4009
    a = *act;
Packit 6c4009
Packit 6c4009
  ss = _hurd_self_sigstate ();
Packit 6c4009
Packit 6c4009
  __spin_lock (&ss->critical_section_lock);
Packit 6c4009
  __spin_lock (&ss->lock);
Packit 6c4009
  old = ss->actions[sig];
Packit 6c4009
  if (act != NULL)
Packit 6c4009
    ss->actions[sig] = a;
Packit 6c4009
Packit 6c4009
  if (act != NULL && sig == SIGCHLD &&
Packit 6c4009
      (a.sa_flags & SA_NOCLDSTOP) != (old.sa_flags & SA_NOCLDSTOP))
Packit 6c4009
    {
Packit 6c4009
      __spin_unlock (&ss->lock);
Packit 6c4009
Packit 6c4009
      /* Inform the proc server whether or not it should send us SIGCHLD for
Packit 6c4009
	 stopped children.  We do this in a critical section so that no
Packit 6c4009
	 SIGCHLD can arrive in the middle and be of indeterminate status.  */
Packit 6c4009
      __USEPORT (PROC,
Packit 6c4009
		 __proc_mod_stopchild (port, !(a.sa_flags & SA_NOCLDSTOP)));
Packit 6c4009
Packit 6c4009
      __spin_lock (&ss->lock);
Packit 6c4009
      pending = ss->pending & ~ss->blocked;
Packit 6c4009
    }
Packit 6c4009
  else if (act != NULL && (a.sa_handler == SIG_IGN || a.sa_handler == SIG_DFL))
Packit 6c4009
    /* We are changing to an action that might be to ignore SIG signals.
Packit 6c4009
       If SIG is blocked and pending and the new action is to ignore it, we
Packit 6c4009
       must remove it from the pending set now; if the action is changed
Packit 6c4009
       back and then SIG is unblocked, the signal pending now should not
Packit 6c4009
       arrive.  So wake up the signal thread to check the new state and do
Packit 6c4009
       the right thing.  */
Packit 6c4009
    pending = ss->pending & __sigmask (sig);
Packit 6c4009
  else
Packit 6c4009
    pending = 0;
Packit 6c4009
Packit 6c4009
  __spin_unlock (&ss->lock);
Packit 6c4009
  __spin_unlock (&ss->critical_section_lock);
Packit 6c4009
Packit 6c4009
  if (pending)
Packit 6c4009
    __msg_sig_post (_hurd_msgport, 0, 0, __mach_task_self ());
Packit 6c4009
Packit 6c4009
  if (oact != NULL)
Packit 6c4009
    *oact = old;
Packit 6c4009
Packit 6c4009
  return 0;
Packit 6c4009
}
Packit 6c4009
libc_hidden_def (__sigaction)
Packit 6c4009
weak_alias (__sigaction, sigaction)