summaryrefslogtreecommitdiffhomepage
path: root/src/nxt_signal.c
diff options
context:
space:
mode:
Diffstat (limited to '')
-rw-r--r--src/nxt_signal.c230
1 files changed, 230 insertions, 0 deletions
diff --git a/src/nxt_signal.c b/src/nxt_signal.c
new file mode 100644
index 00000000..e267e3eb
--- /dev/null
+++ b/src/nxt_signal.c
@@ -0,0 +1,230 @@
+
+/*
+ * Copyright (C) Igor Sysoev
+ * Copyright (C) NGINX, Inc.
+ */
+
+#include <nxt_main.h>
+
+
+/*
+ * Signals are handled only via a main thread event engine work queue.
+ * There are three ways to route signals to the work queue:
+ *
+ * 1) Using signal event notifications if an event facility supports it:
+ * kqueue and epoll/signalfd. This method is used regardless of thread mode.
+ *
+ * 2) Multi-threaded mode: a dedicated signal thread which waits in sigwait()
+ * and post a signal number to the main thread event engine.
+ *
+ * 3) Single-threaded mode: a signal handler which posts a signal number
+ * to the event engine.
+ */
+
+
+static nxt_int_t nxt_signal_action(int signo, void (*handler)(int));
+
+
+nxt_event_signals_t *
+nxt_event_engine_signals(const nxt_event_sig_t *sigev)
+{
+ nxt_event_signals_t *signals;
+
+ signals = nxt_zalloc(sizeof(nxt_event_signals_t));
+ if (signals == NULL) {
+ return NULL;
+ }
+
+ signals->sigev = sigev;
+
+ if (nxt_signal_action(SIGSYS, SIG_IGN) != NXT_OK) {
+ goto fail;
+ }
+
+ if (nxt_signal_action(SIGPIPE, SIG_IGN) != NXT_OK) {
+ goto fail;
+ }
+
+ sigemptyset(&signals->sigmask);
+
+ while (sigev->signo != 0) {
+ sigaddset(&signals->sigmask, sigev->signo);
+ sigev++;
+ }
+
+ if (sigprocmask(SIG_BLOCK, &signals->sigmask, NULL) != 0) {
+ nxt_main_log_alert("sigprocmask(SIG_BLOCK) failed %E", nxt_errno);
+ goto fail;
+ }
+
+ return signals;
+
+fail:
+
+ nxt_free(signals);
+
+ return NULL;
+}
+
+
+static nxt_int_t
+nxt_signal_action(int signo, void (*handler)(int))
+{
+ struct sigaction sa;
+
+ nxt_memzero(&sa, sizeof(struct sigaction));
+ sigemptyset(&sa.sa_mask);
+ sa.sa_handler = handler;
+
+ if (sigaction(signo, &sa, NULL) == 0) {
+ return NXT_OK;
+ }
+
+ nxt_main_log_alert("sigaction(%d) failed %E", signo, nxt_errno);
+
+ return NXT_ERROR;
+}
+
+
+static void
+nxt_signal_handler(int signo)
+{
+ nxt_thread_t *thr;
+
+ thr = nxt_thread();
+
+ /* Thread is running in a single context now. */
+ thr->time.signal++;
+
+ nxt_thread_time_update(thr);
+
+ nxt_main_log_error(NXT_LOG_INFO, "signal handler: %d", signo);
+
+ nxt_event_engine_signal(thr->engine, signo);
+
+ thr->time.signal--;
+}
+
+
+#if (NXT_THREADS)
+
+static void nxt_signal_thread(void *data);
+
+
+nxt_int_t
+nxt_signal_thread_start(nxt_event_engine_t *engine)
+{
+ nxt_thread_link_t *link;
+ const nxt_event_sig_t *sigev;
+
+ if (engine->signals->process == nxt_pid) {
+ return NXT_OK;
+ }
+
+ if (sigprocmask(SIG_BLOCK, &engine->signals->sigmask, NULL) != 0) {
+ nxt_main_log_alert("sigprocmask(SIG_BLOCK) failed %E", nxt_errno);
+ return NXT_ERROR;
+ }
+
+ /*
+ * kqueue sets signal handlers to SIG_IGN and sigwait() ignores
+ * them after the switch of event facility from "kqueue" to "select".
+ */
+
+ for (sigev = engine->signals->sigev; sigev->signo != 0; sigev++) {
+ if (nxt_signal_action(sigev->signo, nxt_signal_handler) != NXT_OK) {
+ return NXT_ERROR;
+ }
+ }
+
+ link = nxt_zalloc(sizeof(nxt_thread_link_t));
+
+ if (nxt_fast_path(link != NULL)) {
+ link->start = nxt_signal_thread;
+ link->data = engine;
+
+ if (nxt_thread_create(&engine->signals->thread, link) == NXT_OK) {
+ engine->signals->process = nxt_pid;
+ return NXT_OK;
+ }
+ }
+
+ return NXT_ERROR;
+}
+
+
+static void
+nxt_signal_thread(void *data)
+{
+ int signo;
+ nxt_err_t err;
+ nxt_thread_t *thr;
+ nxt_event_engine_t *engine;
+
+ engine = data;
+
+ thr = nxt_thread();
+
+ nxt_main_log_debug("signal thread");
+
+ for ( ;; ) {
+ err = sigwait(&engine->signals->sigmask, &signo);
+
+ nxt_thread_time_update(thr);
+
+ if (nxt_fast_path(err == 0)) {
+ nxt_main_log_error(NXT_LOG_INFO, "signo: %d", signo);
+
+ nxt_event_engine_signal(engine, signo);
+
+ } else {
+ nxt_main_log_alert("sigwait() failed %E", err);
+ }
+ }
+}
+
+
+void
+nxt_signal_thread_stop(nxt_event_engine_t *engine)
+{
+ nxt_thread_handle_t thread;
+
+ thread = engine->signals->thread;
+
+ nxt_thread_cancel(thread);
+ nxt_thread_wait(thread);
+}
+
+
+#else /* !(NXT_THREADS) */
+
+
+nxt_int_t
+nxt_signal_handlers_start(nxt_event_engine_t *engine)
+{
+ const nxt_event_sig_t *sigev;
+
+ for (sigev = engine->signals->sigev; sigev->signo != 0; sigev++) {
+ if (nxt_signal_action(sigev->signo, nxt_signal_handler) != NXT_OK) {
+ return NXT_ERROR;
+ }
+ }
+
+ if (sigprocmask(SIG_UNBLOCK, &engine->signals->sigmask, NULL) != 0) {
+ nxt_main_log_alert("sigprocmask(SIG_UNBLOCK) failed %E", nxt_errno);
+ return NXT_ERROR;
+ }
+
+ return NXT_OK;
+}
+
+
+void
+nxt_signal_handlers_stop(nxt_event_engine_t *engine)
+{
+ if (sigprocmask(SIG_BLOCK, &engine->signals->sigmask, NULL) != 0) {
+ nxt_main_log_alert("sigprocmask(SIG_BLOCK) failed %E", nxt_errno);
+ }
+}
+
+#endif