Delve, a better go debugger

作者 Lu Liang 日期 2018-11-22
Delve, a better go debugger

Delve is a debugger for the Go programming language. We can debug and explore go program with it easily.

Installation

Delve can be installed by “go get”. After the installation, you can find it under $HOME/go. All we need to do is to add the folder “$HOME/go/bin” to $PATH.

go get -u github.com/derekparker/delve/cmd/dlv
export PATH=$PATH:$HOME/go/bin

Usage

The following are 3 ways to debug go program with dlv.

  • dlv command.

    dlv attach      Attach to running process and begin debugging.
    dlv connect Connect to a headless debug server.
    dlv core Examine a core dump.
    dlv debug Compile and begin debugging main package in current directory, or the package specified.
    dlv exec Execute a precompiled binary, and begin a debug session.
    dlv help Help about any command
    dlv run Deprecated command. Use 'debug' instead.
    dlv test Compile test binary and begin debugging program.
    dlv trace Compile and begin tracing program.
    dlv version Prints version
  • dlv api

    # start dlv with --headless at remote and use dlv ui to connect in local.
    dlv attach 18707 --headless --api-version=2 --log --listen=9.xx.xx.65:8181

    # start gdlv to connect the remote debug.
    ➜ gdlv connect 9.xx.xx.65:8181
  • dlv ui
    There are a lot of editor plugins for dlv ui. You can install them based on this instruction https://github.com/derekparker/delve/blob/master/Documentation/EditorIntegration.md. Here, we take gdlv as example to do introduction.

     # install gdlv with "go get"
    go get -u github.com/aarzilli/gdlv

    # gdlv usage.
    gdlv connect <address>
    gdlv debug <program's arguments...>
    gdlv run <program file> <program's arguments...>
    gdlv exec <executable> <program's arguments...>
    gdlv test <testflags...>
    gdlv attach <pid> [path to executable]
    gdlv core <executable> <core file>
    gdlv reply <trace directory>

Example.

The following are some samples to show dlv with the hello_world.

the source code of hellow_world

To avoid the program exit immediately, we add some sleep time before print the “hello world”.

package main

import (
"fmt"
"time"
)

func main() {
time.Sleep(time.Duration(60)*time.Second)
fmt.Printf("hello, world\n")
}

attach the program

➜  bin ps -ef|grep hello
root 18192 18133 0 11月21 pts/2 00:00:00 ./hello_world

➜ bin ./dlv attach 18192
Type 'help' for list of commands.
(dlv)

show available commands

(dlv) help
The following commands are available:
args ------------------------ Print function arguments.
break (alias: b) ------------ Sets a breakpoint.
breakpoints (alias: bp) ----- Print out info for active breakpoints.
call ------------------------ Resumes process, injecting a function call (EXPERIMENTAL!!!)
clear ----------------------- Deletes breakpoint.
clearall -------------------- Deletes multiple breakpoints.
condition (alias: cond) ----- Set breakpoint condition.
config ---------------------- Changes configuration parameters.
continue (alias: c) --------- Run until breakpoint or program termination.
deferred -------------------- Executes command in the context of a deferred call.
disassemble (alias: disass) - Disassembler.
down ------------------------ Move the current frame down.
edit (alias: ed) ------------ Open where you are in $DELVE_EDITOR or $EDITOR
exit (alias: quit | q) ------ Exit the debugger.
frame ----------------------- Set the current frame, or execute command on a different frame.
funcs ----------------------- Print list of functions.
goroutine ------------------- Shows or changes current goroutine
goroutines ------------------ List program goroutines.
help (alias: h) ------------- Prints the help message.
list (alias: ls | l) -------- Show source code.
locals ---------------------- Print local variables.
next (alias: n) ------------- Step over to next source line.
on -------------------------- Executes a command when a breakpoint is hit.
print (alias: p) ------------ Evaluate an expression.
regs ------------------------ Print contents of CPU registers.
restart (alias: r) ---------- Restart process.
set ------------------------- Changes the value of a variable.
source ---------------------- Executes a file containing a list of delve commands
sources --------------------- Print list of source files.
stack (alias: bt) ----------- Print stack trace.
step (alias: s) ------------- Single step through program.
step-instruction (alias: si) Single step a single cpu instruction.
stepout --------------------- Step out of the current function.
thread (alias: tr) ---------- Switch to the specified thread.
threads --------------------- Print out info for every traced thread.
trace (alias: t) ------------ Set tracepoint.
types ----------------------- Print list of types
up -------------------------- Move the current frame up.
vars ------------------------ Print package variables.
whatis ---------------------- Prints type of an expression.
Type help followed by a command for full documentation.

show all goroutines

The running goroutine is 19.

