/* * Copyright (C) 2017 "IoT.bzh" * Author José Bollo <jose.bollo@iot.bzh> * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #define _GNU_SOURCE #include <stdlib.h> #include <stdio.h> #include <signal.h> #include <string.h> #include <setjmp.h> #include <time.h> #include <unistd.h> #include <sys/syscall.h> #include <execinfo.h> #include "sig-monitor.h" #include "verbose.h" #define SIG_FOR_TIMER SIGVTALRM /* local handler */ static _Thread_local sigjmp_buf *error_handler; static _Thread_local int in_safe_dumpstack; /* local timers */ static _Thread_local int thread_timer_set; static _Thread_local timer_t thread_timerid; /* internal signal lists */ static int sigerr[] = { SIG_FOR_TIMER, SIGSEGV, SIGFPE, SIGILL, SIGBUS, 0 }; static int sigterm[] = { SIGINT, SIGABRT, SIGTERM, 0 }; static int exiting = 0; /* * Dumps the current stack */ static void dumpstack(int crop, int signum) { int idx, count, rc; void *addresses[100]; char **locations; char buffer[8000]; size_t pos, length; count = backtrace(addresses, sizeof addresses / sizeof *addresses); if (count <= crop) crop = 0; count -= crop; locations = backtrace_symbols(&addresses[crop], count); if (locations == NULL) ERROR("can't get the backtrace (returned %d addresses)", count); else { length = sizeof buffer - 1; pos = 0; idx = 0; while (pos < length && idx < count) { rc = snprintf(&buffer[pos], length - pos, " [%d/%d] %s\n", idx + 1, count, locations[idx]); pos += rc >= 0 ? rc : 0; idx++; } buffer[length] = 0; if (signum) ERROR("BACKTRACE due to signal %s/%d:\n%s", strsignal(signum), signum, buffer); else ERROR("BACKTRACE:\n%s", buffer); free(locations); } } static void safe_dumpstack_cb(int signum, void *closure) { int *args = closure; if (signum) ERROR("Can't provide backtrace: raised signal %s", strsignal(signum)); else dumpstack(args[0], args[1]); } static void safe_dumpstack(int crop, int signum) { int args[2] = { crop + 3, signum }; in_safe_dumpstack = 1; sig_monitor(0, safe_dumpstack_cb, args); in_safe_dumpstack = 0; } /* * Creates a timer for the current thread * * Returns 0 in case of success */ static inline int timeout_create() { int rc; struct sigevent sevp; if (thread_timer_set) rc = 0; else { sevp.sigev_notify = SIGEV_THREAD_ID; sevp.sigev_signo = SIG_FOR_TIMER; sevp.sigev_value.sival_ptr = NULL; #if defined(sigev_notify_thread_id) sevp.sigev_notify_thread_id = (pid_t)syscall(SYS_gettid); #else sevp._sigev_un._tid = (pid_t)syscall(SYS_gettid); #endif rc = timer_create(CLOCK_THREAD_CPUTIME_ID, &sevp, &thread_timerid); thread_timer_set = !rc; } return 0; } /* * Arms the alarm in timeout seconds for the current thread */ static inline int timeout_arm(int timeout) { int rc; struct itimerspec its; rc = timeout_create(); if (rc == 0) { its.it_interval.tv_sec = 0; its.it_interval.tv_nsec = 0; its.it_value.tv_sec = timeout; its.it_value.tv_nsec = 0; rc = timer_settime(thread_timerid, 0, &its, NULL); } return rc; } /* * Disarms the current alarm */ static inline void timeout_disarm() { if (thread_timer_set) timeout_arm(0); } /* * Destroy any alarm resource for the current thread */ static inline void timeout_delete() { if (thread_timer_set) { timer_delete(thread_timerid); thread_timer_set = 0; } } /* install the handlers */ static int install(void (*handler)(int), int *signals) { int result = 1; struct sigaction sa; sa.sa_handler = handler; sigemptyset(&sa.sa_mask); sa.sa_flags = SA_NODEFER; while(*signals > 0) { if (sigaction(*signals, &sa, NULL) < 0) { ERROR("failed to install signal handler for signal %s: %m", strsignal(*signals)); result = 0; } signals++; } return result; } /* * rescue exit */ static void on_rescue_exit(int signum) { ERROR("Rescue exit for signal %d: %s", signum, strsignal(signum)); _exit(exiting); } /* * Do a safe exit */ static void safe_exit(int code) { install(on_rescue_exit, sigerr); install(on_rescue_exit, sigterm); exiting = code; exit(code); } /* Handles signals that terminate the process */ static void on_signal_terminate (int signum) { if (!in_safe_dumpstack) { ERROR("Terminating signal %d received: %s", signum, strsignal(signum)); if (signum == SIGABRT) safe_dumpstack(3, signum); } safe_exit(1); } /* Handles monitored signals that can be continued */ static void on_signal_error(int signum) { if (in_safe_dumpstack) longjmp(*error_handler, signum); ERROR("ALERT! signal %d received: %s", signum, strsignal(signum)); if (error_handler == NULL && signum == SIG_FOR_TIMER) return; safe_dumpstack(3, signum); // unlock signal to allow a new signal to come if (error_handler != NULL) longjmp(*error_handler, signum); ERROR("Unmonitored signal %d received: %s", signum, strsignal(signum)); safe_exit(2); } int sig_monitor_init() { return (install(on_signal_error, sigerr) & install(on_signal_terminate, sigterm)) - 1; } int sig_monitor_init_timeouts() { return timeout_create(); } void sig_monitor_clean_timeouts() { timeout_delete(); } void sig_monitor(int timeout, void (*function)(int sig, void*), void *arg) { volatile int signum, signum2; sigjmp_buf jmpbuf, *older; older = error_handler; signum = setjmp(jmpbuf); if (signum == 0) { error_handler = &jmpbuf; if (timeout) timeout_arm(timeout); function(0, arg); } else { signum2 = setjmp(jmpbuf); if (signum2 == 0) function(signum, arg); } error_handler = older; if (timeout) timeout_disarm(); }