123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539 |
- #
- #
- # Nim's Runtime Library
- # (c) Copyright 2015 Andreas Rumpf
- #
- # See the file "copying.txt", included in this
- # distribution, for details about the copyright.
- #
- # Exception handling code. Carefully coded so that tiny programs which do not
- # use the heap (and nor exceptions) do not include the GC or memory allocator.
- var
- errorMessageWriter*: (proc(msg: string) {.tags: [WriteIOEffect], benign,
- nimcall.})
- ## Function that will be called
- ## instead of stdmsg.write when printing stacktrace.
- ## Unstable API.
- proc c_fwrite(buf: pointer, size, n: csize, f: File): cint {.
- importc: "fwrite", header: "<stdio.h>".}
- proc rawWrite(f: File, s: string|cstring) =
- # we cannot throw an exception here!
- discard c_fwrite(cstring(s), 1, s.len, f)
- when not defined(windows) or not defined(guiapp):
- proc writeToStdErr(msg: cstring) = rawWrite(stdmsg, msg)
- else:
- proc MessageBoxA(hWnd: cint, lpText, lpCaption: cstring, uType: int): int32 {.
- header: "<windows.h>", nodecl.}
- proc writeToStdErr(msg: cstring) =
- discard MessageBoxA(0, msg, nil, 0)
- proc showErrorMessage(data: cstring) {.gcsafe.} =
- if errorMessageWriter != nil:
- errorMessageWriter($data)
- else:
- when defined(genode):
- # stderr not available by default, use the LOG session
- echo data
- else:
- writeToStdErr(data)
- proc quitOrDebug() {.inline.} =
- when not defined(endb):
- quit(1)
- else:
- endbStep() # call the debugger
- proc chckIndx(i, a, b: int): int {.inline, compilerproc, benign.}
- proc chckRange(i, a, b: int): int {.inline, compilerproc, benign.}
- proc chckRangeF(x, a, b: float): float {.inline, compilerproc, benign.}
- proc chckNil(p: pointer) {.noinline, compilerproc, benign.}
- type
- GcFrame = ptr GcFrameHeader
- GcFrameHeader {.compilerproc.} = object
- len: int
- prev: ptr GcFrameHeader
- var
- framePtr {.threadvar.}: PFrame
- excHandler {.threadvar.}: PSafePoint
- # list of exception handlers
- # a global variable for the root of all try blocks
- currException {.threadvar.}: ref Exception
- raiseCounter {.threadvar.}: uint
- gcFramePtr {.threadvar.}: GcFrame
- type
- FrameState = tuple[gcFramePtr: GcFrame, framePtr: PFrame,
- excHandler: PSafePoint, currException: ref Exception]
- proc getFrameState*(): FrameState {.compilerRtl, inl.} =
- return (gcFramePtr, framePtr, excHandler, currException)
- proc setFrameState*(state: FrameState) {.compilerRtl, inl.} =
- gcFramePtr = state.gcFramePtr
- framePtr = state.framePtr
- excHandler = state.excHandler
- currException = state.currException
- proc getFrame*(): PFrame {.compilerRtl, inl.} = framePtr
- proc popFrame {.compilerRtl, inl.} =
- framePtr = framePtr.prev
- when false:
- proc popFrameOfAddr(s: PFrame) {.compilerRtl.} =
- var it = framePtr
- if it == s:
- framePtr = framePtr.prev
- else:
- while it != nil:
- if it == s:
- framePtr = it.prev
- break
- it = it.prev
- proc setFrame*(s: PFrame) {.compilerRtl, inl.} =
- framePtr = s
- proc getGcFrame*(): GcFrame {.compilerRtl, inl.} = gcFramePtr
- proc popGcFrame*() {.compilerRtl, inl.} = gcFramePtr = gcFramePtr.prev
- proc setGcFrame*(s: GcFrame) {.compilerRtl, inl.} = gcFramePtr = s
- proc pushGcFrame*(s: GcFrame) {.compilerRtl, inl.} =
- s.prev = gcFramePtr
- zeroMem(cast[pointer](cast[int](s)+%sizeof(GcFrameHeader)), s.len*sizeof(pointer))
- gcFramePtr = s
- proc pushSafePoint(s: PSafePoint) {.compilerRtl, inl.} =
- s.hasRaiseAction = false
- s.prev = excHandler
- excHandler = s
- proc popSafePoint {.compilerRtl, inl.} =
- excHandler = excHandler.prev
- proc pushCurrentException(e: ref Exception) {.compilerRtl, inl.} =
- e.up = currException
- currException = e
- proc popCurrentException {.compilerRtl, inl.} =
- currException = currException.up
- proc popCurrentExceptionEx(id: uint) {.compilerRtl.} =
- # in cpp backend exceptions can pop-up in the different order they were raised, example #5628
- if currException.raiseId == id:
- currException = currException.up
- else:
- var cur = currException.up
- var prev = currException
- while cur != nil and cur.raiseId != id:
- prev = cur
- cur = cur.up
- if cur == nil:
- showErrorMessage("popCurrentExceptionEx() exception was not found in the exception stack. Aborting...")
- quitOrDebug()
- prev.up = cur.up
- proc closureIterSetupExc(e: ref Exception) {.compilerproc, inline.} =
- if not e.isNil:
- currException = e
- # some platforms have native support for stack traces:
- const
- nativeStackTraceSupported* = (defined(macosx) or defined(linux)) and
- not NimStackTrace
- hasSomeStackTrace = NimStackTrace or
- defined(nativeStackTrace) and nativeStackTraceSupported
- when defined(nativeStacktrace) and nativeStackTraceSupported:
- type
- TDl_info {.importc: "Dl_info", header: "<dlfcn.h>",
- final, pure.} = object
- dli_fname: cstring
- dli_fbase: pointer
- dli_sname: cstring
- dli_saddr: pointer
- proc backtrace(symbols: ptr pointer, size: int): int {.
- importc: "backtrace", header: "<execinfo.h>".}
- proc dladdr(addr1: pointer, info: ptr TDl_info): int {.
- importc: "dladdr", header: "<dlfcn.h>".}
- when not hasThreadSupport:
- var
- tempAddresses: array[0..127, pointer] # should not be alloc'd on stack
- tempDlInfo: TDl_info
- proc auxWriteStackTraceWithBacktrace(s: var string) =
- when hasThreadSupport:
- var
- tempAddresses: array[0..127, pointer] # but better than a threadvar
- tempDlInfo: TDl_info
- # This is allowed to be expensive since it only happens during crashes
- # (but this way you don't need manual stack tracing)
- var size = backtrace(cast[ptr pointer](addr(tempAddresses)),
- len(tempAddresses))
- var enabled = false
- for i in 0..size-1:
- var dlresult = dladdr(tempAddresses[i], addr(tempDlInfo))
- if enabled:
- if dlresult != 0:
- var oldLen = s.len
- add(s, tempDlInfo.dli_fname)
- if tempDlInfo.dli_sname != nil:
- for k in 1..max(1, 25-(s.len-oldLen)): add(s, ' ')
- add(s, tempDlInfo.dli_sname)
- else:
- add(s, '?')
- add(s, "\n")
- else:
- if dlresult != 0 and tempDlInfo.dli_sname != nil and
- c_strcmp(tempDlInfo.dli_sname, "signalHandler") == 0'i32:
- # Once we're past signalHandler, we're at what the user is
- # interested in
- enabled = true
- when not hasThreadSupport:
- var
- tempFrames: array[0..127, PFrame] # should not be alloc'd on stack
- const
- reraisedFromBegin = -10
- reraisedFromEnd = -100
- template reraisedFrom(z): untyped =
- StackTraceEntry(procname: nil, line: z, filename: nil)
- proc auxWriteStackTrace(f: PFrame; s: var seq[StackTraceEntry]) =
- var
- it = f
- i = 0
- while it != nil:
- inc(i)
- it = it.prev
- var last = i-1
- if s.len == 0:
- s = newSeq[StackTraceEntry](i)
- else:
- last = s.len + i - 1
- s.setLen(last+1)
- it = f
- while it != nil:
- s[last] = StackTraceEntry(procname: it.procname,
- line: it.line,
- filename: it.filename)
- it = it.prev
- dec last
- template addFrameEntry(s, f: untyped) =
- var oldLen = s.len
- add(s, f.filename)
- if f.line > 0:
- add(s, '(')
- add(s, $f.line)
- add(s, ')')
- for k in 1..max(1, 25-(s.len-oldLen)): add(s, ' ')
- add(s, f.procname)
- add(s, "\n")
- proc `$`(s: seq[StackTraceEntry]): string =
- result = newStringOfCap(2000)
- for i in 0 .. s.len-1:
- if s[i].line == reraisedFromBegin: result.add "[[reraised from:\n"
- elif s[i].line == reraisedFromEnd: result.add "]]\n"
- else: addFrameEntry(result, s[i])
- proc auxWriteStackTrace(f: PFrame, s: var string) =
- when hasThreadSupport:
- var
- tempFrames: array[0..127, PFrame] # but better than a threadvar
- const
- firstCalls = 32
- var
- it = f
- i = 0
- total = 0
- # setup long head:
- while it != nil and i <= high(tempFrames)-firstCalls:
- tempFrames[i] = it
- inc(i)
- inc(total)
- it = it.prev
- # go up the stack to count 'total':
- var b = it
- while it != nil:
- inc(total)
- it = it.prev
- var skipped = 0
- if total > len(tempFrames):
- # skip N
- skipped = total-i-firstCalls+1
- for j in 1..skipped:
- if b != nil: b = b.prev
- # create '...' entry:
- tempFrames[i] = nil
- inc(i)
- # setup short tail:
- while b != nil and i <= high(tempFrames):
- tempFrames[i] = b
- inc(i)
- b = b.prev
- for j in countdown(i-1, 0):
- if tempFrames[j] == nil:
- add(s, "(")
- add(s, $skipped)
- add(s, " calls omitted) ...\n")
- else:
- addFrameEntry(s, tempFrames[j])
- proc stackTraceAvailable*(): bool
- when hasSomeStackTrace:
- proc rawWriteStackTrace(s: var string) =
- when NimStackTrace:
- if framePtr == nil:
- add(s, "No stack traceback available\n")
- else:
- add(s, "Traceback (most recent call last)\n")
- auxWriteStackTrace(framePtr, s)
- elif defined(nativeStackTrace) and nativeStackTraceSupported:
- add(s, "Traceback from system (most recent call last)\n")
- auxWriteStackTraceWithBacktrace(s)
- else:
- add(s, "No stack traceback available\n")
- proc rawWriteStackTrace(s: var seq[StackTraceEntry]) =
- when NimStackTrace:
- auxWriteStackTrace(framePtr, s)
- else:
- s = @[]
- proc stackTraceAvailable(): bool =
- when NimStackTrace:
- if framePtr == nil:
- result = false
- else:
- result = true
- elif defined(nativeStackTrace) and nativeStackTraceSupported:
- result = true
- else:
- result = false
- else:
- proc stackTraceAvailable*(): bool = result = false
- var onUnhandledException*: (proc (errorMsg: string) {.
- nimcall.}) ## set this error \
- ## handler to override the existing behaviour on an unhandled exception.
- ## The default is to write a stacktrace to ``stderr`` and then call ``quit(1)``.
- ## Unstable API.
- template unhandled(buf, body) =
- if onUnhandledException != nil:
- onUnhandledException($buf)
- else:
- body
- proc raiseExceptionAux(e: ref Exception) =
- if localRaiseHook != nil:
- if not localRaiseHook(e): return
- if globalRaiseHook != nil:
- if not globalRaiseHook(e): return
- when defined(cpp) and not defined(noCppExceptions):
- if e[] of OutOfMemError:
- showErrorMessage(e.name)
- quitOrDebug()
- else:
- pushCurrentException(e)
- raiseCounter.inc
- if raiseCounter == 0:
- raiseCounter.inc # skip zero at overflow
- e.raiseId = raiseCounter
- {.emit: "`e`->raise();".}
- else:
- if excHandler != nil:
- if not excHandler.hasRaiseAction or excHandler.raiseAction(e):
- pushCurrentException(e)
- c_longjmp(excHandler.context, 1)
- elif e[] of OutOfMemError:
- showErrorMessage(e.name)
- quitOrDebug()
- else:
- when hasSomeStackTrace:
- var buf = newStringOfCap(2000)
- if e.trace.len == 0: rawWriteStackTrace(buf)
- else: add(buf, $e.trace)
- add(buf, "Error: unhandled exception: ")
- add(buf, e.msg)
- add(buf, " [")
- add(buf, $e.name)
- add(buf, "]\n")
- unhandled(buf):
- showErrorMessage(buf)
- quitOrDebug()
- else:
- # ugly, but avoids heap allocations :-)
- template xadd(buf, s, slen) =
- if L + slen < high(buf):
- copyMem(addr(buf[L]), cstring(s), slen)
- inc L, slen
- template add(buf, s) =
- xadd(buf, s, s.len)
- var buf: array[0..2000, char]
- var L = 0
- if e.trace.len != 0:
- add(buf, $e.trace) # gc allocation
- add(buf, "Error: unhandled exception: ")
- add(buf, e.msg)
- add(buf, " [")
- xadd(buf, e.name, e.name.len)
- add(buf, "]\n")
- when defined(nimNoArrayToCstringConversion):
- template tbuf(): untyped = addr buf
- else:
- template tbuf(): untyped = buf
- unhandled(tbuf()):
- showErrorMessage(tbuf())
- quitOrDebug()
- proc raiseExceptionEx(e: ref Exception, ename, procname, filename: cstring, line: int) {.compilerRtl.} =
- if e.name.isNil: e.name = ename
- when hasSomeStackTrace:
- if e.trace.len == 0:
- rawWriteStackTrace(e.trace)
- elif framePtr != nil:
- e.trace.add reraisedFrom(reraisedFromBegin)
- auxWriteStackTrace(framePtr, e.trace)
- e.trace.add reraisedFrom(reraisedFromEnd)
- else:
- if procname != nil and filename != nil:
- e.trace.add StackTraceEntry(procname: procname, filename: filename, line: line)
- raiseExceptionAux(e)
- proc raiseException(e: ref Exception, ename: cstring) {.compilerRtl.} =
- raiseExceptionEx(e, ename, nil, nil, 0)
- proc reraiseException() {.compilerRtl.} =
- if currException == nil:
- sysFatal(ReraiseError, "no exception to reraise")
- else:
- raiseExceptionAux(currException)
- proc writeStackTrace() =
- when hasSomeStackTrace:
- var s = ""
- rawWriteStackTrace(s)
- cast[proc (s: cstring) {.noSideEffect, tags: [], nimcall.}](showErrorMessage)(s)
- else:
- cast[proc (s: cstring) {.noSideEffect, tags: [], nimcall.}](showErrorMessage)("No stack traceback available\n")
- proc getStackTrace(): string =
- when hasSomeStackTrace:
- result = ""
- rawWriteStackTrace(result)
- else:
- result = "No stack traceback available\n"
- proc getStackTrace(e: ref Exception): string =
- if not isNil(e):
- result = $e.trace
- else:
- result = ""
- when not defined(gcDestructors):
- proc getStackTraceEntries*(e: ref Exception): seq[StackTraceEntry] =
- ## Returns the attached stack trace to the exception ``e`` as
- ## a ``seq``. This is not yet available for the JS backend.
- shallowCopy(result, e.trace)
- when defined(nimRequiresNimFrame):
- const nimCallDepthLimit {.intdefine.} = 2000
- proc callDepthLimitReached() {.noinline.} =
- writeStackTrace()
- showErrorMessage("Error: call depth limit reached in a debug build (" &
- $nimCallDepthLimit & " function calls). You can change it with " &
- "-d:nimCallDepthLimit=<int> but really try to avoid deep " &
- "recursions instead.\n")
- quitOrDebug()
- proc nimFrame(s: PFrame) {.compilerRtl, inl, exportc: "nimFrame".} =
- s.calldepth = if framePtr == nil: 0 else: framePtr.calldepth+1
- s.prev = framePtr
- framePtr = s
- if s.calldepth == nimCallDepthLimit: callDepthLimitReached()
- else:
- proc pushFrame(s: PFrame) {.compilerRtl, inl, exportc: "nimFrame".} =
- # XXX only for backwards compatibility
- s.prev = framePtr
- framePtr = s
- when defined(endb):
- var
- dbgAborting: bool # whether the debugger wants to abort
- when not defined(noSignalHandler) and not defined(useNimRtl):
- proc signalHandler(sign: cint) {.exportc: "signalHandler", noconv.} =
- template processSignal(s, action: untyped) {.dirty.} =
- if s == SIGINT: action("SIGINT: Interrupted by Ctrl-C.\n")
- elif s == SIGSEGV:
- action("SIGSEGV: Illegal storage access. (Attempt to read from nil?)\n")
- elif s == SIGABRT:
- when defined(endb):
- if dbgAborting: return # the debugger wants to abort
- action("SIGABRT: Abnormal termination.\n")
- elif s == SIGFPE: action("SIGFPE: Arithmetic error.\n")
- elif s == SIGILL: action("SIGILL: Illegal operation.\n")
- elif s == SIGBUS:
- action("SIGBUS: Illegal storage access. (Attempt to read from nil?)\n")
- else:
- block platformSpecificSignal:
- when declared(SIGPIPE):
- if s == SIGPIPE:
- action("SIGPIPE: Pipe closed.\n")
- break platformSpecificSignal
- action("unknown signal\n")
- # print stack trace and quit
- when defined(memtracker):
- logPendingOps()
- when hasSomeStackTrace:
- GC_disable()
- var buf = newStringOfCap(2000)
- rawWriteStackTrace(buf)
- processSignal(sign, buf.add) # nice hu? currying a la Nim :-)
- showErrorMessage(buf)
- GC_enable()
- else:
- var msg: cstring
- template asgn(y) =
- msg = y
- processSignal(sign, asgn)
- showErrorMessage(msg)
- when defined(endb): dbgAborting = true
- quit(1) # always quit when SIGABRT
- proc registerSignalHandler() =
- c_signal(SIGINT, signalHandler)
- c_signal(SIGSEGV, signalHandler)
- c_signal(SIGABRT, signalHandler)
- c_signal(SIGFPE, signalHandler)
- c_signal(SIGILL, signalHandler)
- c_signal(SIGBUS, signalHandler)
- when declared(SIGPIPE):
- c_signal(SIGPIPE, signalHandler)
- registerSignalHandler() # call it in initialization section
- proc setControlCHook(hook: proc () {.noconv.}) =
- # ugly cast, but should work on all architectures:
- type SignalHandler = proc (sign: cint) {.noconv, benign.}
- c_signal(SIGINT, cast[SignalHandler](hook))
|