#include <cstddef> #include <pthread.h> #include <signal.h> #include <stdlib.h> #include <unistd.h> #include <errno.h> #include <cstdlib> #include <ctime> #include <climits> #include <vector> #include <cstring> #include "logging_test.h" Faulty_Logger faulty_logger; NXLog good_logger; void print_sigmask () { sigset_t orig_mask; sigemptyset (&orig_mask); pthread_sigmask (SIG_SETMASK, NULL, &orig_mask); bool empty = true; for (std::size_t i = 0; i < NSIG; ++i) { if (sigismember (&orig_mask, i)) { nxdbg_good << "Signal i (" << i << ") in signal mask." << std::endl; empty = false; } } if (empty) { nxdbg_good << "Signal mask empty."; } } void* log_task (void* /* unused */) { /* print_sigmask (); */ for (std::size_t i = 0; i < 10; ++i) { nxinfo << "Log message " << i << std::endl; } } void sig_handler (int signo) { nxinfo << "Received signal " << signo << std::endl; } void setup_faulty_logger () { faulty_logger.log_level (true); faulty_logger.log_unix_time (false); faulty_logger.log_time (true); faulty_logger.log_location (true); faulty_logger.log_thread_id (true); faulty_logger.level (NXDEBUG); } void setup_good_logger () { good_logger.log_level (true); good_logger.log_unix_time (false); good_logger.log_time (true); good_logger.log_location (true); good_logger.log_thread_id (true); good_logger.level (NXDEBUG); } pthread_t spawn_thread () { pthread_t thread_id; int pthread_ret; sigset_t block_mask, orig_mask; sigemptyset (&orig_mask); sigfillset (&block_mask); pthread_sigmask (SIG_BLOCK, &block_mask, &orig_mask); pthread_ret = pthread_create (&thread_id, NULL, log_task, NULL); pthread_sigmask (SIG_SETMASK, &orig_mask, NULL); return (thread_id); } void install_signal_handler () { struct sigaction sa; sa.sa_handler = sig_handler; sigemptyset (&sa.sa_mask); sa.sa_flags = SA_RESTART; if (-1 == sigaction (SIGUSR1, &sa, NULL)) { nxerr_good << "Unable to install signal handler!" << std::endl; } else { nxdbg_good << "Signal handler registered successfully for SIGUSR1." << std::endl; } } void killing_process_work (pid_t parent_pid) { /* Seed PRNG. */ std::srand (std::time (0)); for (std::size_t i = 0; i < 25; ++i) { /* Sleep for 4 seconds + some random number up to a second. */ std::size_t rand_add = (std::rand () % 1000000); usleep (4000000 + rand_add); /* Send SIGUSR1 to parent process. */ nxdbg_good << "Sending SIGUSR1 (" << SIGUSR1 << ") to parent_pid (" << parent_pid << ")" << std::endl; if (kill (parent_pid, SIGUSR1)) { int saved_errno = errno; nxerr_good << "Failed to deliver signal to parent, aborting." << std::endl; nxerr_good << "Error " << saved_errno << ": " << strerror (saved_errno) << std::endl; exit (EXIT_FAILURE); } } exit (EXIT_SUCCESS); } void killing_process_init (int argc, char **argv) { /* We're in the "killing process". */ pid_t parent_pid = getppid (); setup_good_logger (); for (std::size_t i = 0; i < argc; ++i) { nxdbg_good << "argv[" << i << "]: " << argv[i] << std::endl; } char *end = NULL; errno = 0; long parent_pid_check = std::strtol (argv[1], &end, 0); if ((errno == ERANGE) && (parent_pid_check == LONG_MAX)) { /* Overflow, handle gracefully. */ parent_pid_check = 1; } if ((errno == ERANGE) && (parent_pid_check == LONG_MIN)) { /* Underflow, handle gracefully. */ parent_pid_check = 1; } if (*end) { /* Conversion error (for inputs like "<number>X", end will point to X.) */ parent_pid_check = 1; } if (parent_pid != parent_pid_check) { nxinfo_good << "Parent PID verification via first argument failed, trusting getppid ()." << std::endl; } killing_process_work (parent_pid); } int main (int argc, char **argv) { if (argc > 1) { killing_process_init (argc, argv); } else { /* That's the main process. */ /* First, fork and create the "killing process". */ pid_t pid = fork (); if (0 == pid) { /* Child process. */ pid_t parent_pid = getppid (); /* Prepare to pass-through parent PID. */ std::stringstream ss; ss << parent_pid; std::vector<std::string> new_argv; new_argv.push_back (std::string (argv[0])); new_argv.push_back (ss.str ()); std::vector<char *> new_argv_c_str; for (std::vector<std::string>::iterator it = new_argv.begin (); it != new_argv.end (); ++it) { const char *elem = (*it).c_str (); new_argv_c_str.push_back (strndup (elem, std::strlen (elem))); } /* Add null pointer as last element. */ new_argv_c_str.push_back (0); /* Relaunch, with argv[1] containing the ppid. */ if (0 != execvp (new_argv_c_str.front (), &(new_argv_c_str.front ()))) { const int saved_errno = errno; std::cerr << "Failed to start \"killing process\"! Panic!" << std::endl; std::cerr << "System error: " << std::strerror (saved_errno) << std::endl; exit (EXIT_FAILURE); } } else if (0 > pid) { const int saved_errno = errno; std::cerr << "Error while forking main process! Panic!" << std::endl; std::cerr << "System error: " << std::strerror (saved_errno) << std::endl; exit (EXIT_FAILURE); } else { /* Main process. */ /* Falls through to general code below. */ } } setup_faulty_logger (); pthread_t thread_id = spawn_thread (); setup_good_logger (); install_signal_handler (); /* print_sigmask (); */ log_task (NULL); pthread_join (thread_id, NULL); exit (EXIT_SUCCESS); }