| 1 | // dwarf2-signal.h - Catch runtime signals and turn them into exceptions.
|
|---|
| 2 |
|
|---|
| 3 | /* Copyright (C) 2000, 2001 Free Software Foundation
|
|---|
| 4 |
|
|---|
| 5 | This file is part of libgcj.
|
|---|
| 6 |
|
|---|
| 7 | Use this file for a target for which the dwarf2 unwinder in libgcc
|
|---|
| 8 | can unwind through signal handlers.
|
|---|
| 9 |
|
|---|
| 10 | This software is copyrighted work licensed under the terms of the
|
|---|
| 11 | Libgcj License. Please consult the file "LIBGCJ_LICENSE" for
|
|---|
| 12 | details. */
|
|---|
| 13 |
|
|---|
| 14 | #ifndef JAVA_SIGNAL_H
|
|---|
| 15 | #define JAVA_SIGNAL_H 1
|
|---|
| 16 |
|
|---|
| 17 | #include <signal.h>
|
|---|
| 18 | #include <sys/syscall.h>
|
|---|
| 19 |
|
|---|
| 20 | #define HANDLE_SEGV 1
|
|---|
| 21 | #undef HANDLE_FPE
|
|---|
| 22 |
|
|---|
| 23 | #define SIGNAL_HANDLER(_name) \
|
|---|
| 24 | static void _Jv_##_name (int, siginfo_t *_sip, void *_p)
|
|---|
| 25 |
|
|---|
| 26 | class java::lang::Throwable;
|
|---|
| 27 |
|
|---|
| 28 | // Unwind the stack to the point at which the signal was generated and
|
|---|
| 29 | // then throw an exception. With the dwarf2 unwinder we don't usually
|
|---|
| 30 | // need to do anything, with some minor exceptions.
|
|---|
| 31 |
|
|---|
| 32 | #ifdef __alpha__
|
|---|
| 33 | #define MAKE_THROW_FRAME(_exception) \
|
|---|
| 34 | do \
|
|---|
| 35 | { \
|
|---|
| 36 | /* Alpha either leaves PC pointing at a faulting instruction or the \
|
|---|
| 37 | following instruction, depending on the signal. SEGV always does \
|
|---|
| 38 | the former, so we adjust the saved PC to point to the following \
|
|---|
| 39 | instruction; this is what the handler in libgcc expects. */ \
|
|---|
| 40 | struct sigcontext *_sc = (struct sigcontext *)_p; \
|
|---|
| 41 | _sc->sc_pc += 4; \
|
|---|
| 42 | } \
|
|---|
| 43 | while (0)
|
|---|
| 44 |
|
|---|
| 45 | #elif defined(__ia64__)
|
|---|
| 46 |
|
|---|
| 47 | #define MAKE_THROW_FRAME(_exception) \
|
|---|
| 48 | do \
|
|---|
| 49 | { \
|
|---|
| 50 | /* IA-64 either leaves PC pointing at a faulting instruction or the \
|
|---|
| 51 | following instruction, depending on the signal. SEGV always does \
|
|---|
| 52 | the former, so we adjust the saved PC to point to the following \
|
|---|
| 53 | instruction; this is what the handler in libgcc expects. */ \
|
|---|
| 54 | /* Note that we are lying to the unwinder here, which expects the \
|
|---|
| 55 | faulting pc, not pc+1. But we claim the unwind information can't \
|
|---|
| 56 | be changed by such a ld or st instruction, so it doesn't matter. */ \
|
|---|
| 57 | struct sigcontext *_sc = (struct sigcontext *)_p; \
|
|---|
| 58 | _sc->sc_ip++; \
|
|---|
| 59 | } \
|
|---|
| 60 | while (0)
|
|---|
| 61 | #elif defined(__sparc__)
|
|---|
| 62 | /* We could do the unwind of the signal frame quickly by hand here like
|
|---|
| 63 | sparc-signal.h does under Solaris, but that makes debugging unwind
|
|---|
| 64 | failures almost impossible. */
|
|---|
| 65 | #if !defined(__arch64__)
|
|---|
| 66 | #define MAKE_THROW_FRAME(_exception) \
|
|---|
| 67 | do \
|
|---|
| 68 | { \
|
|---|
| 69 | /* Sparc-32 leaves PC pointing at a faulting instruction \
|
|---|
| 70 | always. \
|
|---|
| 71 | We advance the PC one instruction past the exception causing PC. \
|
|---|
| 72 | This is done because FDEs are found with "context->ra - 1" in the \
|
|---|
| 73 | unwinder. \
|
|---|
| 74 | Also, the dwarf2 unwind machinery is going to add 8 to the \
|
|---|
| 75 | PC it uses on Sparc. So we adjust the PC here. We do it here \
|
|---|
| 76 | because we run once for such an exception, however the Sparc specific\
|
|---|
| 77 | unwind can run multiple times for the same exception and it would \
|
|---|
| 78 | adjust the PC more than once resulting in a bogus value. */ \
|
|---|
| 79 | struct sig_regs { \
|
|---|
| 80 | unsigned int psr, pc, npc, y, u_regs[16]; \
|
|---|
| 81 | } *regp; \
|
|---|
| 82 | unsigned int insn; \
|
|---|
| 83 | __asm__ __volatile__("ld [%%i7 + 8], %0" : "=r" (insn)); \
|
|---|
| 84 | /* mov __NR_sigaction, %g1; Old signal stack layout */ \
|
|---|
| 85 | if (insn == 0x821020d8) \
|
|---|
| 86 | regp = (struct sig_regs *) _sip; \
|
|---|
| 87 | else \
|
|---|
| 88 | /* mov __NR_rt_sigaction, %g1; New signal stack layout */ \
|
|---|
| 89 | regp = (struct sig_regs *) (_sip + 1); \
|
|---|
| 90 | regp->pc = ((regp->pc + 4) - 8); \
|
|---|
| 91 | } \
|
|---|
| 92 | while (0)
|
|---|
| 93 | #else
|
|---|
| 94 | #define MAKE_THROW_FRAME(_exception) \
|
|---|
| 95 | do \
|
|---|
| 96 | { \
|
|---|
| 97 | /* Sparc-64 leaves PC pointing at a faulting instruction \
|
|---|
| 98 | always. \
|
|---|
| 99 | We advance the PC one instruction past the exception causing PC. \
|
|---|
| 100 | This is done because FDEs are found with "context->ra - 1" in the \
|
|---|
| 101 | unwinder. \
|
|---|
| 102 | Also, the dwarf2 unwind machinery is going to add 8 to the \
|
|---|
| 103 | PC it uses on Sparc. So we adjust the PC here. We do it here \
|
|---|
| 104 | because we run once for such an exception, however the Sparc specific\
|
|---|
| 105 | unwind can run multiple times for the same exception and it would \
|
|---|
| 106 | adjust the PC more than once resulting in a bogus value. */ \
|
|---|
| 107 | struct pt_regs { \
|
|---|
| 108 | unsigned long u_regs[16]; \
|
|---|
| 109 | unsigned long tstate, tpc, tnpc; \
|
|---|
| 110 | unsigned int y, fprs; \
|
|---|
| 111 | } *regp = (struct pt_regs *) (_sip + 1); \
|
|---|
| 112 | regp->tpc = ((regp->tpc + 4) - 8); \
|
|---|
| 113 | } \
|
|---|
| 114 | while (0)
|
|---|
| 115 | #endif
|
|---|
| 116 | #else
|
|---|
| 117 | #define MAKE_THROW_FRAME(_exception) \
|
|---|
| 118 | do \
|
|---|
| 119 | { \
|
|---|
| 120 | (void)_p; \
|
|---|
| 121 | } \
|
|---|
| 122 | while (0)
|
|---|
| 123 | #endif
|
|---|
| 124 |
|
|---|
| 125 | #if defined(__sparc__)
|
|---|
| 126 | #if defined(__arch64__)
|
|---|
| 127 | extern "C" {
|
|---|
| 128 | static void __rt_sigreturn_stub(void)
|
|---|
| 129 | {
|
|---|
| 130 | __asm__("mov %0, %%g1\n\t"
|
|---|
| 131 | "ta 0x6d\n\t"
|
|---|
| 132 | : /* no outputs */
|
|---|
| 133 | : "i" (__NR_rt_sigreturn));
|
|---|
| 134 | }
|
|---|
| 135 | struct kernel_sigaction
|
|---|
| 136 | {
|
|---|
| 137 | void (*k_sa_sigaction)(int,siginfo_t *,void *);
|
|---|
| 138 | unsigned long k_sa_flags;
|
|---|
| 139 | void (*k_sa_restorer)(void);
|
|---|
| 140 | sigset_t k_sa_mask;
|
|---|
| 141 | };
|
|---|
| 142 | }
|
|---|
| 143 | #define INIT_SEGV \
|
|---|
| 144 | do \
|
|---|
| 145 | { \
|
|---|
| 146 | nullp = new java::lang::NullPointerException (); \
|
|---|
| 147 | struct kernel_sigaction act; \
|
|---|
| 148 | unsigned long stub = ((unsigned long)&__rt_sigreturn_stub); \
|
|---|
| 149 | act.k_sa_sigaction = _Jv_catch_segv; \
|
|---|
| 150 | sigemptyset (&act.k_sa_mask); \
|
|---|
| 151 | act.k_sa_flags = SA_SIGINFO; \
|
|---|
| 152 | act.k_sa_restorer = NULL; \
|
|---|
| 153 | syscall (SYS_rt_sigaction, SIGSEGV, &act, NULL, \
|
|---|
| 154 | stub - 8, _NSIG / 8); \
|
|---|
| 155 | } \
|
|---|
| 156 | while (0)
|
|---|
| 157 |
|
|---|
| 158 | #define INIT_FPE \
|
|---|
| 159 | do \
|
|---|
| 160 | { \
|
|---|
| 161 | arithexception = new java::lang::ArithmeticException \
|
|---|
| 162 | (JvNewStringLatin1 ("/ by zero")); \
|
|---|
| 163 | struct kernel_sigaction act; \
|
|---|
| 164 | unsigned long stub = ((unsigned long)&__rt_sigreturn_stub); \
|
|---|
| 165 | act.k_sa_sigaction = _Jv_catch_fpe; \
|
|---|
| 166 | sigemptyset (&act.k_sa_mask); \
|
|---|
| 167 | act.k_sa_flags = SA_SIGINFO; \
|
|---|
| 168 | act.k_sa_restorer = NULL; \
|
|---|
| 169 | syscall (SYS_rt_sigaction, SIGFPE, &act, NULL, \
|
|---|
| 170 | stub - 8, _NSIG / 8); \
|
|---|
| 171 | } \
|
|---|
| 172 | while (0)
|
|---|
| 173 | #else /* __arch64__ */
|
|---|
| 174 |
|
|---|
| 175 | extern "C" {
|
|---|
| 176 | struct kernel_sigaction
|
|---|
| 177 | {
|
|---|
| 178 | void (*k_sa_sigaction)(int,siginfo_t *,void *);
|
|---|
| 179 | unsigned long k_sa_mask, k_sa_flags;
|
|---|
| 180 | void (*k_sa_restorer)(void);
|
|---|
| 181 | };
|
|---|
| 182 | }
|
|---|
| 183 |
|
|---|
| 184 | #define INIT_SEGV \
|
|---|
| 185 | do \
|
|---|
| 186 | { \
|
|---|
| 187 | struct kernel_sigaction act; \
|
|---|
| 188 | nullp = new java::lang::NullPointerException (); \
|
|---|
| 189 | act.k_sa_sigaction = _Jv_catch_segv; \
|
|---|
| 190 | act.k_sa_mask = 0; \
|
|---|
| 191 | act.k_sa_flags = SA_SIGINFO; \
|
|---|
| 192 | act.k_sa_restorer = NULL; \
|
|---|
| 193 | syscall (SYS_sigaction, -SIGSEGV, &act, NULL); \
|
|---|
| 194 | } \
|
|---|
| 195 | while (0)
|
|---|
| 196 |
|
|---|
| 197 | #define INIT_FPE \
|
|---|
| 198 | do \
|
|---|
| 199 | { \
|
|---|
| 200 | arithexception = new java::lang::ArithmeticException \
|
|---|
| 201 | (JvNewStringLatin1 ("/ by zero")); \
|
|---|
| 202 | struct kernel_sigaction act; \
|
|---|
| 203 | act.k_sa_sigaction = _Jv_catch_fpe; \
|
|---|
| 204 | act.k_sa_mask = 0; \
|
|---|
| 205 | act.k_sa_flags = SA_SIGINFO; \
|
|---|
| 206 | act.k_sa_restorer = NULL; \
|
|---|
| 207 | syscall (SYS_sigaction, -SIGFPE, &act, NULL); \
|
|---|
| 208 | } \
|
|---|
| 209 | while (0)
|
|---|
| 210 | #endif
|
|---|
| 211 | #elif !defined(__ia64__)
|
|---|
| 212 | #define INIT_SEGV \
|
|---|
| 213 | do \
|
|---|
| 214 | { \
|
|---|
| 215 | nullp = new java::lang::NullPointerException (); \
|
|---|
| 216 | struct sigaction act; \
|
|---|
| 217 | act.sa_sigaction = _Jv_catch_segv; \
|
|---|
| 218 | sigemptyset (&act.sa_mask); \
|
|---|
| 219 | act.sa_flags = SA_SIGINFO; \
|
|---|
| 220 | syscall (SYS_sigaction, SIGSEGV, &act, NULL); \
|
|---|
| 221 | } \
|
|---|
| 222 | while (0)
|
|---|
| 223 |
|
|---|
| 224 | #define INIT_FPE \
|
|---|
| 225 | do \
|
|---|
| 226 | { \
|
|---|
| 227 | arithexception = new java::lang::ArithmeticException \
|
|---|
| 228 | (JvNewStringLatin1 ("/ by zero")); \
|
|---|
| 229 | struct sigaction act; \
|
|---|
| 230 | act.sa_sigaction = _Jv_catch_fpe; \
|
|---|
| 231 | sigemptyset (&act.sa_mask); \
|
|---|
| 232 | act.sa_flags = SA_SIGINFO; \
|
|---|
| 233 | syscall (SYS_sigaction, SIGFPE, &act, NULL); \
|
|---|
| 234 | } \
|
|---|
| 235 | while (0)
|
|---|
| 236 |
|
|---|
| 237 | /* We use syscall(SYS_sigaction) in INIT_SEGV and INIT_FPE instead of
|
|---|
| 238 | * sigaction() because on some systems the pthreads wrappers for
|
|---|
| 239 | * signal handlers are not compiled with unwind information, so it's
|
|---|
| 240 | * not possible to unwind through them. This is a problem that will
|
|---|
| 241 | * go away once all systems have pthreads libraries that are
|
|---|
| 242 | * compiled with full unwind info. */
|
|---|
| 243 |
|
|---|
| 244 | #else /* __ia64__ */
|
|---|
| 245 |
|
|---|
| 246 | // On IA64, unwind information is mandatory, so we can unwind
|
|---|
| 247 | // correctly through glibc frames. Thus we call the ordinary
|
|---|
| 248 | // sigaction.
|
|---|
| 249 |
|
|---|
| 250 | #define INIT_SEGV \
|
|---|
| 251 | do \
|
|---|
| 252 | { \
|
|---|
| 253 | nullp = new java::lang::NullPointerException (); \
|
|---|
| 254 | struct sigaction act; \
|
|---|
| 255 | act.sa_sigaction = _Jv_catch_segv; \
|
|---|
| 256 | sigemptyset (&act.sa_mask); \
|
|---|
| 257 | act.sa_flags = SA_SIGINFO; \
|
|---|
| 258 | sigaction (SIGSEGV, &act, NULL); \
|
|---|
| 259 | } \
|
|---|
| 260 | while (0)
|
|---|
| 261 |
|
|---|
| 262 | #define INIT_FPE \
|
|---|
| 263 | do \
|
|---|
| 264 | { \
|
|---|
| 265 | arithexception = new java::lang::ArithmeticException \
|
|---|
| 266 | (JvNewStringLatin1 ("/ by zero")); \
|
|---|
| 267 | struct sigaction act; \
|
|---|
| 268 | act.sa_sigaction = _Jv_catch_fpe; \
|
|---|
| 269 | sigemptyset (&act.sa_mask); \
|
|---|
| 270 | act.sa_flags = SA_SIGINFO; \
|
|---|
| 271 | sigaction (SIGFPE, &act, NULL); \
|
|---|
| 272 | } \
|
|---|
| 273 | while (0)
|
|---|
| 274 | #endif /* __ia64__ || __sparc__ */
|
|---|
| 275 | #endif /* JAVA_SIGNAL_H */
|
|---|