Skip to content

Commit b08c8a0

Browse files
committed
all: implement gdb sub-command for easy debugging
1 parent ef2ac09 commit b08c8a0

File tree

4 files changed

+122
-3
lines changed

4 files changed

+122
-3
lines changed

colorwriter.go

+39
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,39 @@
1+
package main
2+
3+
import (
4+
"io"
5+
)
6+
7+
// ANSI escape codes for terminal colors.
8+
const (
9+
TermColorReset = "\x1b[0m"
10+
TermColorYellow = "\x1b[93m"
11+
)
12+
13+
// ColorWriter wraps an io.Writer but adds a prefix and a terminal color.
14+
type ColorWriter struct {
15+
Out io.Writer
16+
Color string
17+
Prefix string
18+
line []byte
19+
}
20+
21+
// Write implements io.Writer, but with an added prefix and terminal color.
22+
func (w *ColorWriter) Write(p []byte) (n int, err error) {
23+
for _, c := range p {
24+
if c == '\n' {
25+
w.line = append(w.line, []byte(TermColorReset)...)
26+
w.line = append(w.line, '\n')
27+
// Write this line.
28+
_, err := w.Out.Write(w.line)
29+
w.line = w.line[:0]
30+
w.line = append(w.line, []byte(w.Color+w.Prefix)...)
31+
if err != nil {
32+
return 0, err
33+
}
34+
} else {
35+
w.line = append(w.line, c)
36+
}
37+
}
38+
return len(p), nil
39+
}

main.go

+73-2
Original file line numberDiff line numberDiff line change
@@ -8,8 +8,10 @@ import (
88
"io/ioutil"
99
"os"
1010
"os/exec"
11+
"os/signal"
1112
"path/filepath"
1213
"strings"
14+
"syscall"
1315

1416
"github.com/aykevl/go-llvm"
1517
"github.com/aykevl/tinygo/compiler"
@@ -199,6 +201,70 @@ func Flash(pkgName, target, port string, printIR, dumpSSA, debug bool, printSize
199201
})
200202
}
201203

