-- Load up the debugger module and assign it to a variable.
local dbg = require("debugger")
print()
print[[
Welcome to the interactive debugger.lua tutorial.
You'll want to open tutorial.lua in an editor to follow along.
]]
print[[
You are now in the debugger! (Woo! \o/).
debugger.lua doesn't support traditional breakpoints.
Instead you call the dbg() object to set a breakpoint.
Notice how it prints out your current file and
line as well as which function you are in.
Keep a close watch on this as you follow along.
It should be at line XXX, a line after the dbg() call.
Sometimes functions don't have global names.
It might print the method name, local variable
that held the function, or file:line where it starts.
Type 's' to step to the next line.
(s = Step to the next executable line)
]]
-- Multi-line strings are executable statements apparently
-- need to put this in an local to make the tutorial flow nicely.
local str1 = [[
The 's' command steps to the next executable line.
This may step you into a function call.
In this case, then next line was a C function that printed this message.
You can't step into C functions, so it just steps over them.
If you hit , the debugger will rerun your last command.
Hit 5 times to step into and through func1().
Watch the line numbers.
]]
local str2 = [[
Stop!
You've now stepped through func1()
Notice how entering and exiting a function takes a step.
Now try the 'n' command.
(n = step to the Next line in the source code)
]]
local function func1()
print(" Stepping through func1()...")
print(" Almost there...")
end
local function func2()
print(" You used the 'n' command.")
print(" So it's skipping over the lines in func2().")
local function f()
print(" ... and anything it might call.")
end
f()
print()
print[[
The 'n' command also steps to the next line in the source file.
Unlike the 's' command, it steps over function
calls, not into them.
Now try the 'c' command to continue on to the next breakpoint.
(c = Continue execution)
]]
end
dbg()
print(str1)
func1()
print(str2)
func2()
local function func3()
print[[
You are now sitting at a breakpoint inside of a func3().
Let's say you got here by stepping into the function.
After poking around for a bit, you just want to step until the
function returns, but don't want to
run the next command over and over.
For this you would use the 'f' command. Try it now.
(f = Finish current function)
]]
dbg()
print[[
Now you are inside func4(), right after where it called func3().
func4() has some arguments, local variables and upvalues.
Let's assume you want to see them.
Try the 'l' command to list all the locally available variables.
(l = Local variables)
Type 'c' to continue on to the next section.
]]
end
local my_upvalue1 = "Wee an upvalue"
local my_upvalue2 = "Awww, can't see this one"
globalvar = "Weeee a global"
function func4(a, b, ...)
local c = "sea"
local varargs_copy = {...}
-- Functions only get upvalues if you reference them.
local d = my_upvalue1.." ... with stuff appended to it"
func3()
print[[
Some things to notice about the local variables list.
'(*vargargs)'
This is the list of varargs passed to the function.
(only works with Lua 5.2)
'(*temporary)'
Other values like this may (or may not) appear as well.
They are temporary values used by the lua interpreter.
They may be stripped out in the future.
'my_upvalue1'
This is a local variable defined outside of but
referenced by the function. Upvalues show up
*only* when you reference them within your
function. 'my_upvalue2' isn't in the list
because func4() doesn't reference it.
Listing the locals is nice, but sometimes it's just noise.
Often times it's useful to print just a single variable,
evaluate an expression, or call a function to see what it returns.
For that you use the 'p' command.
Try these commands:
p my_upvalue1
p 1 + 1
p print("foo")
p math.cos(0)
You can also interact with varargs,
but it depends on your Lua version.
In Lua 5.2 you can do this:
p select(2, ...)
In Lua 5.1 or LuaJIT you need to copy
the varargs into a table and unpack them:
p select(2, unpack(varargs_copy))
Type 'c' to continue to the next section.
]]
dbg()
end
func4(1, "two", "vararg1", "vararg2", "vararg3")
local function func5()
local my_var = "func5()"
print[[
You are now in func5() which was called from func6().
func6() was called from func7().
Try the 't' command to print out a backtrace and see for yourself.
(t = backTrace)
Type 'c' to continue to the next section
]]
dbg()
print[[
Notice that func5(), func6() and func7() all have a
'my_var' local. You can print the func5()'s my_var easily enough.
What if you wanted to see what local variables were in func6()
or func7() to see how you got where you were?
For that you use the 'u' and 'd' commands.
(u = move Up a stack frame)
(d = move Down a stack frame)
Try the 'u' and 'd' commands a few times.
Print out the value of my_var using the 'p' command each time.
Type 'c' to continue.
]]
dbg()
end
local function func6()
local my_var = "func6()"
func5()
end
local function func7()
local my_var = "func7()"
func6()
end
func7()
print[[
That leaves only one more command.
Wouldn't it be nice if there was a way to remember
all these one letter debugger commands?
Type 'h' to show the command list.
(h = Help)
Type 'c' to continue.
]]
dbg()
print[[
The following loop uses an assert-style breakpoint.
It will only engage when the conditional fails. (when i == 5)
Type 'c' to continue.
]]
for i=0, 10 do
print("i = "..tostring(i))
dbg(i ~= 5)
end
print[[
Last but not least, is the dbg.call() function.
It works sort of like Lua's xpcall() function,
but starts the debugger when an uncaught error occurs.
Note that dbg.call() does *not* take a list of varargs though.
You must call it on a function that takes no arguments.
dbg.call(function()
-- Potentially buggy code goes here.
end)
Wrap it around your program's main loop or main entry point.
Then when your program crashes, you won't need to go back
and add breakpoints.
That pretty much wraps ups the basics.
Hopefully you find debugger.lua to be simple but useful.
]]
dbg.call(function()
local foo = "foo"
-- Try adding a string and integer
local bar = foo + 12
-- Program never makes it to here...
end)