On this page
Stack Traces
The StackTraces
module provides simple stack traces that are both human readable and easy to use programmatically.
Viewing a stack trace
The primary function used to obtain a stack trace is stacktrace()
:
julia> stacktrace()
3-element Array{StackFrame,1}:
eval at boot.jl:265
[inlined code from REPL.jl:3] eval_user_input at REPL.jl:62
[inlined code from REPL.jl:92] anonymous at task.jl:63
Calling stacktrace()
returns a vector of StackFrame
s. For ease of use, the alias StackTrace
can be used in place of Vector{StackFrame}
. (Examples with ...
indicate that output may vary depending on how the code is run.)
julia> example() = stacktrace()
example (generic function with 1 method)
julia> example()
6-element Array{StackFrame,1}:
in example() at none:1
in eval(::Module, ::Any) at boot.jl:234
in eval_user_input(::Any, ::Bool) at client.jl:117
in eval(::Module, ::Any) at boot.jl:234
in eval_user_input(::Any, ::Bool) at client.jl:117
in _start() at client.jl:355
julia> @noinline child() = stacktrace()
child (generic function with 1 method)
julia> @noinline parent() = child()
parent (generic function with 1 method)
julia> grandparent() = parent()
grandparent (generic function with 1 method)
julia> grandparent()
8-element Array{StackFrame,1}:
in child() at none:1
in parent() at none:1
in grandparent() at none:1
...
Note that when calling stacktrace()
you’ll typically see a frame with eval(...) at boot.jl
. When calling stacktrace()
from the REPL you’ll also have a few extra frames in the stack from REPL.jl
, usually looking something like this:
julia> example() = stacktrace()
example (generic function with 1 method)
julia> example()
5-element Array{StackFrame,1}:
in example() at REPL[1]:1
in eval(::Module, ::Any) at boot.jl:234
in eval_user_input(::Any, ::Base.REPL.REPLBackend) at REPL.jl:62
in macro expansion at REPL.jl:92 [inlined]
in (::Base.REPL.##1#2{Base.REPL.REPLBackend})() at event.jl:46
Extracting useful information
Each StackFrame
contains the function name, file name, line number, lambda info, a flag indicating whether the frame has been inlined, a flag indicating whether it is a C function (by default C functions do not appear in the stack trace), and an integer representation of the pointer returned by backtrace()
:
julia> top_frame = stacktrace()[1]
in eval(::Module, ::Any) at boot.jl:234
julia> top_frame.func
:eval
julia> top_frame.file
Symbol("./boot.jl")
julia> top_frame.line
234
julia> top_frame.linfo
Nullable{LambdaInfo}(LambdaInfo for eval(::Module, ::Any))
julia> top_frame.inlined
false
julia> top_frame.from_c
false
julia> top_frame.pointer
13203085684
This makes stack trace information available programmatically for logging, error handling, and more.
Error handling
While having easy access to information about the current state of the callstack can be helpful in many places, the most obvious application is in error handling and debugging.
julia> @noinline bad_function() = undeclared_variable
bad_function (generic function with 1 method)
julia> @noinline example() = try
bad_function()
catch
stacktrace()
end
example (generic function with 1 method)
julia> example()
6-element Array{StackFrame,1}:
in example() at none:4
in eval(::Module, ::Any) at boot.jl:234
...
You may notice that in the example above the first stack frame points points at line 4, where stacktrace()
is called, rather than line 2, where bad_function
is called, and bad_function
‘s frame is missing entirely. This is understandable, given that stacktrace()
is called from the context of the catch
. While in this example it’s fairly easy to find the actual source of the error, in complex cases tracking down the source of the error becomes nontrivial.
This can be remedied by calling catch_stacktrace()
instead of stacktrace()
. Instead of returning callstack information for the current context, catch_stacktrace()
returns stack information for the context of the most recent exception:
julia> @noinline bad_function() = undeclared_variable
bad_function (generic function with 1 method)
julia> @noinline example() = try
bad_function()
catch
catch_stacktrace()
end
example (generic function with 1 method)
julia> example()
7-element Array{StackFrame,1}:
in bad_function() at none:1
in example() at none:2
...
Notice that the stack trace now indicates the appropriate line number and the missing frame.
julia> @noinline child() = error("Whoops!")
child (generic function with 1 method)
julia> @noinline parent() = child()
parent (generic function with 1 method)
julia> @noinline function grandparent()
try
parent()
catch err
println("ERROR: ", err.msg)
catch_stacktrace()
end
end
grandparent (generic function with 1 method)
julia> grandparent()
ERROR: Whoops!
8-element Array{StackFrame,1}:
in child() at none:1
in parent() at none:1
in grandparent() at none:3
...
Comparison with backtrace()
A call to backtrace()
returns a vector of Ptr{Void}
, which may then be passed into stacktrace()
for translation:
julia> trace = backtrace()
20-element Array{Ptr{Void},1}:
Ptr{Void} @0x0000000100a26fc2
Ptr{Void} @0x00000001029435df
Ptr{Void} @0x0000000102943635
Ptr{Void} @0x00000001009e9620
Ptr{Void} @0x00000001009fe1e8
Ptr{Void} @0x00000001009fc7b6
Ptr{Void} @0x00000001009fdae3
Ptr{Void} @0x00000001009fe0d2
Ptr{Void} @0x0000000100a1321b
Ptr{Void} @0x00000001009f64e7
Ptr{Void} @0x000000010265ac5d
Ptr{Void} @0x000000010265acc1
Ptr{Void} @0x00000001009e9620
Ptr{Void} @0x000000031007744b
Ptr{Void} @0x0000000310077537
Ptr{Void} @0x00000001009e9620
Ptr{Void} @0x000000031006feec
Ptr{Void} @0x00000003100701b0
Ptr{Void} @0x00000001009e9635
Ptr{Void} @0x0000000100a06418
julia> stacktrace(trace)
5-element Array{StackFrame,1}:
in backtrace() at error.jl:26
in eval(::Module, ::Any) at boot.jl:231
in eval_user_input(::Any, ::Base.REPL.REPLBackend) at REPL.jl:62
in macro expansion at REPL.jl:92 [inlined]
in (::Base.REPL.##1#2{Base.REPL.REPLBackend})() at event.jl:46
Notice that the vector returned by backtrace()
had 15 pointers, while the vector returned by stacktrace()
only has 4. This is because, by default, stacktrace()
removes any lower-level C functions from the stack. If you want to include stack frames from C calls, you can do it like this:
julia> stacktrace(trace, true)
26-element Array{StackFrame,1}:
in jl_backtrace_from_here at stackwalk.c:104
in backtrace() at error.jl:26
in ip:0x102943635
in jl_call_method_internal at julia_internal.h:86 [inlined]
in jl_apply_generic at gf.c:1805
in do_call at interpreter.c:65
in eval at interpreter.c:188
in eval_body at interpreter.c:469
in jl_interpret_call at interpreter.c:573
in jl_toplevel_eval_flex at toplevel.c:543
in jl_toplevel_eval_in_warn at builtins.c:571
in eval(::Module, ::Any) at boot.jl:231
in ip:0x10265acc1
in jl_call_method_internal at julia_internal.h:86 [inlined]
in jl_apply_generic at gf.c:1805
in eval_user_input(::Any, ::Base.REPL.REPLBackend) at REPL.jl:62
in ip:0x310077537
in jl_call_method_internal at julia_internal.h:86 [inlined]
in jl_apply_generic at gf.c:1805
in macro expansion at REPL.jl:92 [inlined]
in (::Base.REPL.##1#2{Base.REPL.REPLBackend})() at event.jl:46
in ip:0x3100701b0
in jl_call_method_internal at julia_internal.h:86 [inlined]
in jl_apply_generic at gf.c:1795
in jl_apply at julia.h:1388 [inlined]
in start_task at task.c:247
Individual pointers returned by backtrace()
can be translated into StackFrame
s by passing them into StackTraces.lookup()
:
julia> pointer = backtrace()[1];
julia> frame = StackTraces.lookup(pointer)
1-element Array{StackFrame,1}:
in jl_backtrace_from_here at stackwalk.c:105
julia> println("The top frame is from $(frame[1].func)!")
The top frame is from jl_backtrace_from_here!
© 2009–2016 Jeff Bezanson, Stefan Karpinski, Viral B. Shah, and other contributors
Licensed under the MIT License.
https://docs.julialang.org/en/release-0.5/manual/stacktraces/