// Copyright 2011 The Go Authors. All rights reserved. // Use of this source code is governed by a BSD-style // license that can be found in the LICENSE file. package runtime import ( "internal/abi" "internal/runtime/sys" "unsafe" ) const ( _SEM_FAILCRITICALERRORS = 0x0001 _SEM_NOGPFAULTERRORBOX = 0x0002 _SEM_NOOPENFILEERRORBOX = 0x8000 _WER_FAULT_REPORTING_NO_UI = 0x0020 ) func preventErrorDialogs() { errormode := stdcall0(_GetErrorMode) stdcall1(_SetErrorMode, errormode|_SEM_FAILCRITICALERRORS|_SEM_NOGPFAULTERRORBOX|_SEM_NOOPENFILEERRORBOX) // Disable WER fault reporting UI. // Do this even if WER is disabled as a whole, // as WER might be enabled later with setTraceback("wer") // and we still want the fault reporting UI to be disabled if this happens. var werflags uintptr stdcall2(_WerGetFlags, currentProcess, uintptr(unsafe.Pointer(&werflags))) stdcall1(_WerSetFlags, werflags|_WER_FAULT_REPORTING_NO_UI) } // enableWER re-enables Windows error reporting without fault reporting UI. func enableWER() { // re-enable Windows Error Reporting errormode := stdcall0(_GetErrorMode) if errormode&_SEM_NOGPFAULTERRORBOX != 0 { stdcall1(_SetErrorMode, errormode^_SEM_NOGPFAULTERRORBOX) } } // in sys_windows_386.s, sys_windows_amd64.s, sys_windows_arm.s, and sys_windows_arm64.s func exceptiontramp() func firstcontinuetramp() func lastcontinuetramp() func sehtramp() func sigresume() func initExceptionHandler() { stdcall2(_AddVectoredExceptionHandler, 1, abi.FuncPCABI0(exceptiontramp)) if GOARCH == "386" { // use SetUnhandledExceptionFilter for windows-386. // note: SetUnhandledExceptionFilter handler won't be called, if debugging. stdcall1(_SetUnhandledExceptionFilter, abi.FuncPCABI0(lastcontinuetramp)) } else { stdcall2(_AddVectoredContinueHandler, 1, abi.FuncPCABI0(firstcontinuetramp)) stdcall2(_AddVectoredContinueHandler, 0, abi.FuncPCABI0(lastcontinuetramp)) } } // isAbort returns true, if context r describes exception raised // by calling runtime.abort function. // //go:nosplit func isAbort(r *context) bool { pc := r.ip() if GOARCH == "386" || GOARCH == "amd64" || GOARCH == "arm" { // In the case of an abort, the exception IP is one byte after // the INT3 (this differs from UNIX OSes). Note that on ARM, // this means that the exception IP is no longer aligned. pc-- } return isAbortPC(pc) } // isgoexception reports whether this exception should be translated // into a Go panic or throw. // // It is nosplit to avoid growing the stack in case we're aborting // because of a stack overflow. // //go:nosplit func isgoexception(info *exceptionrecord, r *context) bool { // Only handle exception if executing instructions in Go binary // (not Windows library code). // TODO(mwhudson): needs to loop to support shared libs if r.ip() < firstmoduledata.text || firstmoduledata.etext < r.ip() { return false } // Go will only handle some exceptions. switch info.exceptioncode { default: return false case _EXCEPTION_ACCESS_VIOLATION: case _EXCEPTION_IN_PAGE_ERROR: case _EXCEPTION_INT_DIVIDE_BY_ZERO: case _EXCEPTION_INT_OVERFLOW: case _EXCEPTION_FLT_DENORMAL_OPERAND: case _EXCEPTION_FLT_DIVIDE_BY_ZERO: case _EXCEPTION_FLT_INEXACT_RESULT: case _EXCEPTION_FLT_OVERFLOW: case _EXCEPTION_FLT_UNDERFLOW: case _EXCEPTION_BREAKPOINT: case _EXCEPTION_ILLEGAL_INSTRUCTION: // breakpoint arrives this way on arm64 } return true } const ( callbackVEH = iota callbackFirstVCH callbackLastVCH ) // sigFetchGSafe is like getg() but without panicking // when TLS is not set. // Only implemented on windows/386, which is the only // arch that loads TLS when calling getg(). Others // use a dedicated register. func sigFetchGSafe() *g func sigFetchG() *g { if GOARCH == "386" { return sigFetchGSafe() } return getg() } // sigtrampgo is called from the exception handler function, sigtramp, // written in assembly code. // Return EXCEPTION_CONTINUE_EXECUTION if the exception is handled, // else return EXCEPTION_CONTINUE_SEARCH. // // It is nosplit for the same reason as exceptionhandler. // //go:nosplit func sigtrampgo(ep *exceptionpointers, kind int) int32 { gp := sigFetchG() if gp == nil { return _EXCEPTION_CONTINUE_SEARCH } var fn func(info *exceptionrecord, r *context, gp *g) int32 switch kind { case callbackVEH: fn = exceptionhandler case callbackFirstVCH: fn = firstcontinuehandler case callbackLastVCH: fn = lastcontinuehandler default: throw("unknown sigtramp callback") } // Check if we are running on g0 stack, and if we are, // call fn directly instead of creating the closure. // for the systemstack argument. // // A closure can't be marked as nosplit, so it might // call morestack if we are at the g0 stack limit. // If that happens, the runtime will call abort // and end up in sigtrampgo again. // TODO: revisit this workaround if/when closures // can be compiled as nosplit. // // Note that this scenario should only occur on // TestG0StackOverflow. Any other occurrence should // be treated as a bug. var ret int32 if gp != gp.m.g0 { systemstack(func() { ret = fn(ep.record, ep.context, gp) }) } else { ret = fn(ep.record, ep.context, gp) } if ret == _EXCEPTION_CONTINUE_SEARCH { return ret } // Check if we need to set up the control flow guard workaround. // On Windows, the stack pointer in the context must lie within // system stack limits when we resume from exception. // Store the resume SP and PC in alternate registers // and return to sigresume on the g0 stack. // sigresume makes no use of the stack at all, // loading SP from RX and jumping to RY, being RX and RY two scratch registers. // Note that blindly smashing RX and RY is only safe because we know sigpanic // will not actually return to the original frame, so the registers // are effectively dead. But this does mean we can't use the // same mechanism for async preemption. if ep.context.ip() == abi.FuncPCABI0(sigresume) { // sigresume has already been set up by a previous exception. return ret } prepareContextForSigResume(ep.context) ep.context.set_sp(gp.m.g0.sched.sp) ep.context.set_ip(abi.FuncPCABI0(sigresume)) return ret } // Called by sigtramp from Windows VEH handler. // Return value signals whether the exception has been handled (EXCEPTION_CONTINUE_EXECUTION) // or should be made available to other handlers in the chain (EXCEPTION_CONTINUE_SEARCH). // // This is nosplit to avoid growing the stack until we've checked for // _EXCEPTION_BREAKPOINT, which is raised by abort() if we overflow the g0 stack. // //go:nosplit func exceptionhandler(info *exceptionrecord, r *context, gp *g) int32 { if !isgoexception(info, r) { return _EXCEPTION_CONTINUE_SEARCH } if gp.throwsplit || isAbort(r) { // We can't safely sigpanic because it may grow the stack. // Or this is a call to abort. // Don't go through any more of the Windows handler chain. // Crash now. winthrow(info, r, gp) } // After this point, it is safe to grow the stack. // Make it look like a call to the signal func. // Have to pass arguments out of band since // augmenting the stack frame would break // the unwinding code. gp.sig = info.exceptioncode gp.sigcode0 = info.exceptioninformation[0] gp.sigcode1 = info.exceptioninformation[1] gp.sigpc = r.ip() // Only push runtime·sigpanic if r.ip() != 0. // If r.ip() == 0, probably panicked because of a // call to a nil func. Not pushing that onto sp will // make the trace look like a call to runtime·sigpanic instead. // (Otherwise the trace will end at runtime·sigpanic and we // won't get to see who faulted.) // Also don't push a sigpanic frame if the faulting PC // is the entry of asyncPreempt. In this case, we suspended // the thread right between the fault and the exception handler // starting to run, and we have pushed an asyncPreempt call. // The exception is not from asyncPreempt, so not to push a // sigpanic call to make it look like that. Instead, just // overwrite the PC. (See issue #35773) if r.ip() != 0 && r.ip() != abi.FuncPCABI0(asyncPreempt) { sp := unsafe.Pointer(r.sp()) delta := uintptr(sys.StackAlign) sp = add(sp, -delta) r.set_sp(uintptr(sp)) if usesLR { *((*uintptr)(sp)) = r.lr() r.set_lr(r.ip()) } else { *((*uintptr)(sp)) = r.ip() } } r.set_ip(abi.FuncPCABI0(sigpanic0)) return _EXCEPTION_CONTINUE_EXECUTION } // sehhandler is reached as part of the SEH chain. // // It is nosplit for the same reason as exceptionhandler. // //go:nosplit func sehhandler(_ *exceptionrecord, _ uint64, _ *context, dctxt *_DISPATCHER_CONTEXT) int32 { g0 := getg() if g0 == nil || g0.m.curg == nil { // No g available, nothing to do here. return _EXCEPTION_CONTINUE_SEARCH_SEH } // The Windows SEH machinery will unwind the stack until it finds // a frame with a handler for the exception or until the frame is // outside the stack boundaries, in which case it will call the // UnhandledExceptionFilter. Unfortunately, it doesn't know about // the goroutine stack, so it will stop unwinding when it reaches the // first frame not running in g0. As a result, neither non-Go exceptions // handlers higher up the stack nor UnhandledExceptionFilter will be called. // // To work around this, manually unwind the stack until the top of the goroutine // stack is reached, and then pass the control back to Windows. gp := g0.m.curg ctxt := dctxt.ctx() var base, sp uintptr for { entry := stdcall3(_RtlLookupFunctionEntry, ctxt.ip(), uintptr(unsafe.Pointer(&base)), 0) if entry == 0 { break } stdcall8(_RtlVirtualUnwind, 0, base, ctxt.ip(), entry, uintptr(unsafe.Pointer(ctxt)), 0, uintptr(unsafe.Pointer(&sp)), 0) if sp < gp.stack.lo || gp.stack.hi <= sp { break } } return _EXCEPTION_CONTINUE_SEARCH_SEH } // It seems Windows searches ContinueHandler's list even // if ExceptionHandler returns EXCEPTION_CONTINUE_EXECUTION. // firstcontinuehandler will stop that search, // if exceptionhandler did the same earlier. // // It is nosplit for the same reason as exceptionhandler. // //go:nosplit func firstcontinuehandler(info *exceptionrecord, r *context, gp *g) int32 { if !isgoexception(info, r) { return _EXCEPTION_CONTINUE_SEARCH } return _EXCEPTION_CONTINUE_EXECUTION } // lastcontinuehandler is reached, because runtime cannot handle // current exception. lastcontinuehandler will print crash info and exit. // // It is nosplit for the same reason as exceptionhandler. // //go:nosplit func lastcontinuehandler(info *exceptionrecord, r *context, gp *g) int32 { if islibrary || isarchive { // Go DLL/archive has been loaded in a non-go program. // If the exception does not originate from go, the go runtime // should not take responsibility of crashing the process. return _EXCEPTION_CONTINUE_SEARCH } // VEH is called before SEH, but arm64 MSVC DLLs use SEH to trap // illegal instructions during runtime initialization to determine // CPU features, so if we make it to the last handler and we're // arm64 and it's an illegal instruction and this is coming from // non-Go code, then assume it's this runtime probing happen, and // pass that onward to SEH. if GOARCH == "arm64" && info.exceptioncode == _EXCEPTION_ILLEGAL_INSTRUCTION && (r.ip() < firstmoduledata.text || firstmoduledata.etext < r.ip()) { return _EXCEPTION_CONTINUE_SEARCH } winthrow(info, r, gp) return 0 // not reached } // Always called on g0. gp is the G where the exception occurred. // //go:nosplit func winthrow(info *exceptionrecord, r *context, gp *g) { g0 := getg() if panicking.Load() != 0 { // traceback already printed exit(2) } panicking.Store(1) // In case we're handling a g0 stack overflow, blow away the // g0 stack bounds so we have room to print the traceback. If // this somehow overflows the stack, the OS will trap it. g0.stack.lo = 0 g0.stackguard0 = g0.stack.lo + stackGuard g0.stackguard1 = g0.stackguard0 print("Exception ", hex(info.exceptioncode), " ", hex(info.exceptioninformation[0]), " ", hex(info.exceptioninformation[1]), " ", hex(r.ip()), "\n") print("PC=", hex(r.ip()), "\n") if g0.m.incgo && gp == g0.m.g0 && g0.m.curg != nil { if iscgo { print("signal arrived during external code execution\n") } gp = g0.m.curg } print("\n") g0.m.throwing = throwTypeRuntime g0.m.caughtsig.set(gp) level, _, docrash := gotraceback() if level > 0 { tracebacktrap(r.ip(), r.sp(), r.lr(), gp) tracebackothers(gp) dumpregs(r) } if docrash { dieFromException(info, r) } exit(2) } func sigpanic() { gp := getg() if !canpanic() { throw("unexpected signal during runtime execution") } switch gp.sig { case _EXCEPTION_ACCESS_VIOLATION, _EXCEPTION_IN_PAGE_ERROR: if gp.sigcode1 < 0x1000 { panicmem() } if gp.paniconfault { panicmemAddr(gp.sigcode1) } if inUserArenaChunk(gp.sigcode1) { // We could check that the arena chunk is explicitly set to fault, // but the fact that we faulted on accessing it is enough to prove // that it is. print("accessed data from freed user arena ", hex(gp.sigcode1), "\n") } else { print("unexpected fault address ", hex(gp.sigcode1), "\n") } throw("fault") case _EXCEPTION_INT_DIVIDE_BY_ZERO: panicdivide() case _EXCEPTION_INT_OVERFLOW: panicoverflow() case _EXCEPTION_FLT_DENORMAL_OPERAND, _EXCEPTION_FLT_DIVIDE_BY_ZERO, _EXCEPTION_FLT_INEXACT_RESULT, _EXCEPTION_FLT_OVERFLOW, _EXCEPTION_FLT_UNDERFLOW: panicfloat() } throw("fault") } // Following are not implemented. func initsig(preinit bool) { } func sigenable(sig uint32) { } func sigdisable(sig uint32) { } func sigignore(sig uint32) { } func signame(sig uint32) string { return "" } //go:nosplit func crash() { dieFromException(nil, nil) } // dieFromException raises an exception that bypasses all exception handlers. // This provides the expected exit status for the shell. // //go:nosplit func dieFromException(info *exceptionrecord, r *context) { if info == nil { gp := getg() if gp.sig != 0 { // Try to reconstruct an exception record from // the exception information stored in gp. info = &exceptionrecord{ exceptionaddress: gp.sigpc, exceptioncode: gp.sig, numberparameters: 2, } info.exceptioninformation[0] = gp.sigcode0 info.exceptioninformation[1] = gp.sigcode1 } else { // By default, a failing Go application exits with exit code 2. // Use this value when gp does not contain exception info. info = &exceptionrecord{ exceptioncode: 2, } } } const FAIL_FAST_GENERATE_EXCEPTION_ADDRESS = 0x1 stdcall3(_RaiseFailFastException, uintptr(unsafe.Pointer(info)), uintptr(unsafe.Pointer(r)), FAIL_FAST_GENERATE_EXCEPTION_ADDRESS) } // gsignalStack is unused on Windows. type gsignalStack struct{}