gambas-source-code/main/gbx/gbx_signal.c
Benoît Minisini fcc386d0f6 When a signal handler is set, ensure that the signal is not blocked.
[INTERPRETER]
* BUG: When a signal handler is set, ensure that the signal is not blocked.
2023-04-12 20:55:05 +02:00

501 lines
9.9 KiB
C

/***************************************************************************
gbx_signal.c
(c) 2000-2017 Benoît Minisini <benoit.minisini@gambas-basic.org>
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 2, or (at your option)
any later version.
This program 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 General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston,
MA 02110-1301, USA.
***************************************************************************/
#define __GBX_SIGNAL_C
#include "gb_alloc.h"
#include "gb_error.h"
#include "gb_array.h"
#include "gbx_api.h"
#include "gbx_c_process.h"
#include "gbx_c_task.h"
#include "gbx_signal.h"
//#define DEBUG_ME 1
uint64_t SIGNAL_check_mask = 0;
static SIGNAL_HANDLER *_handlers = NULL;
static int _pipe[2] = { -1, -1 };
static volatile int _count = 0;
static int _raising_callback = 0;
enum { CB_NONE = 0, CB_HANDLER = 1, CB_ACTION = 2 };
static int get_callback(struct sigaction *action, void (**callback)())
{
if (action->sa_handler != SIG_DFL && action->sa_handler != SIG_IGN)
{
if (action->sa_flags & SA_SIGINFO)
{
*callback = (void *)action->sa_sigaction;
return CB_ACTION;
}
else
{
*callback = (void *)action->sa_handler;
return CB_HANDLER;
}
}
else
{
*callback = NULL;
return CB_NONE;
}
}
void SIGNAL_install(SIGNAL_HANDLER *handler, int signum, void (*callback)(int, siginfo_t *, void *))
{
struct sigaction action;
sigset_t sig;
#if DEBUG_ME
fprintf(stderr, "SIGNAL_install: %d %p\n", signum, callback);
#endif
handler->signum = signum;
action.sa_flags = SA_SIGINFO;
// According to manpage, the emitting signal is blocked by default.
sigemptyset(&action.sa_mask);
action.sa_sigaction = callback;
if (sigaction(signum, NULL, &handler->old_action) != 0 || sigaction(signum, &action, NULL) != 0)
ERROR_panic("Cannot install signal handler: %s", strerror(errno));
// Ensure that the signal is not blocked
sigemptyset(&sig);
sigaddset(&sig, signum);
sigprocmask(SIG_UNBLOCK, &sig, NULL);
}
void SIGNAL_uninstall(SIGNAL_HANDLER *handler, int signum)
{
#if DEBUG_ME
fprintf(stderr, "SIGNAL_uninstall: %d\n", signum);
#endif
if (sigaction(signum, &handler->old_action, NULL) != 0)
ERROR_panic("Cannot uninstall signal handler");
while (handler->callbacks)
SIGNAL_unregister(handler->signum, handler->callbacks);
}
void SIGNAL_previous(SIGNAL_HANDLER *handler, int signum, siginfo_t *info, void *context)
{
void (*cb)();
switch (get_callback(&handler->old_action, &cb))
{
case CB_ACTION:
(*cb)(signum, info, context);
break;
case CB_HANDLER:
(*cb)(signum);
break;
default:
; /* do nothing */
}
}
static SIGNAL_HANDLER *find_handler(int signum)
{
int i;
for (i = 0; i < ARRAY_count(_handlers); i++)
{
if (_handlers[i].signum == signum)
return &_handlers[i];
}
return NULL;
}
static SIGNAL_HANDLER *add_handler(void)
{
if (!_handlers)
ARRAY_create_inc(&_handlers, 1);
return ARRAY_add_void(&_handlers);
}
static void handle_signal(int signum, siginfo_t *info, void *context)
{
char buffer;
int save_errno;
static volatile int64_t lock = 0;
if (lock & (1 << signum))
return;
save_errno = errno;
lock |= 1 << signum;
#if DEBUG_ME
char digit;
write(2, "[SIGNAL:", 8);
digit = '0' + (signum / 10);
write(2, &digit, 1);
digit = '0' + (signum % 10);
write(2, &digit, 1);
write(2, "]\n", 2);
#endif
if (_count)
{
buffer = signum;
for(;;)
{
if (write(_pipe[1], &buffer, 1) == 1)
break;
if (errno != EINTR)
{
ERROR_warning("cannot write signal #%d into signal pipe: %s", signum, strerror(errno));
break;
}
}
}
SIGNAL_previous(find_handler(signum), signum, info, context);
errno = save_errno;
lock &= ~(1 << signum);
}
static bool _must_purge_callbacks = FALSE;
static int _purge_signum;
static SIGNAL_HANDLER *_purge_handler;
static void purge_callbacks(void)
{
SIGNAL_CALLBACK *cb, *next_cb;
_raising_callback--;
if (_raising_callback)
return;
#if DEBUG_ME
fprintf(stderr, ">> purge_callbacks\n");
#endif
while (_must_purge_callbacks)
{
_must_purge_callbacks = FALSE;
cb = _purge_handler->callbacks;
while (cb)
{
#if DEBUG_ME
fprintf(stderr, "purge_callbacks: cb = %p\n", cb);
#endif
next_cb = cb->next;
if (!cb->callback)
SIGNAL_unregister(_purge_signum, cb);
cb = next_cb;
}
}
#if DEBUG_ME
fprintf(stderr, "<< purge_callbacks\n");
#endif
}
void SIGNAL_raise_callbacks(int fd, int type, void *data)
{
SIGNAL_HANDLER *handler;
SIGNAL_CALLBACK *cb;
char signum;
int ret;
/*old = signal(SIGCHLD, signal_child);*/
#if DEBUG_ME
fprintf(stderr, "SIGNAL_raise_callbacks: fd = %d blocking = %d\n", fd, (fcntl(fd, F_GETFL) & O_NONBLOCK) == 0);
#endif
for(;;)
{
ret = read(fd, &signum, 1);
if (ret != 1)
{
#if DEBUG_ME
fprintf(stderr, "SIGNAL_raise_callbacks: read -> %d / errno = %d\n", ret, errno);
#endif
return;
}
#if DEBUG_ME
fprintf(stderr, "SIGNAL_raise_callbacks: signum = %d\n", signum);
#endif
handler = find_handler(signum);
if (!handler)
{
#if DEBUG_ME
fprintf(stderr, "SIGNAL_raise_callbacks: no handler\n");
#endif
return;
}
#if DEBUG_ME
fprintf(stderr, ">> SIGNAL_raise_callbacks (%d)\n", _raising_callback);
#endif
_raising_callback++;
_purge_signum = signum;
_purge_handler = handler;
ON_ERROR(purge_callbacks)
{
cb = handler->callbacks;
while (cb)
{
#if DEBUG_ME
fprintf(stderr, "SIGNAL_raise_callbacks: cb = %p cb->callback = %p\n", cb, cb->callback);
#endif
if (cb->callback)
(*cb->callback)((int)signum, cb->data);
cb = cb->next;
}
}
END_ERROR
#if DEBUG_ME
fprintf(stderr, "SIGNAL_raise_callbacks: purge_callbacks\n");
#endif
purge_callbacks();
#if DEBUG_ME
fprintf(stderr, "<< SIGNAL_raise_callbacks (%d)\n", _raising_callback);
#endif
}
}
static void create_pipe(void)
{
if (pipe(_pipe) != 0)
ERROR_panic("Cannot create signal handler pipes: %s", strerror(errno));
if (_pipe[0] == 0)
BREAKPOINT();
fcntl(_pipe[0], F_SETFD, FD_CLOEXEC);
fcntl(_pipe[1], F_SETFD, FD_CLOEXEC);
// Allows to read the signal pipe without blocking
fcntl(_pipe[0], F_SETFL, fcntl(_pipe[0], F_GETFL) | O_NONBLOCK);
GB_Watch(_pipe[0], GB_WATCH_READ, (void *)SIGNAL_raise_callbacks, 0);
#if DEBUG_ME
fprintf(stderr, "create_pipe: fd = %d\n", _pipe[0]);
#endif
}
static void delete_pipe(void)
{
#if DEBUG_ME
fprintf(stderr, "delete_pipe: fd = %d\n", _pipe[0]);
#endif
GB_Watch(_pipe[0], GB_WATCH_NONE, NULL, 0);
close(_pipe[0]);
close(_pipe[1]);
_pipe[0] = -1;
_pipe[1] = -1;
}
SIGNAL_CALLBACK *SIGNAL_register(int signum, void (*callback)(int, intptr_t), intptr_t data)
{
SIGNAL_HANDLER *handler;
SIGNAL_CALLBACK *cb;
if (!_count)
create_pipe();
_count++;
handler = find_handler(signum);
#if DEBUG_ME
fprintf(stderr, "SIGNAL_register: find_handler(%d) -> %p\n", signum, handler);
#endif
if (!handler)
{
handler = add_handler();
SIGNAL_install(handler, signum, handle_signal);
}
ALLOC(&cb, sizeof(SIGNAL_CALLBACK));
cb->prev = NULL;
cb->next = handler->callbacks;
cb->callback = callback;
cb->data = data;
if (cb->next)
cb->next->prev = cb;
handler->callbacks = cb;
#if DEBUG_ME
fprintf(stderr, "SIGNAL_register: %d -> %p (%p)\n", signum, cb, cb->callback);
#endif
#if DEBUG_ME
fprintf(stderr, "handler->callbacks %p:", handler);
SIGNAL_CALLBACK *save = cb;
cb = handler->callbacks;
while (cb)
{
fprintf(stderr, " -> %p (%p)", cb, cb->callback);
cb = cb->next;
}
fprintf(stderr, "\n");
cb = save;
#endif
return cb;
}
void SIGNAL_unregister(int signum, SIGNAL_CALLBACK *cb)
{
SIGNAL_HANDLER *handler = find_handler(signum);
if (!handler)
return;
if (_raising_callback)
{
#if DEBUG_ME
fprintf(stderr, "SIGNAL_unregister: disable %d %p (%p)\n", signum, cb, cb->callback);
#endif
cb->callback = NULL;
_must_purge_callbacks = TRUE;
return;
}
#if DEBUG_ME
fprintf(stderr, "SIGNAL_unregister: remove %d %p (%p)\n", signum, cb, cb->callback);
#endif
if (cb->prev)
cb->prev->next = cb->next;
if (cb->next)
cb->next->prev = cb->prev;
if (cb == handler->callbacks)
handler->callbacks = cb->next;
IFREE(cb);
_count--;
if (_count == 0)
delete_pipe();
#if DEBUG_ME
fprintf(stderr, "handler->callbacks %p:", handler);
cb = handler->callbacks;
while (cb)
{
fprintf(stderr, " -> %p (%p)", cb, cb->callback);
cb = cb->next;
}
fprintf(stderr, "\n");
#endif
}
void SIGNAL_exit(void)
{
int i;
SIGNAL_HANDLER *handler;
if (_handlers)
{
_raising_callback = 0;
for (i = 0; i < ARRAY_count(_handlers); i++)
{
handler = &_handlers[i];
SIGNAL_uninstall(handler, handler->signum);
}
ARRAY_delete(&_handlers);
}
}
int SIGNAL_get_fd(void)
{
return _pipe[0];
}
void SIGNAL_has_forked(void)
{
if (!_count)
return;
GB_Watch(_pipe[0], GB_WATCH_NONE, NULL, 0);
close(_pipe[0]);
close(_pipe[1]);
create_pipe();
}
void SIGNAL_do_check(int signum)
{
struct sigaction action;
SIGNAL_HANDLER *handler = find_handler(signum);
void (*cb)();
if (!handler)
return;
sigaction(signum, NULL, &action);
get_callback(&action, &cb);
#if DEBUG_ME
fprintf(stderr, "SIGNAL_check: %d -> %d (action.sa_sigaction = %p)\n", signum, cb == handle_signal, cb);
if (cb != handle_signal)
BREAKPOINT();
#endif
if (cb == handle_signal)
return;
SIGNAL_install(handler, signum, handle_signal);
if (signum == SIGCHLD)
{
CPROCESS_callback_child();
CTASK_callback_child();
}
}
void SIGNAL_must_check(int signum)
{
SIGNAL_check_mask |= (1 << signum);
}