/*-*- mode:c;indent-tabs-mode:nil;c-basic-offset:2;tab-width:8;coding:utf-8 -*-│ │vi: set net ft=c ts=2 sts=2 sw=2 fenc=utf-8 :vi│ ╞══════════════════════════════════════════════════════════════════════════════╡ │ Copyright 2020 Justine Alexandra Roberts Tunney │ │ │ │ Permission to use, copy, modify, and/or distribute this software for │ │ any purpose with or without fee is hereby granted, provided that the │ │ above copyright notice and this permission notice appear in all copies. │ │ │ │ THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL │ │ WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED │ │ WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE │ │ AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL │ │ DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR │ │ PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER │ │ TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR │ │ PERFORMANCE OF THIS SOFTWARE. │ ╚─────────────────────────────────────────────────────────────────────────────*/ #include "libc/calls/internal.h" #include "libc/calls/strace.internal.h" #include "libc/calls/typedef/sigaction_f.h" #include "libc/calls/ucontext.h" #include "libc/nt/enum/exceptionhandleractions.h" #include "libc/nt/enum/signal.h" #include "libc/nt/struct/ntexceptionpointers.h" #include "libc/str/str.h" #include "libc/sysv/consts/sicode.h" textwindows unsigned __wincrash(struct NtExceptionPointers *ep) { int sig, rva, code; struct Goodies { ucontext_t ctx; struct siginfo si; } g; STRACE("__wincrash"); switch (ep->ExceptionRecord->ExceptionCode) { case kNtSignalBreakpoint: code = TRAP_BRKPT; sig = SIGTRAP; break; case kNtSignalIllegalInstruction: code = ILL_ILLOPC; sig = SIGILL; break; case kNtSignalPrivInstruction: code = ILL_PRVOPC; sig = SIGILL; break; case kNtSignalGuardPage: case kNtSignalInPageError: code = SEGV_MAPERR; sig = SIGSEGV; break; case kNtSignalAccessViolation: code = SEGV_ACCERR; sig = SIGSEGV; break; case kNtSignalInvalidHandle: case kNtSignalInvalidParameter: case kNtSignalAssertionFailure: code = SI_USER; sig = SIGABRT; break; case kNtSignalFltDivideByZero: code = FPE_FLTDIV; sig = SIGFPE; break; case kNtSignalFltOverflow: code = FPE_FLTOVF; sig = SIGFPE; break; case kNtSignalFltUnderflow: code = FPE_FLTUND; sig = SIGFPE; break; case kNtSignalFltInexactResult: code = FPE_FLTRES; sig = SIGFPE; break; case kNtSignalFltDenormalOperand: case kNtSignalFltInvalidOperation: case kNtSignalFltStackCheck: case kNtSignalIntegerDivideByZero: case kNtSignalFloatMultipleFaults: case kNtSignalFloatMultipleTraps: code = FPE_FLTINV; sig = SIGFPE; break; case kNtSignalDllNotFound: case kNtSignalOrdinalNotFound: case kNtSignalEntrypointNotFound: case kNtSignalDllInitFailed: code = SI_KERNEL; sig = SIGSYS; break; default: return kNtExceptionContinueSearch; } bzero(&g, sizeof(g)); g.si.si_code = code; rva = __sighandrvas[sig]; if (rva >= kSigactionMinRva) { ntcontext2linux(&g.ctx, ep->ContextRecord); ((sigaction_f)(_base + rva))(sig, &g.si, &g.ctx); } return kNtExceptionContinueExecution; }