mirror of
https://github.com/nasa/trick.git
synced 2025-01-07 13:48:43 +00:00
157 lines
6.0 KiB
C++
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) ;
|
|
}
|
|
|