trick/trick_source/sim_services/Executive/Executive_signal_handler.cpp
2024-02-27 15:07:48 -06:00

157 lines
6.0 KiB
C++

/*
PURPOSE: (This function switches on signal type to act on specific signal. This function performs the signal
handling for both master and child processes.)
REFERENCE: (((Bailey, R.W, and Paddock, E.J.) (Trick Simulation Environment) (NASA:JSC #37943) (JSC / Engineering
Directorate / Automation and Robotics Division) (June 1994) (--)))
ASSUMPTIONS AND LIMITATIONS: ((None))
CLASS: (N/A)
LIBRARY DEPENDENCY: ((none))
PROGRAMMERS: (((Eddie J. Paddock) (MDSSC) (April 1992) (--) (Realtime)) ((Scott Killingsworth) (LinCom) (September
1997) (--) (Issue #1016))) */
#include <stdlib.h>
#include <signal.h>
#include <setjmp.h>
#include <unistd.h>
#ifdef __APPLE__
#include <mach-o/dyld.h>
#include <execinfo.h>
#endif
#include "trick/Executive.hh"
/**
@design
-# This signal handler processes SIGBUS and SIGSEGV
-# Write a message to stderr that a signal has been caught.
-# If attach_debugger == true, attempt to attach a debugger to the current process.
-# Else if stack_trace == true, attempt to attach a debugger to print a stack trace.
-# Exit the process with a -2 return.
*/
void Trick::Executive::signal_handler(int sig) {
// print a message to stderr. Use the write function because it is async signal safe.
write( 2 , "\033[31mProcess terminated by signal " , 34 ) ;
switch (sig) {
case SIGBUS: /* Bus error */
write( 2 , "SIGBUS" , 6 ) ;
break;
case SIGSEGV: /* Segmentation fault */
write( 2 , "SIGSEGV" , 7 ) ;
break;
case SIGABRT: /* Abort */
write( 2 , "SIGABRT" , 7 ) ;
break;
default: /* Unrecognized signal */
write( 2 , "unknown" , 7 ) ;
break;
}
write( 2 , "\033[0m\n" , 5 ) ;
/**********************************************************************
* Attempt to attach with debugger or print stack trace. Not a requirement.
* sprintf and system are not async signal safe, but we don't have anything to lose.
*/
if ( ! debugger_command.empty() ) {
#if __linux
char command[1024];
if (attach_debugger == true) {
snprintf(command, sizeof(command), "%s -silent /proc/%d/exe %d", debugger_command.c_str(), getpid(), getpid());
system(command);
} else if (stack_trace == true ) {
snprintf(command, sizeof(command), "%s -silent -batch -x ${TRICK_HOME}/share/trick/gdb_commands "
"/proc/%d/exe %d", debugger_command.c_str(), getpid(), getpid());
system(command);
}
//=========================================================================================`
#elif __APPLE__
char command[2048];
char path[1024] ;
uint32_t size = sizeof(path) ;
if (_NSGetExecutablePath(path, &size) == 0 ) {
if (attach_debugger == true) {
snprintf(command, sizeof(command),
"NOTICE!! An instance of LLDB is started and keeping your simulation\n"
"executable suspended. LLDB has not, and cannot attach to your simulation\n"
"(i.e., its parent process) without it and your sim being killed\n"
"by Apple's `System Integrity Protection` (SIP) (presumably for security\n"
"reasons). You may be able to attach LLDB to your suspended simulation\n"
"from a different terminal (an independent process) with: $ lldb -p %d \n\n", getpid());
write( 2, command, strlen(command));
snprintf(command, sizeof(command),
"Note that this may also require the apppropriate `elevated privileges` or that\n"
"your sim executable is code signed (using Apple's codesign utility.)\n");
write( 2, command, strlen(command));
snprintf(command, sizeof(command), "%s", debugger_command.c_str());
system(command);
} else if (stack_trace == true ) {
write( 2 , "=============\n" , 14 ) ;
write( 2 , " STACK TRACE \n" , 14 ) ;
write( 2 , "=============\n" , 14 ) ;
void* callstack[128];
int frames = backtrace(callstack, 128);
backtrace_symbols_fd(callstack, frames, 2); // 1 = stdout 2 = stderr
}
}
//=========================================================================================
#endif
}
// Sim state is unknown, don't return from handler.
_exit(sig) ;
}
/**
@design
-# If enable_freeze == true toggle the mode between run/freeze. Design [@ref d_exec_signal_0]
-# Else if this the first time a <CTRL-C> is handled set the exec_command to ExitCmd.
-# If a graceful shutdown has been tried then write an exit message and exit.
*/
void Trick::Executive::ctrl_c_handler() {
static bool graceful_shutdown_tried = false ;
/* if enable_freeze == true toggle the mode between run/freeze. */
if (enable_freeze == true) {
if ( mode == Run) {
exec_command = FreezeCmd ;
} else {
exec_command = RunCmd ;
}
} else if ( graceful_shutdown_tried == false ) {
write( 2 , "\033[31mSetting exec_command = ExitCmd to shutdown simulation.\nIf process still unresponsive, type <CTRL-C> again for immediate exit.\033[0m\n" , 135 ) ;
exec_command = ExitCmd ;
graceful_shutdown_tried = true ;
} else {
write( 2 , "\033[31mProcess terminated by signal SIGINT <CTRL-C>\033[0m\n" , 54 ) ;
_exit(SIGINT) ;
}
}
/**
@design
-# Write an exit message
-# Exit the simulation immediately
*/
void Trick::Executive::term_handler() {
write( 2 , "\033[31mProcess terminated by signal SIGTERM\033[0m\n" , 46 ) ;
_exit(SIGTERM) ;
}
/**
@design
-# Write an exit message
-# Exit the simulation immediately
*/
void Trick::Executive::usr1_handler() {
write( 2 , "\033[31mProcess terminated by call to exec_signal_terminate()\033[0m\n" , 63 ) ;
_exit(SIGUSR1) ;
}