Files
picomap/cmd/load/main.go

182 lines
3.7 KiB
Go

package main
import (
"flag"
"fmt"
"log/slog"
"os"
"os/exec"
"path/filepath"
"sync"
"time"
"github.com/theater/picomap/lib/client"
"github.com/theater/picomap/lib/picotool"
)
var target = flag.String("target", "all", "which firmware to load: picomap, picomap_test, or all")
func main() {
flag.Parse()
wd, err := os.Getwd()
if err != nil {
slog.Error("fatal", "err", err)
os.Exit(1)
}
buildDir := filepath.Join(wd, "firmware", "build")
if err := run(buildDir); err != nil {
slog.Error("fatal", "err", err)
os.Exit(1)
}
}
func build(buildDir string) error {
slog.Info("configuring")
cmake := exec.Command("cmake", "-S", filepath.Join(filepath.Dir(buildDir)), "-B", buildDir)
cmake.Stdout = os.Stdout
cmake.Stderr = os.Stderr
if err := cmake.Run(); err != nil {
return fmt.Errorf("cmake failed: %w", err)
}
slog.Info("building")
cmd := exec.Command("make", "-C", buildDir)
cmd.Stdout = os.Stdout
cmd.Stderr = os.Stderr
if err := cmd.Run(); err != nil {
return fmt.Errorf("build failed: %w", err)
}
return nil
}
func boardSerial(id [8]byte) string {
return fmt.Sprintf("%02X%02X%02X%02X%02X%02X%02X%02X",
id[0], id[1], id[2], id[3], id[4], id[5], id[6], id[7])
}
type deviceInfo struct {
dev string
serial string
uf2 string
name string
}
func run(buildDir string) error {
if err := build(buildDir); err != nil {
return err
}
devs, err := client.ListSerial()
if err != nil {
return err
}
allTargets := []struct {
name string
uf2 string
}{
{"picomap", filepath.Join(buildDir, "picomap.uf2")},
{"picomap_test", filepath.Join(buildDir, "picomap_test.uf2")},
}
var targets []struct {
name string
uf2 string
}
switch *target {
case "all":
targets = allTargets
case "picomap":
targets = allTargets[:1]
case "picomap_test":
targets = allTargets[1:]
default:
return fmt.Errorf("unknown target %q", *target)
}
if len(devs) < len(targets) {
return fmt.Errorf("need %d device(s), found %d", len(targets), len(devs))
}
devices := make([]deviceInfo, len(targets))
errs := make([]error, len(targets))
var wg sync.WaitGroup
for i := range targets {
log := slog.With("dev", devs[i])
wg.Go(func() {
log.Info("connecting for info")
c, err := client.NewSerial(devs[i], 2*time.Second)
if err != nil {
errs[i] = err
return
}
info, err := c.Info()
c.Close()
if err != nil {
errs[i] = err
return
}
devices[i] = deviceInfo{
dev: devs[i],
serial: boardSerial(info.BoardID),
uf2: targets[i].uf2,
name: targets[i].name,
}
log.Info("got info", "serial", devices[i].serial, "firmware", info.FirmwareName)
})
}
wg.Wait()
for i, err := range errs {
if err != nil {
return fmt.Errorf("[%s] info: %w", devs[i], err)
}
}
for i := range devices {
log := slog.With("serial", devices[i].serial)
wg.Go(func() {
log.Info("sending PICOBOOT")
c, err := client.NewSerial(devices[i].dev, 2*time.Second)
if err != nil {
errs[i] = err
return
}
err = c.PICOBOOT()
c.Close()
if err != nil {
errs[i] = fmt.Errorf("PICOBOOT %s: %w", devices[i].serial, err)
return
}
log.Info("PICOBOOT sent")
})
}
wg.Wait()
for i, err := range errs {
if err != nil {
return fmt.Errorf("[%s] %w", devices[i].serial, err)
}
}
for i := range devices {
log := slog.With("serial", devices[i].serial)
wg.Go(func() {
log.Info("loading", "uf2", devices[i].name)
errs[i] = picotool.Load(devices[i].uf2, devices[i].serial, 10*time.Second)
if errs[i] == nil {
log.Info("loaded", "uf2", devices[i].name)
}
})
}
wg.Wait()
for i, err := range errs {
if err != nil {
return fmt.Errorf("[%s] load: %w", devices[i].serial, err)
}
}
slog.Info("done")
return nil
}