2 Unix SMB/CIFS implementation.
4 common events code for signal events
6 Copyright (C) Andrew Tridgell 2007
8 ** NOTE! The following LGPL license applies to the tevent
9 ** library. This does NOT imply that all of Samba is released
12 This library is free software; you can redistribute it and/or
13 modify it under the terms of the GNU Lesser General Public
14 License as published by the Free Software Foundation; either
15 version 3 of the License, or (at your option) any later version.
17 This library is distributed in the hope that it will be useful,
18 but WITHOUT ANY WARRANTY; without even the implied warranty of
19 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
20 Lesser General Public License for more details.
22 You should have received a copy of the GNU Lesser General Public
23 License along with this library; if not, see <http://www.gnu.org/licenses/>.
27 #include "system/filesys.h"
28 #include "system/wait.h"
29 #define TEVENT_DEPRECATED 1
31 #include "tevent_internal.h"
32 #include "tevent_util.h"
34 /* maximum number of SA_SIGINFO signals to hold in the queue.
35 NB. This *MUST* be a power of 2, in order for the ring buffer
36 wrap to work correctly. Thanks to Petr Vandrovec <petr@vandrovec.name>
39 #define TEVENT_SA_INFO_QUEUE_COUNT 256
41 size_t tevent_num_signals(void)
43 return TEVENT_NUM_SIGNALS
;
46 size_t tevent_sa_info_queue_count(void)
48 return TEVENT_SA_INFO_QUEUE_COUNT
;
51 struct tevent_sigcounter
{
56 #if defined(HAVE___SYNC_FETCH_AND_ADD)
57 #define TEVENT_SIG_INCREMENT(s) __sync_fetch_and_add(&((s).count), 1)
58 #elif defined(HAVE_ATOMIC_ADD_32)
59 #define TEVENT_SIG_INCREMENT(s) atomic_add_32(&((s).count), 1)
61 #define TEVENT_SIG_INCREMENT(s) (s).count++
63 #define TEVENT_SIG_SEEN(s, n) (s).seen += (n)
64 #define TEVENT_SIG_PENDING(s) ((s).seen != (s).count)
66 struct tevent_common_signal_list
{
67 struct tevent_common_signal_list
*prev
, *next
;
68 struct tevent_signal
*se
;
72 the poor design of signals means that this table must be static global
74 static struct tevent_sig_state
{
75 struct tevent_common_signal_list
*sig_handlers
[TEVENT_NUM_SIGNALS
+1];
76 struct sigaction
*oldact
[TEVENT_NUM_SIGNALS
+1];
77 struct tevent_sigcounter signal_count
[TEVENT_NUM_SIGNALS
+1];
78 struct tevent_sigcounter got_signal
;
80 /* with SA_SIGINFO we get quite a lot of info per signal */
81 siginfo_t
*sig_info
[TEVENT_NUM_SIGNALS
+1];
82 struct tevent_sigcounter sig_blocked
[TEVENT_NUM_SIGNALS
+1];
87 return number of sigcounter events not processed yet
89 static uint32_t tevent_sig_count(struct tevent_sigcounter s
)
91 return s
.count
- s
.seen
;
95 signal handler - redirects to registered signals
97 static void tevent_common_signal_handler(int signum
)
99 struct tevent_common_signal_list
*sl
;
100 struct tevent_context
*ev
= NULL
;
101 int saved_errno
= errno
;
103 TEVENT_SIG_INCREMENT(sig_state
->signal_count
[signum
]);
104 TEVENT_SIG_INCREMENT(sig_state
->got_signal
);
106 /* Write to each unique event context. */
107 for (sl
= sig_state
->sig_handlers
[signum
]; sl
; sl
= sl
->next
) {
108 if (sl
->se
->event_ctx
&& sl
->se
->event_ctx
!= ev
) {
109 ev
= sl
->se
->event_ctx
;
110 tevent_common_wakeup(ev
);
119 signal handler with SA_SIGINFO - redirects to registered signals
121 static void tevent_common_signal_handler_info(int signum
, siginfo_t
*info
,
124 uint32_t count
= tevent_sig_count(sig_state
->signal_count
[signum
]);
125 /* sig_state->signal_count[signum].seen % TEVENT_SA_INFO_QUEUE_COUNT
126 * is the base of the unprocessed signals in the ringbuffer. */
127 uint32_t ofs
= (sig_state
->signal_count
[signum
].seen
+ count
) %
128 TEVENT_SA_INFO_QUEUE_COUNT
;
129 sig_state
->sig_info
[signum
][ofs
] = *info
;
131 tevent_common_signal_handler(signum
);
133 /* handle SA_SIGINFO */
134 if (count
+1 == TEVENT_SA_INFO_QUEUE_COUNT
) {
135 /* we've filled the info array - block this signal until
136 these ones are delivered */
137 #ifdef HAVE_UCONTEXT_T
139 * This is the only way for this to work.
140 * By default signum is blocked inside this
141 * signal handler using a temporary mask,
142 * but what we really need to do now is
143 * block it in the callers mask, so it
144 * stays blocked when the temporary signal
145 * handler mask is replaced when we return
146 * from here. The callers mask can be found
147 * in the ucontext_t passed in as the
148 * void *uctx argument.
150 ucontext_t
*ucp
= (ucontext_t
*)uctx
;
151 sigaddset(&ucp
->uc_sigmask
, signum
);
154 * WARNING !!! WARNING !!!!
156 * This code doesn't work.
157 * By default signum is blocked inside this
158 * signal handler, but calling sigprocmask
159 * modifies the temporary signal mask being
160 * used *inside* this handler, which will be
161 * replaced by the callers signal mask once
162 * we return from here. See Samba
163 * bug #9550 for details.
167 sigaddset(&set
, signum
);
168 sigprocmask(SIG_BLOCK
, &set
, NULL
);
170 TEVENT_SIG_INCREMENT(sig_state
->sig_blocked
[signum
]);
175 static int tevent_common_signal_list_destructor(struct tevent_common_signal_list
*sl
)
177 if (sig_state
->sig_handlers
[sl
->se
->signum
]) {
178 DLIST_REMOVE(sig_state
->sig_handlers
[sl
->se
->signum
], sl
);
184 destroy a signal event
186 static int tevent_signal_destructor(struct tevent_signal
*se
)
189 tevent_common_check_double_free(se
, "tevent_signal double free");
192 se
->destroyed
= true;
194 TALLOC_FREE(se
->additional_data
);
196 if (se
->event_ctx
!= NULL
) {
197 tevent_trace_signal_callback(se
->event_ctx
, se
, TEVENT_EVENT_TRACE_DETACH
);
198 DLIST_REMOVE(se
->event_ctx
->signal_events
, se
);
201 if (sig_state
->sig_handlers
[se
->signum
] == NULL
) {
202 /* restore old handler, if any */
203 if (sig_state
->oldact
[se
->signum
]) {
204 sigaction(se
->signum
, sig_state
->oldact
[se
->signum
], NULL
);
205 TALLOC_FREE(sig_state
->oldact
[se
->signum
]);
208 if (se
->sa_flags
& SA_SIGINFO
) {
209 if (sig_state
->sig_info
[se
->signum
]) {
210 TALLOC_FREE(sig_state
->sig_info
[se
->signum
]);
216 se
->event_ctx
= NULL
;
228 return NULL on failure (memory allocation error)
230 struct tevent_signal
*tevent_common_add_signal(struct tevent_context
*ev
,
234 tevent_signal_handler_t handler
,
236 const char *handler_name
,
237 const char *location
)
239 struct tevent_signal
*se
;
240 struct tevent_common_signal_list
*sl
;
241 sigset_t set
, oldset
;
244 ret
= tevent_common_wakeup_init(ev
);
250 if (signum
>= TEVENT_NUM_SIGNALS
) {
255 /* the sig_state needs to be on a global context as it can last across
256 multiple event contexts */
257 if (sig_state
== NULL
) {
258 sig_state
= talloc_zero(NULL
, struct tevent_sig_state
);
259 if (sig_state
== NULL
) {
264 se
= talloc_zero(mem_ctx
?mem_ctx
:ev
, struct tevent_signal
);
265 if (se
== NULL
) return NULL
;
267 sl
= talloc_zero(se
, struct tevent_common_signal_list
);
274 *se
= (struct tevent_signal
) {
277 .sa_flags
= sa_flags
,
279 .private_data
= private_data
,
280 .handler_name
= handler_name
,
281 .location
= location
,
282 .additional_data
= sl
,
285 /* Ensure, no matter the destruction order, that we always have a handle on the global sig_state */
286 if (!talloc_reference(se
, sig_state
)) {
291 /* only install a signal handler if not already installed */
292 if (sig_state
->sig_handlers
[signum
] == NULL
) {
293 struct sigaction act
;
295 act
.sa_handler
= tevent_common_signal_handler
;
296 act
.sa_flags
= sa_flags
;
298 if (sa_flags
& SA_SIGINFO
) {
299 act
.sa_handler
= NULL
;
300 act
.sa_sigaction
= tevent_common_signal_handler_info
;
301 if (sig_state
->sig_info
[signum
] == NULL
) {
302 sig_state
->sig_info
[signum
] =
303 talloc_zero_array(sig_state
, siginfo_t
,
304 TEVENT_SA_INFO_QUEUE_COUNT
);
305 if (sig_state
->sig_info
[signum
] == NULL
) {
312 sig_state
->oldact
[signum
] = talloc_zero(sig_state
, struct sigaction
);
313 if (sig_state
->oldact
[signum
] == NULL
) {
317 if (sigaction(signum
, &act
, sig_state
->oldact
[signum
]) == -1) {
318 talloc_free(sig_state
->oldact
[signum
]);
319 sig_state
->oldact
[signum
] = NULL
;
325 DLIST_ADD(se
->event_ctx
->signal_events
, se
);
327 /* Make sure the signal doesn't come in while we're mangling list. */
329 sigaddset(&set
, signum
);
330 sigprocmask(SIG_BLOCK
, &set
, &oldset
);
331 tevent_trace_signal_callback(se
->event_ctx
, se
, TEVENT_EVENT_TRACE_ATTACH
);
332 DLIST_ADD(sig_state
->sig_handlers
[signum
], sl
);
333 sigprocmask(SIG_SETMASK
, &oldset
, NULL
);
335 talloc_set_destructor(se
, tevent_signal_destructor
);
336 talloc_set_destructor(sl
, tevent_common_signal_list_destructor
);
341 int tevent_common_invoke_signal_handler(struct tevent_signal
*se
,
342 int signum
, int count
, void *siginfo
,
345 struct tevent_context
*handler_ev
= se
->event_ctx
;
348 if (removed
!= NULL
) {
352 if (se
->event_ctx
== NULL
) {
357 if (se
->wrapper
!= NULL
) {
358 handler_ev
= se
->wrapper
->wrap_ev
;
360 tevent_wrapper_push_use_internal(handler_ev
, se
->wrapper
);
361 se
->wrapper
->ops
->before_signal_handler(
362 se
->wrapper
->wrap_ev
,
363 se
->wrapper
->private_state
,
364 se
->wrapper
->main_ev
,
372 tevent_trace_signal_callback(se
->event_ctx
, se
, TEVENT_EVENT_TRACE_BEFORE_HANDLER
);
373 se
->handler(handler_ev
, se
, signum
, count
, siginfo
, se
->private_data
);
374 if (se
->wrapper
!= NULL
) {
375 se
->wrapper
->ops
->after_signal_handler(
376 se
->wrapper
->wrap_ev
,
377 se
->wrapper
->private_state
,
378 se
->wrapper
->main_ev
,
385 tevent_wrapper_pop_use_internal(handler_ev
, se
->wrapper
);
390 if (se
->sa_flags
& SA_RESETHAND
) {
396 talloc_set_destructor(se
, NULL
);
402 if (removed
!= NULL
) {
411 check if a signal is pending
412 return != 0 if a signal was pending
414 int tevent_common_check_signal(struct tevent_context
*ev
)
418 if (!sig_state
|| !TEVENT_SIG_PENDING(sig_state
->got_signal
)) {
422 for (i
=0;i
<TEVENT_NUM_SIGNALS
+1;i
++) {
423 struct tevent_common_signal_list
*sl
, *next
;
424 struct tevent_sigcounter counter
= sig_state
->signal_count
[i
];
425 uint32_t count
= tevent_sig_count(counter
);
428 /* Ensure we null out any stored siginfo_t entries
429 * after processing for debugging purposes. */
430 bool clear_processed_siginfo
= false;
436 for (sl
=sig_state
->sig_handlers
[i
];sl
;sl
=next
) {
437 struct tevent_signal
*se
= sl
->se
;
442 if (se
->sa_flags
& SA_SIGINFO
) {
445 clear_processed_siginfo
= true;
447 for (j
=0;j
<count
;j
++) {
448 /* sig_state->signal_count[i].seen
449 * % TEVENT_SA_INFO_QUEUE_COUNT is
450 * the base position of the unprocessed
451 * signals in the ringbuffer. */
452 uint32_t ofs
= (counter
.seen
+ j
)
453 % TEVENT_SA_INFO_QUEUE_COUNT
;
454 bool removed
= false;
456 ret
= tevent_common_invoke_signal_handler(
458 (void*)&sig_state
->sig_info
[i
][ofs
],
461 tevent_abort(ev
, "tevent_common_invoke_signal_handler() failed");
471 ret
= tevent_common_invoke_signal_handler(se
, i
, count
,
474 tevent_abort(ev
, "tevent_common_invoke_signal_handler() failed");
479 if (clear_processed_siginfo
&& sig_state
->sig_info
[i
] != NULL
) {
481 for (j
=0;j
<count
;j
++) {
482 uint32_t ofs
= (counter
.seen
+ j
)
483 % TEVENT_SA_INFO_QUEUE_COUNT
;
484 memset((void*)&sig_state
->sig_info
[i
][ofs
],
491 TEVENT_SIG_SEEN(sig_state
->signal_count
[i
], count
);
492 TEVENT_SIG_SEEN(sig_state
->got_signal
, count
);
495 if (TEVENT_SIG_PENDING(sig_state
->sig_blocked
[i
])) {
496 /* We'd filled the queue, unblock the
497 signal now the queue is empty again.
498 Note we MUST do this after the
499 TEVENT_SIG_SEEN(sig_state->signal_count[i], count)
500 call to prevent a new signal running
501 out of room in the sig_state->sig_info[i][]
506 TEVENT_SIG_SEEN(sig_state
->sig_blocked
[i
],
507 tevent_sig_count(sig_state
->sig_blocked
[i
]));
508 sigprocmask(SIG_UNBLOCK
, &set
, NULL
);
516 void tevent_cleanup_pending_signal_handlers(struct tevent_signal
*se
)
518 tevent_signal_destructor(se
);
519 talloc_set_destructor(se
, NULL
);
523 void tevent_signal_set_tag(struct tevent_signal
*se
, uint64_t tag
)
532 uint64_t tevent_signal_get_tag(const struct tevent_signal
*se
)