summaryrefslogtreecommitdiff
path: root/libpthread/sysdeps/viengoos/pt-sigstate.c
blob: 7894d99a40da1f551a1894abe8777a29b951dfe8 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
/* Set a thread's signal state.  Hurd on L4 version.
   Copyright (C) 2002, 2005, 2008 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., 59 Temple Place - Suite 330,
   Boston, MA 02111-1307, USA.  */

#include <pthread.h>
#include <assert.h>
#include <signal.h>
#include <viengoos/math.h>

#include <pt-internal.h>

error_t
__pthread_sigstate (struct __pthread *thread, int how,
		    const sigset_t *set, sigset_t *oset,
		    int clear_pending)
{
  struct signal_state *ss = &thread->ss;
  pthread_mutex_lock (&ss->lock);

  if (oset)
    *oset = ss->blocked;

  if (set)
    {
      /* Mask out SIGKILL and SIGSTOP.  */
      sigset_t s = *set;
      sigdelset (&s, SIGKILL);
      sigdelset (&s, SIGSTOP);

      switch (how)
	{
	case SIG_BLOCK:
	  ss->blocked |= s;
	  break;
	case SIG_UNBLOCK:
	  ss->blocked &= ~s;
	  break;
	case SIG_SETMASK:
	  ss->blocked = s;
	  break;
	default:
	  errno = EINVAL;
	  pthread_mutex_unlock (&ss->lock);
	  return -1;
	}
    }

  if (clear_pending)
    sigemptyset (&ss->pending);

  /* A "signal shall remain pending until it is unblocked" (2.4.1).

     "If there are any pending unblocked signals after the call to
     sigprocmask(), at least one of those signals shall be delivered
     before the call to sigprocmask() returns."
     (pthread_sigmask).  */
  sigset_t extant = ~ss->blocked & ss->pending;
  if (! extant)
    extant = ~ss->blocked & process_pending;

  pthread_mutex_unlock (&ss->lock);

  if (extant)
    raise (vg_lsb64 (extant));

  return 0;
}