2015-02-26 15:02:31 +00:00
/*
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>
2024-02-27 21:07:48 +00:00
# include <execinfo.h>
2015-02-26 15:02:31 +00:00
# endif
2015-06-01 15:28:29 +00:00
# include "trick/Executive.hh"
2015-02-26 15:02:31 +00:00
/**
@ 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 ;
2016-04-15 19:57:22 +00:00
case SIGABRT : /* Abort */
write ( 2 , " SIGABRT " , 7 ) ;
break ;
2015-02-26 15:02:31 +00:00
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 .
*/
2016-04-20 19:12:03 +00:00
if ( ! debugger_command . empty ( ) ) {
2015-02-26 15:02:31 +00:00
# if __linux
2016-04-20 19:12:03 +00:00
char command [ 1024 ] ;
2015-02-26 15:02:31 +00:00
if ( attach_debugger = = true ) {
2022-11-15 21:00:05 +00:00
snprintf ( command , sizeof ( command ) , " %s -silent /proc/%d/exe %d " , debugger_command . c_str ( ) , getpid ( ) , getpid ( ) ) ;
2015-02-26 15:02:31 +00:00
system ( command ) ;
} else if ( stack_trace = = true ) {
2022-11-15 21:00:05 +00:00
snprintf ( command , sizeof ( command ) , " %s -silent -batch -x ${TRICK_HOME}/share/trick/gdb_commands "
2016-04-20 19:12:03 +00:00
" /proc/%d/exe %d " , debugger_command . c_str ( ) , getpid ( ) , getpid ( ) ) ;
2015-02-26 15:02:31 +00:00
system ( command ) ;
}
2024-02-27 21:07:48 +00:00
//=========================================================================================`
# 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
}
}
//=========================================================================================
2015-02-26 15:02:31 +00:00
# endif
2016-04-20 19:12:03 +00:00
}
2015-02-26 15:02:31 +00:00
// 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. \n If 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 ) ;
}