204+
// Flash a program on a microcontroller and drop into a GDB shell.
205+
//
206+
// Note: this command is expected to execute just before exiting, as it
207+
// modifies global state.
208+
func FlashGDB(pkgName, target, port string, printIR, dumpSSA bool, printSizes string) error {
209+
spec, err := LoadTarget(target)
210+
if err != nil {
211+
return err
212+
}
213+
214+
if spec.GDB == "" {
215+
return errors.New("gdb not configured in the target specification")
216+
}
217+
218+
debug := true // always enable debug symbols
219+
return Compile(pkgName, "", spec, printIR, dumpSSA, debug, printSizes, func(tmppath string) error {
220+
if len(spec.OCDDaemon) != 0 {
221+
// We need a separate debugging daemon for on-chip debugging.
222+
daemon := exec.Command(spec.OCDDaemon[0], spec.OCDDaemon[1:]...)
223+
// Make it clear which output is from the daemon.
224+
daemon.Stderr = &ColorWriter{
225+
Out: os.Stderr,
226+
Prefix: spec.OCDDaemon[0] + ": ",
227+
Color: TermColorYellow,
228+
}
229+
// Make sure the daemon doesn't receive Ctrl-C that is intended for
230+
// GDB (to break the currently executing program).
231+
// https://stackoverflow.com/a/35435038/559350
232+
daemon.SysProcAttr = &syscall.SysProcAttr{
233+
Setpgid: true,
234+
Pgid: 0,
235+
}
236+
// Start now, and kill it on exit.
237+
daemon.Start()
238+
defer func() {
239+
daemon.Process.Signal(os.Interrupt)
240+
// Maybe we should send a .Kill() after x seconds?
241+
daemon.Wait()
242+
}()
243+
}
244+
245+
// Ignore Ctrl-C, it must be passed on to GDB.
246+
c := make(chan os.Signal, 1)
247+
signal.Notify(c, os.Interrupt)
248+
go func() {
249+
for range c {
250+
}
251+
}()
252+
253+
// Construct and execute a gdb command.
254+
// By default: gdb -ex run <binary>
255+
// Exit GDB with Ctrl-D.
256+
params := []string{tmppath}
257+
for _, cmd := range spec.GDBCmds {
258+
params = append(params, "-ex", cmd)
259+
}
260+
cmd := exec.Command(spec.GDB, params...)
261+
cmd.Stdin = os.Stdin
262+
cmd.Stdout = os.Stdout
263+
cmd.Stderr = os.Stderr
264+
return cmd.Run()
265+
})
266+
}
267+
202268
// Run the specified package directly (using JIT or interpretation).
203269
func Run(pkgName string) error {
204270
config := compiler.Config{
@@ -284,13 +350,18 @@ func main() {
284350
fmt.Fprintln(os.Stderr, "error:", err)
285351
os.Exit(1)
286352
}
287-
case "flash":
353+
case "flash", "gdb":
288354
if *outpath != "" {
289355
fmt.Fprintln(os.Stderr, "Output cannot be specified with the flash command.")
290356
usage()
291357
os.Exit(1)
292358
}
293-
err := Flash(flag.Arg(0), *target, *port, *printIR, *dumpSSA, !*nodebug, *printSize)
359+
var err error
360+
if command == "flash" {
361+
err = Flash(flag.Arg(0), *target, *port, *printIR, *dumpSSA, !*nodebug, *printSize)
362+
} else {
363+
err = FlashGDB(flag.Arg(0), *target, *port, *printIR, *dumpSSA, *printSize)
364+
}
294365
if err != nil {
295366
fmt.Fprintln(os.Stderr, "error:", err)
296367
os.Exit(1)

target.go

+6
Original file line numberDiff line numberDiff line change
@@ -20,6 +20,9 @@ type TargetSpec struct {
2020
PreLinkArgs []string `json:"pre-link-args"`
2121
Objcopy string `json:"objcopy"`
2222
Flasher string `json:"flash"`
23+
OCDDaemon []string `json:"ocd-daemon"`
24+
GDB string `json:"gdb"`
25+
GDBCmds []string `json:"gdb-initial-cmds"`
2326
}
2427

2528
// Load a target specification
@@ -30,13 +33,16 @@ func LoadTarget(target string) (*TargetSpec, error) {
3033
Linker: "cc",
3134
PreLinkArgs: []string{"-no-pie"}, // WARNING: clang < 5.0 requires -nopie
3235
Objcopy: "objcopy",
36+
GDB: "gdb",
37+
GDBCmds: []string{"run"},
3338
}
3439

3540
// See whether there is a target specification for this target (e.g.
3641
// Arduino).
3742
path := filepath.Join(sourceDir(), "targets", strings.ToLower(target)+".json")
3843
if fp, err := os.Open(path); err == nil {
3944
defer fp.Close()
45+
*spec = TargetSpec{} // reset all fields
4046
err := json.NewDecoder(fp).Decode(spec)
4147
if err != nil {
4248
return nil, err

targets/pca10040.json

+4-1
Original file line numberDiff line numberDiff line change
@@ -4,5 +4,8 @@
44
"linker": "arm-none-eabi-gcc",
55
"pre-link-args": ["-nostdlib", "-nostartfiles", "-mcpu=cortex-m4", "-mthumb", "-T", "targets/nrf52.ld", "-Wl,--gc-sections", "-fno-exceptions", "-fno-unwind-tables", "-ffunction-sections", "-fdata-sections", "-Os", "-DNRF52832_XXAA", "-Ilib/CMSIS/CMSIS/Include", "lib/nrfx/mdk/system_nrf52.c", "src/device/nrf/nrf52.s"],
66
"objcopy": "arm-none-eabi-objcopy",
7-
"flash": "nrfjprog -f nrf52 --sectorerase --program {hex} --reset"
7+
"flash": "nrfjprog -f nrf52 --sectorerase --program {hex} --reset",
8+
"ocd-daemon": ["openocd", "-f", "interface/jlink.cfg", "-c", "transport select swd", "-f", "target/nrf51.cfg"],
9+
"gdb": "arm-none-eabi-gdb",
10+
"gdb-initial-cmds": ["target remote :3333", "monitor halt", "load", "monitor reset", "c"]
811
}

0 commit comments

Comments
 (0)