Viewing file: ctrl_c.c (5.69 KB) -rw-r--r-- Select action/file-type: (+) | (+) | (+) | Code (+) | Session (+) | (+) | SDB (+) | (+) | (+) | (+) | (+) | (+) |
/**************************************************************************** * * * GNAT COMPILER COMPONENTS * * * * C T R L _ C * * * * C Implementation File * * * * Copyright (C) 2002-2022, Free Software Foundation, Inc. * * * * GNAT is free software; you can redistribute it and/or modify it under * * terms of the GNU General Public License as published by the Free Soft- * * ware Foundation; either version 3, or (at your option) any later ver- * * sion. GNAT is distributed in the hope that it will be useful, but WITH- * * OUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY * * or FITNESS FOR A PARTICULAR PURPOSE. * * * * As a special exception under Section 7 of GPL version 3, you are granted * * additional permissions described in the GCC Runtime Library Exception, * * version 3.1, as published by the Free Software Foundation. * * * * You should have received a copy of the GNU General Public License and * * a copy of the GCC Runtime Library Exception along with this program; * * see the files COPYING3 and COPYING.RUNTIME respectively. If not, see * * <http://www.gnu.org/licenses/>. * * * * GNAT was originally developed by the GNAT team at New York University. * * Extensive contributions were provided by Ada Core Technologies Inc. * * * ****************************************************************************/
#ifndef IN_RTS #include "config.h" #include "system.h" #endif
/* Services to intercept Ctrl-C */
/* __gnat_install_int_handler will install the specified handler. If called for the first time, it will also save the original handler */ void __gnat_install_int_handler (void (*) (void));
/* __gnat_uninstall_int_handler will reinstall the original handler */ void __gnat_uninstall_int_handler (void);
/* POSIX implementation */
#if (defined (__unix__) || defined (_AIX) || defined (__APPLE__)) \ || defined (VMS) && !defined (__vxworks)
#ifdef VMS /* On VMS _gnat_handle_vms_condition gets control first, and it has to resignal the Ctrl/C in order for sigaction to gain control and execute the user handler routine, but in doing so propagates the condition causing the program to terminate. So instead we install a dummy handler routine and put the real user handler in a special global variable so that __gnat_handle_vms_condition can declare an AST to asynchronously execute the Ctrl/C user handler at some future time and allow __gnat_handle_vms_condition to return and not be held up waiting for the potentially unbounded time required to execute the Crtl/C handler. */ void dummy_handler () {}
/* Lives in init.c. */ extern void (*__gnat_ctrl_c_handler) (void); #endif
#include <signal.h>
void (*sigint_intercepted) (void) = 0;
struct sigaction original_act;
static void __gnat_int_handler (int sig __attribute__ ((unused))) { if (sigint_intercepted != 0) sigint_intercepted (); }
/* Install handler and save original handler. */
void __gnat_install_int_handler (void (*proc) (void)) { struct sigaction act;
if (sigint_intercepted == 0) { act.sa_handler = __gnat_int_handler; #if defined (__Lynx__) || defined (VMS) || defined(__DJGPP__) /* LynxOS, VMS and DJGPP do not support SA_RESTART. */ act.sa_flags = 0; #else act.sa_flags = SA_RESTART; #endif sigemptyset (&act.sa_mask); sigaction (SIGINT, &act, &original_act); }
#ifdef VMS sigint_intercepted = &dummy_handler; __gnat_ctrl_c_handler = proc; #else sigint_intercepted = proc; #endif }
/* Restore original handler */
void __gnat_uninstall_int_handler (void) { if (sigint_intercepted != 0) { sigaction (SIGINT, &original_act, 0); sigint_intercepted = 0; } #ifdef VMS if (__gnat_ctrl_c_handler) __gnat_ctrl_c_handler = 0; #endif }
/* Windows implementation */
#elif defined (__MINGW32__)
#include "mingw32.h" #include <windows.h>
void (*sigint_intercepted) (void) = NULL;
static BOOL WINAPI __gnat_int_handler (DWORD dwCtrlType) { switch (dwCtrlType) { case CTRL_C_EVENT: case CTRL_BREAK_EVENT: if (sigint_intercepted != 0) { sigint_intercepted (); return TRUE; } break;
case CTRL_CLOSE_EVENT: case CTRL_LOGOFF_EVENT: case CTRL_SHUTDOWN_EVENT: break; }
return FALSE; }
void __gnat_install_int_handler (void (*proc) (void)) { if (sigint_intercepted == NULL) SetConsoleCtrlHandler (__gnat_int_handler, TRUE);
sigint_intercepted = proc; }
void __gnat_uninstall_int_handler (void) { if (sigint_intercepted != NULL) SetConsoleCtrlHandler (__gnat_int_handler, FALSE);
sigint_intercepted = NULL; }
/* Default implementation: do nothing */
#else
void __gnat_install_int_handler (void (*proc) (void) __attribute__ ((unused))) { }
void __gnat_uninstall_int_handler (void) { } #endif
|