root/maint/gnulib/lib/sigsegv.in.h

/* [previous][next][first][last][top][bottom][index][help] */
   1 /* Page fault handling library.
   2    Copyright (C) 1998-2021 Free Software Foundation, Inc.
   3 
   4    This program is free software: you can redistribute it and/or modify
   5    it under the terms of the GNU General Public License as published by
   6    the Free Software Foundation; either version 2 of the License, or
   7    (at your option) any later version.
   8 
   9    This program is distributed in the hope that it will be useful,
  10    but WITHOUT ANY WARRANTY; without even the implied warranty of
  11    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  12    GNU General Public License for more details.
  13 
  14    You should have received a copy of the GNU General Public License
  15    along with this program.  If not, see <https://www.gnu.org/licenses/>.  */
  16 
  17 /* Written by Bruno Haible.  */
  18 
  19 #ifndef _SIGSEGV_H
  20 #define _SIGSEGV_H
  21 
  22 /* Get size_t.  */
  23 #include <stddef.h>
  24 
  25 /* Define the fault context structure.  */
  26 #if defined __linux__ || defined __ANDROID__ \
  27     || (defined __FreeBSD__ && (defined __arm__ || defined __armhf__ || defined __arm64__)) \
  28     || defined __NetBSD__ \
  29     || defined _AIX || defined __sun \
  30     || defined __CYGWIN__
  31 /* Linux, FreeBSD, NetBSD, AIX, Solaris, Cygwin */
  32 # include <ucontext.h>
  33 #elif (defined __APPLE__ && defined __MACH__)
  34 /* macOS */
  35 # include <sys/ucontext.h>
  36 #elif defined __HAIKU__
  37 /* Haiku */
  38 # include <signal.h>
  39 #endif
  40 
  41 /* Correct the value of SIGSTKSZ on some systems.
  42    glibc >= 2.34: When _GNU_SOURCE is defined, SIGSTKSZ is no longer a
  43    compile-time constant.  But most programs need a simple constant.
  44    AIX 64-bit: original value 4096 is too small.
  45    HP-UX: original value 8192 is too small.
  46    Solaris 11/x86_64: original value 8192 is too small.  */
  47 #include <signal.h>
  48 #if __GLIBC__ >= 2
  49 # undef SIGSTKSZ
  50 # if defined __ia64__
  51 #  define SIGSTKSZ 262144
  52 # else
  53 #  define SIGSTKSZ 65536
  54 # endif
  55 #endif
  56 #if defined _AIX && defined _ARCH_PPC64
  57 # undef SIGSTKSZ
  58 # define SIGSTKSZ 8192
  59 #endif
  60 #if defined __hpux || (defined __sun && (defined __x86_64__ || defined __amd64__))
  61 # undef SIGSTKSZ
  62 # define SIGSTKSZ 16384
  63 #endif
  64 
  65 /* HAVE_SIGSEGV_RECOVERY
  66    is defined if the system supports catching SIGSEGV.  */
  67 #if defined __linux__ || defined __ANDROID__ || defined __GNU__ \
  68     || defined __FreeBSD_kernel__ || (defined __FreeBSD__ && !(defined __sparc__ || defined __sparc64__)) || defined __DragonFly__ \
  69     || defined __NetBSD__ \
  70     || defined __OpenBSD__ \
  71     || (defined __APPLE__ && defined __MACH__) \
  72     || defined _AIX || defined __sgi || defined __sun \
  73     || defined __CYGWIN__ || defined __HAIKU__
  74 /* Linux, Hurd, GNU/kFreeBSD, FreeBSD, NetBSD, OpenBSD, macOS, AIX, IRIX, Solaris, Cygwin, Haiku */
  75 # define HAVE_SIGSEGV_RECOVERY 1
  76 #endif
  77 
  78 /* HAVE_STACK_OVERFLOW_RECOVERY
  79    is defined if stack overflow can be caught.  */
  80 #if defined __linux__ || defined __ANDROID__ || defined __GNU__ \
  81     || defined __FreeBSD_kernel__ || (defined __FreeBSD__ && !(defined __sparc__ || defined __sparc64__)) || defined __DragonFly__ \
  82     || (defined __NetBSD__ && !(defined __sparc__ || defined __sparc64__)) \
  83     || defined __OpenBSD__ \
  84     || (defined __APPLE__ && defined __MACH__) \
  85     || defined _AIX || defined __sgi || defined __sun \
  86     || defined __CYGWIN__ || defined __HAIKU__
  87 /* Linux, Hurd, GNU/kFreeBSD, FreeBSD, NetBSD, OpenBSD, macOS, AIX, IRIX, Solaris, Cygwin, Haiku */
  88 # define HAVE_STACK_OVERFLOW_RECOVERY 1
  89 #endif
  90 
  91 
  92 #ifdef __cplusplus
  93 extern "C" {
  94 #endif
  95 
  96 #define LIBSIGSEGV_VERSION 0x020D    /* version number: (major<<8) + minor */
  97 extern int libsigsegv_version;       /* Likewise */
  98 
  99 /* -------------------------------------------------------------------------- */
 100 
 101 #if 1 /* really only HAVE_SIGSEGV_RECOVERY */
 102 
 103 /*
 104  * The mask of bits that are set to zero in a fault address that gets passed
 105  * to a global SIGSEGV handler.
 106  * On some platforms, the precise fault address is not known, only the memory
 107  * page into which the fault address falls. This is apparently allowed by POSIX:
 108  * <http://pubs.opengroup.org/onlinepubs/9699919799/basedefs/signal.h.html>
 109  * says: "For some implementations, the value of si_addr may be inaccurate."
 110  * In this case, the returned fault address is rounded down to a multiple of
 111  * getpagesize() = sysconf(_SC_PAGESIZE).
 112  * On such platforms, we define SIGSEGV_FAULT_ADDRESS_ALIGNMENT to be an upper
 113  * bound for getpagesize() (and, like getpagesize(), also a power of 2).
 114  * On the platforms where the returned fault address is the precise one, we
 115  * define SIGSEGV_FAULT_ADDRESS_ALIGNMENT to 1.
 116  */
 117 # if defined __NetBSD__ && (defined __sparc__ || defined __sparc64__)
 118   /* getpagesize () is 0x1000 or 0x2000, depending on hardware.  */
 119 #  define SIGSEGV_FAULT_ADDRESS_ALIGNMENT 0x2000UL
 120 # elif defined __linux__ && (defined __s390__ || defined __s390x__)
 121   /* getpagesize () is 0x1000.  */
 122 #  define SIGSEGV_FAULT_ADDRESS_ALIGNMENT 0x1000UL
 123 # else
 124 #  define SIGSEGV_FAULT_ADDRESS_ALIGNMENT 1UL
 125 # endif
 126 
 127 /*
 128  * The type of a global SIGSEGV handler.
 129  * The fault address, with the bits (SIGSEGV_FAULT_ADDRESS_ALIGNMENT - 1)
 130  * cleared, is passed as argument.
 131  * The access type (read access or write access) is not passed; your handler
 132  * has to know itself how to distinguish these two cases.
 133  * The second argument is 0, meaning it could also be a stack overflow, or 1,
 134  * meaning the handler should seriously try to fix the fault.
 135  * The return value should be nonzero if the handler has done its job
 136  * and no other handler should be called, or 0 if the handler declines
 137  * responsibility for the given address.
 138  *
 139  * The handler is run at a moment when nothing about the global state of the
 140  * program is known. Therefore it cannot use facilities that manipulate global
 141  * variables or locks. In particular, it cannot use malloc(); use mmap()
 142  * instead. It cannot use fopen(); use open() instead. Etc. All global
 143  * variables that are accessed by the handler should be marked 'volatile'.
 144  */
 145 typedef int (*sigsegv_handler_t) (void* fault_address, int serious);
 146 
 147 /*
 148  * Installs a global SIGSEGV handler.
 149  * This should be called once only, and it ignores any previously installed
 150  * SIGSEGV handler.
 151  * Returns 0 on success, or -1 if the system doesn't support catching SIGSEGV.
 152  */
 153 extern int sigsegv_install_handler (sigsegv_handler_t handler);
 154 
 155 /*
 156  * Deinstalls the global SIGSEGV handler.
 157  * This goes back to the state where no SIGSEGV handler is installed.
 158  */
 159 extern void sigsegv_deinstall_handler (void);
 160 
 161 /*
 162  * Prepares leaving a SIGSEGV handler (through longjmp or similar means).
 163  * Control is transferred by calling CONTINUATION with CONT_ARG1, CONT_ARG2,
 164  * CONT_ARG3 as arguments.
 165  * CONTINUATION must not return.
 166  * The sigsegv_leave_handler function may return if called from a SIGSEGV
 167  * handler; its return value should be used as the handler's return value.
 168  * The sigsegv_leave_handler function does not return if called from a
 169  * stack overflow handler.
 170  */
 171 extern int sigsegv_leave_handler (void (*continuation) (void*, void*, void*), void* cont_arg1, void* cont_arg2, void* cont_arg3);
 172 
 173 #endif /* HAVE_SIGSEGV_RECOVERY */
 174 
 175 #if 1 /* really only HAVE_STACK_OVERFLOW_RECOVERY */
 176 
 177 /*
 178  * The type of a context passed to a stack overflow handler.
 179  * This type is system dependent; on some platforms it is an 'ucontext_t *',
 180  * on some platforms it is a 'struct sigcontext *', on others merely an
 181  * opaque 'void *'.
 182  */
 183 # if defined __linux__ || defined __ANDROID__ \
 184      || (defined __FreeBSD__ && (defined __arm__ || defined __armhf__ || defined __arm64__)) \
 185      || defined __NetBSD__ \
 186      || (defined __APPLE__ && defined __MACH__) \
 187      || defined _AIX || defined __sun \
 188      || defined __CYGWIN__ || defined __HAIKU__
 189 typedef ucontext_t *stackoverflow_context_t;
 190 # elif defined __GNU__ \
 191        || defined __FreeBSD_kernel__ || (defined __FreeBSD__ && !(defined __sparc__ || defined __sparc64__)) || defined __DragonFly__ \
 192        || defined __OpenBSD__ || defined __sgi
 193 typedef struct sigcontext *stackoverflow_context_t;
 194 # else
 195 typedef void *stackoverflow_context_t;
 196 # endif
 197 
 198 /*
 199  * The type of a stack overflow handler.
 200  * Such a handler should perform a longjmp call in order to reduce the amount
 201  * of stack needed. It must not return.
 202  * The emergency argument is 0 when the stack could be repared, or 1 if the
 203  * application should better save its state and exit now.
 204  *
 205  * The handler is run at a moment when nothing about the global state of the
 206  * program is known. Therefore it cannot use facilities that manipulate global
 207  * variables or locks. In particular, it cannot use malloc(); use mmap()
 208  * instead. It cannot use fopen(); use open() instead. Etc. All global
 209  * variables that are accessed by the handler should be marked 'volatile'.
 210  */
 211 typedef void (*stackoverflow_handler_t) (int emergency, stackoverflow_context_t scp);
 212 
 213 /*
 214  * Installs a stack overflow handler.
 215  * The extra_stack argument is a pointer to a pre-allocated area used as a
 216  * stack for executing the handler. It typically comes from a static variable
 217  * or from heap-allocated memoty; placing it on the main stack may fail on
 218  * some operating systems.
 219  * Its size, passed in extra_stack_size, should be sufficiently large.  The
 220  * following code determines an appropriate size:
 221  *   #include <signal.h>
 222  *   #ifndef SIGSTKSZ         / * glibc defines SIGSTKSZ for this purpose * /
 223  *   # define SIGSTKSZ 16384  / * on most platforms, 16 KB are sufficient * /
 224  *   #endif
 225  * Returns 0 on success, or -1 if the system doesn't support catching stack
 226  * overflow.
 227  */
 228 extern int stackoverflow_install_handler (stackoverflow_handler_t handler,
 229                                           void* extra_stack, size_t extra_stack_size);
 230 
 231 /*
 232  * Deinstalls the stack overflow handler.
 233  */
 234 extern void stackoverflow_deinstall_handler (void);
 235 
 236 #endif /* HAVE_STACK_OVERFLOW_RECOVERY */
 237 
 238 /* -------------------------------------------------------------------------- */
 239 
 240 #ifdef __cplusplus
 241 }
 242 #endif
 243 
 244 #endif /* _SIGSEGV_H */

/* [previous][next][first][last][top][bottom][index][help] */