| 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 */
|
|---|