@@ -8,8 +8,10 @@ import (
8
8
"io/ioutil"
9
9
"os"
10
10
"os/exec"
11
+ "os/signal"
11
12
"path/filepath"
12
13
"strings"
14
+ "syscall"
13
15
14
16
"github.com/aykevl/go-llvm"
15
17
"github.com/aykevl/tinygo/compiler"
@@ -199,6 +201,70 @@ func Flash(pkgName, target, port string, printIR, dumpSSA, debug bool, printSize
199
201
})
200
202
}
201
203
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
+
202
268
// Run the specified package directly (using JIT or interpretation).
203
269
func Run (pkgName string ) error {
204
270
config := compiler.Config {
@@ -284,13 +350,18 @@ func main() {
284
350
fmt .Fprintln (os .Stderr , "error:" , err )
285
351
os .Exit (1 )
286
352
}
287
- case "flash" :
353
+ case "flash" , "gdb" :
288
354
if * outpath != "" {
289
355
fmt .Fprintln (os .Stderr , "Output cannot be specified with the flash command." )
290
356
usage ()
291
357
os .Exit (1 )
292
358
}
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
+ }
294
365
if err != nil {
295
366
fmt .Fprintln (os .Stderr , "error:" , err )
296
367
os .Exit (1 )
0 commit comments