(dlv) goroutines
[5 goroutines]
Goroutine 1 - User: /usr/lib/golang/src/runtime/time.go:65 time.Sleep (0x4429e0)
Goroutine 2 - User: /usr/lib/golang/src/runtime/proc.go:288 runtime.gopark (0x428eac)
Goroutine 17 - User: /usr/lib/golang/src/runtime/proc.go:288 runtime.gopark (0x428eac)
Goroutine 18 - User: /usr/lib/golang/src/runtime/proc.go:288 runtime.gopark (0x428eac)
* Goroutine 19 - User: /usr/lib/golang/src/runtime/lock_futex.go:227 runtime.notetsleepg (0x40d582)

show stack for specified goroutine

(dlv) goroutine 1 stack
0 0x0000000000428eac in runtime.gopark
at /usr/lib/golang/src/runtime/proc.go:288
1 0x0000000000428f9e in runtime.goparkunlock
at /usr/lib/golang/src/runtime/proc.go:293
2 0x00000000004429e0 in time.Sleep
at /usr/lib/golang/src/runtime/time.go:65
3 0x0000000000489bf0 in main.main
at /root/tmp/go/hello_world.go:9
4 0x0000000000428a06 in runtime.main
at /usr/lib/golang/src/runtime/proc.go:195
5 0x0000000000451721 in runtime.goexit
at /usr/lib/golang/src/runtime/asm_amd64.s:2337

(dlv) goroutine 2 stack
0 0x0000000000428eac in runtime.gopark
at /usr/lib/golang/src/runtime/proc.go:288
1 0x0000000000428f9e in runtime.goparkunlock
at /usr/lib/golang/src/runtime/proc.go:293
2 0x0000000000428ccc in runtime.forcegchelper
at /usr/lib/golang/src/runtime/proc.go:245
3 0x0000000000451721 in runtime.goexit
at /usr/lib/golang/src/runtime/asm_amd64.s:2337

(dlv) goroutine 19 stack
0 0x000000000040d582 in runtime.notetsleepg
at /usr/lib/golang/src/runtime/lock_futex.go:227
1 0x00000000004431f5 in runtime.timerproc
at /usr/lib/golang/src/runtime/time.go:216
2 0x0000000000451721 in runtime.goexit
at /usr/lib/golang/src/runtime/asm_amd64.s:2337

show threads

(dlv) threads
* Thread 18192 at 0x452bd3 /usr/lib/golang/src/runtime/sys_linux_amd64.s:480 runtime.futex
Thread 18193 at 0x452bd3 /usr/lib/golang/src/runtime/sys_linux_amd64.s:480 runtime.futex
Thread 18194 at 0x452bd3 /usr/lib/golang/src/runtime/sys_linux_amd64.s:480 runtime.futex
Thread 18195 at 0x452bd3 /usr/lib/golang/src/runtime/sys_linux_amd64.s:480 runtime.futex
Thread 18196 at 0x452bd3 /usr/lib/golang/src/runtime/sys_linux_amd64.s:480 runtime.futex

list relevant source code

(dlv) l
> runtime.futex() /usr/lib/golang/src/runtime/sys_linux_amd64.s:480 (PC: 0x452bd3)
475: MOVQ ts+16(FP), R10
476: MOVQ addr2+24(FP), R8
477: MOVL val3+32(FP), R9
478: MOVL $202, AX
479: SYSCALL
=> 480: MOVL AX, ret+40(FP)
481: RET
482:
483: // int32 clone(int32 flags, void *stk, M *mp, G *gp, void (*fn)(void));
484: TEXT runtime·clone(SB),NOSPLIT,$0
485: MOVL flags+0(FP), DI
(dlv)

use up/down to go through the current frame.

(dlv) down
> runtime.futex() /usr/lib/golang/src/runtime/sys_linux_amd64.s:480 (PC: 0x452bd3)
Frame 4: /usr/lib/golang/src/runtime/time.go:216 (PC: 4431f5)
211: // At least one timer pending. Sleep until then.
212: timers.sleeping = true
213: timers.sleepUntil = now + delta
214: noteclear(&timers.waitnote)
215: unlock(&timers.lock)
=> 216: notetsleepg(&timers.waitnote, delta)
217: }
218: }
219:
220: func timejump() *g {
221: if faketime == 0 {
(dlv) up
> runtime.futex() /usr/lib/golang/src/runtime/sys_linux_amd64.s:480 (PC: 0x452bd3)
Frame 5: /usr/lib/golang/src/runtime/asm_amd64.s:2337 (PC: 451721)
2332: RET
2333:
2334: // The top-most function running on a goroutine
2335: // returns to goexit+PCQuantum.
2336: TEXT runtime·goexit(SB),NOSPLIT,$0-0
=>2337: BYTE $0x90 // NOP
2338: CALL runtime·goexit1(SB) // does not return
2339: // traceback from goexit1 must hit code range of goexit
2340: BYTE $0x90 // NOP
2341:
2342: TEXT runtime·prefetcht0(SB),NOSPLIT,$0-8
(dlv)

Refer to: