2026-04-03 13:25:31 +09:00
|
|
|
package main
|
|
|
|
|
|
|
|
|
|
import (
|
2026-04-07 08:37:34 +09:00
|
|
|
"encoding/hex"
|
2026-04-07 21:36:50 +09:00
|
|
|
"flag"
|
2026-04-03 13:25:31 +09:00
|
|
|
"fmt"
|
2026-04-07 07:43:16 +09:00
|
|
|
"log/slog"
|
2026-04-07 08:37:34 +09:00
|
|
|
"net"
|
2026-04-03 13:25:31 +09:00
|
|
|
"os"
|
|
|
|
|
"os/exec"
|
|
|
|
|
"path/filepath"
|
2026-04-05 21:48:47 +09:00
|
|
|
"sync"
|
2026-04-03 13:25:31 +09:00
|
|
|
"time"
|
|
|
|
|
|
2026-04-03 17:41:44 +09:00
|
|
|
"github.com/theater/picomap/lib/client"
|
2026-04-03 13:25:31 +09:00
|
|
|
"github.com/theater/picomap/lib/picotool"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
func main() {
|
2026-04-07 08:37:34 +09:00
|
|
|
if len(os.Args) < 2 {
|
|
|
|
|
slog.Error("usage: picomap <info|load|test> [args...]")
|
2026-04-03 13:25:31 +09:00
|
|
|
os.Exit(1)
|
|
|
|
|
}
|
2026-04-07 08:37:34 +09:00
|
|
|
cmd := os.Args[1]
|
|
|
|
|
args := os.Args[2:]
|
2026-04-03 13:25:31 +09:00
|
|
|
|
2026-04-07 08:37:34 +09:00
|
|
|
var err error
|
|
|
|
|
switch cmd {
|
|
|
|
|
case "info":
|
2026-04-07 21:36:50 +09:00
|
|
|
err = cmdInfo(args)
|
2026-04-07 08:37:34 +09:00
|
|
|
case "load":
|
|
|
|
|
err = cmdLoad(args)
|
2026-04-07 09:18:43 +09:00
|
|
|
case "log":
|
|
|
|
|
err = cmdLog(args)
|
2026-04-07 08:37:34 +09:00
|
|
|
case "test":
|
2026-04-11 07:25:16 +09:00
|
|
|
err = cmdTestGroup(args)
|
2026-04-07 08:37:34 +09:00
|
|
|
default:
|
2026-04-07 09:18:43 +09:00
|
|
|
slog.Error("usage: picomap <info|load|log|test> [args...]")
|
2026-04-07 08:37:34 +09:00
|
|
|
os.Exit(1)
|
|
|
|
|
}
|
|
|
|
|
if err != nil {
|
2026-04-07 07:43:16 +09:00
|
|
|
slog.Error("fatal", "err", err)
|
2026-04-03 13:25:31 +09:00
|
|
|
os.Exit(1)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2026-04-07 08:37:34 +09:00
|
|
|
type deviceResult struct {
|
|
|
|
|
dev string
|
|
|
|
|
info *client.ResponseInfo
|
|
|
|
|
err error
|
|
|
|
|
}
|
|
|
|
|
|
2026-04-07 22:30:26 +09:00
|
|
|
func printInfo(via string, from string, info *client.ResponseInfo) {
|
2026-04-07 21:36:50 +09:00
|
|
|
slog.Info("device",
|
2026-04-07 22:30:26 +09:00
|
|
|
"via", via,
|
|
|
|
|
"from", from,
|
2026-04-07 21:36:50 +09:00
|
|
|
"board_id", hex.EncodeToString(info.BoardID[:]),
|
|
|
|
|
"mac", net.HardwareAddr(info.MAC[:]).String(),
|
|
|
|
|
"ip", net.IP(info.IP[:]).String(),
|
|
|
|
|
"firmware", info.FirmwareName)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func cmdInfo(args []string) error {
|
|
|
|
|
fs := flag.NewFlagSet("info", flag.ExitOnError)
|
|
|
|
|
udpAddr := fs.String("udp", "", "connect via UDP to this IP address")
|
2026-04-07 21:44:35 +09:00
|
|
|
iface := fs.String("iface", "", "bind to this network interface (for broadcast)")
|
2026-04-07 21:36:50 +09:00
|
|
|
fs.Parse(args)
|
|
|
|
|
|
2026-04-07 22:12:20 +09:00
|
|
|
if *udpAddr == "" && *iface != "" {
|
|
|
|
|
bcast, err := client.InterfaceBroadcast(*iface)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return err
|
2026-04-07 21:44:35 +09:00
|
|
|
}
|
2026-04-07 22:12:20 +09:00
|
|
|
*udpAddr = bcast
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if *udpAddr != "" {
|
|
|
|
|
c, err := client.NewUDP(*udpAddr, *iface, 500*time.Millisecond)
|
2026-04-07 21:36:50 +09:00
|
|
|
if err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
defer c.Close()
|
2026-04-07 22:12:20 +09:00
|
|
|
infos, err := c.InfoAll()
|
2026-04-07 21:36:50 +09:00
|
|
|
if err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
2026-04-07 22:12:20 +09:00
|
|
|
if len(infos) == 0 {
|
|
|
|
|
return fmt.Errorf("no devices responded")
|
|
|
|
|
}
|
2026-04-07 22:30:26 +09:00
|
|
|
for _, r := range infos {
|
|
|
|
|
printInfo(*udpAddr, r.From, r.Value)
|
2026-04-07 22:12:20 +09:00
|
|
|
}
|
2026-04-07 21:36:50 +09:00
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
|
2026-04-07 08:37:34 +09:00
|
|
|
devs, err := client.ListSerial()
|
|
|
|
|
if err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
if len(devs) == 0 {
|
|
|
|
|
return fmt.Errorf("no devices found")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
results := make([]deviceResult, len(devs))
|
|
|
|
|
var wg sync.WaitGroup
|
|
|
|
|
for i, dev := range devs {
|
|
|
|
|
results[i].dev = dev
|
|
|
|
|
wg.Go(func() {
|
2026-04-07 22:12:20 +09:00
|
|
|
c, err := client.NewSerial(dev, 500*time.Millisecond)
|
2026-04-07 08:37:34 +09:00
|
|
|
if err != nil {
|
|
|
|
|
results[i].err = err
|
|
|
|
|
return
|
|
|
|
|
}
|
2026-04-07 22:12:20 +09:00
|
|
|
info, err := c.Info()
|
2026-04-07 08:37:34 +09:00
|
|
|
c.Close()
|
|
|
|
|
if err != nil {
|
|
|
|
|
results[i].err = err
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
results[i].info = info
|
|
|
|
|
})
|
|
|
|
|
}
|
|
|
|
|
wg.Wait()
|
|
|
|
|
|
|
|
|
|
for _, r := range results {
|
|
|
|
|
if r.err != nil {
|
|
|
|
|
slog.Error("device error", "dev", r.dev, "err", r.err)
|
|
|
|
|
continue
|
|
|
|
|
}
|
2026-04-07 22:30:26 +09:00
|
|
|
printInfo(r.dev, r.dev, r.info)
|
2026-04-07 08:37:34 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
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 buildFirmware(buildDir string) error {
|
2026-04-07 07:43:16 +09:00
|
|
|
slog.Info("configuring")
|
2026-04-05 21:33:19 +09:00
|
|
|
cmake := exec.Command("cmake", "-S", filepath.Join(filepath.Dir(buildDir)), "-B", buildDir)
|
2026-04-05 21:30:21 +09:00
|
|
|
cmake.Stdout = os.Stdout
|
|
|
|
|
cmake.Stderr = os.Stderr
|
|
|
|
|
if err := cmake.Run(); err != nil {
|
|
|
|
|
return fmt.Errorf("cmake failed: %w", err)
|
|
|
|
|
}
|
|
|
|
|
|
2026-04-07 07:43:16 +09:00
|
|
|
slog.Info("building")
|
2026-04-03 13:25:31 +09:00
|
|
|
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)
|
|
|
|
|
}
|
2026-04-05 21:48:47 +09:00
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
|
2026-04-07 09:18:43 +09:00
|
|
|
func cmdLog(_ []string) error {
|
|
|
|
|
devs, err := client.ListSerial()
|
|
|
|
|
if err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
if len(devs) == 0 {
|
|
|
|
|
return fmt.Errorf("no devices found")
|
|
|
|
|
}
|
|
|
|
|
for _, dev := range devs {
|
|
|
|
|
log := slog.With("dev", dev)
|
2026-04-07 22:12:20 +09:00
|
|
|
c, err := client.NewSerial(dev, 500*time.Millisecond)
|
2026-04-07 09:18:43 +09:00
|
|
|
if err != nil {
|
|
|
|
|
log.Error("connect error", "err", err)
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
resp, err := c.Log()
|
|
|
|
|
c.Close()
|
|
|
|
|
if err != nil {
|
|
|
|
|
log.Error("log error", "err", err)
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
if len(resp.Entries) == 0 {
|
|
|
|
|
log.Info("no debug messages")
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
for _, e := range resp.Entries {
|
|
|
|
|
log.Info("dlog", "t_us", e.TimestampUS, "msg", e.Message)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
|
2026-04-07 08:37:34 +09:00
|
|
|
func cmdLoad(args []string) error {
|
|
|
|
|
target := "all"
|
|
|
|
|
if len(args) > 0 {
|
|
|
|
|
target = args[0]
|
|
|
|
|
}
|
2026-04-05 21:48:47 +09:00
|
|
|
|
2026-04-07 08:37:34 +09:00
|
|
|
wd, err := os.Getwd()
|
|
|
|
|
if err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
buildDir := filepath.Join(wd, "firmware", "build")
|
2026-04-07 08:34:29 +09:00
|
|
|
|
2026-04-07 08:37:34 +09:00
|
|
|
if err := buildFirmware(buildDir); err != nil {
|
2026-04-05 21:48:47 +09:00
|
|
|
return err
|
|
|
|
|
}
|
2026-04-03 13:25:31 +09:00
|
|
|
|
2026-04-04 23:16:25 +09:00
|
|
|
devs, err := client.ListSerial()
|
2026-04-03 13:25:31 +09:00
|
|
|
if err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
2026-04-07 08:34:29 +09:00
|
|
|
|
|
|
|
|
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
|
|
|
|
|
}
|
2026-04-07 08:37:34 +09:00
|
|
|
switch target {
|
2026-04-07 08:34:29 +09:00
|
|
|
case "all":
|
|
|
|
|
targets = allTargets
|
|
|
|
|
case "picomap":
|
|
|
|
|
targets = allTargets[:1]
|
|
|
|
|
case "picomap_test":
|
|
|
|
|
targets = allTargets[1:]
|
|
|
|
|
default:
|
2026-04-07 08:37:34 +09:00
|
|
|
return fmt.Errorf("unknown target %q", target)
|
2026-04-05 21:48:47 +09:00
|
|
|
}
|
|
|
|
|
|
2026-04-07 08:34:29 +09:00
|
|
|
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))
|
2026-04-05 21:48:47 +09:00
|
|
|
|
|
|
|
|
var wg sync.WaitGroup
|
2026-04-07 08:34:29 +09:00
|
|
|
for i := range targets {
|
2026-04-07 07:43:16 +09:00
|
|
|
log := slog.With("dev", devs[i])
|
2026-04-06 17:20:13 +09:00
|
|
|
wg.Go(func() {
|
2026-04-07 22:12:20 +09:00
|
|
|
c, err := client.NewSerial(devs[i], 500*time.Millisecond)
|
2026-04-05 21:48:47 +09:00
|
|
|
if err != nil {
|
|
|
|
|
errs[i] = err
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
info, err := c.Info()
|
|
|
|
|
c.Close()
|
|
|
|
|
if err != nil {
|
|
|
|
|
errs[i] = err
|
|
|
|
|
return
|
|
|
|
|
}
|
2026-04-07 08:34:29 +09:00
|
|
|
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)
|
2026-04-06 17:20:13 +09:00
|
|
|
})
|
2026-04-05 21:48:47 +09:00
|
|
|
}
|
|
|
|
|
wg.Wait()
|
|
|
|
|
for i, err := range errs {
|
2026-04-03 16:59:11 +09:00
|
|
|
if err != nil {
|
2026-04-07 07:43:16 +09:00
|
|
|
return fmt.Errorf("[%s] info: %w", devs[i], err)
|
2026-04-03 13:25:31 +09:00
|
|
|
}
|
2026-04-05 21:48:47 +09:00
|
|
|
}
|
|
|
|
|
|
2026-04-07 08:34:29 +09:00
|
|
|
for i := range devices {
|
|
|
|
|
log := slog.With("serial", devices[i].serial)
|
2026-04-06 17:20:13 +09:00
|
|
|
wg.Go(func() {
|
2026-04-07 07:43:16 +09:00
|
|
|
log.Info("sending PICOBOOT")
|
2026-04-07 22:12:20 +09:00
|
|
|
c, err := client.NewSerial(devices[i].dev, 500*time.Millisecond)
|
2026-04-05 21:48:47 +09:00
|
|
|
if err != nil {
|
|
|
|
|
errs[i] = err
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
err = c.PICOBOOT()
|
|
|
|
|
c.Close()
|
|
|
|
|
if err != nil {
|
2026-04-07 08:34:29 +09:00
|
|
|
errs[i] = fmt.Errorf("PICOBOOT %s: %w", devices[i].serial, err)
|
2026-04-07 07:43:16 +09:00
|
|
|
return
|
2026-04-05 21:48:47 +09:00
|
|
|
}
|
2026-04-07 07:43:16 +09:00
|
|
|
log.Info("PICOBOOT sent")
|
2026-04-06 17:20:13 +09:00
|
|
|
})
|
2026-04-05 21:48:47 +09:00
|
|
|
}
|
|
|
|
|
wg.Wait()
|
2026-04-07 08:34:29 +09:00
|
|
|
for i, err := range errs {
|
2026-04-03 17:41:44 +09:00
|
|
|
if err != nil {
|
2026-04-07 08:34:29 +09:00
|
|
|
return fmt.Errorf("[%s] %w", devices[i].serial, err)
|
2026-04-03 16:59:11 +09:00
|
|
|
}
|
2026-04-03 13:25:31 +09:00
|
|
|
}
|
|
|
|
|
|
2026-04-07 08:37:34 +09:00
|
|
|
uf2s := make([]string, len(targets))
|
|
|
|
|
for i := range targets {
|
|
|
|
|
uf2s[i] = targets[i].uf2
|
|
|
|
|
}
|
|
|
|
|
|
2026-04-07 08:34:29 +09:00
|
|
|
for i := range devices {
|
|
|
|
|
log := slog.With("serial", devices[i].serial)
|
2026-04-06 17:20:13 +09:00
|
|
|
wg.Go(func() {
|
2026-04-07 08:34:29 +09:00
|
|
|
log.Info("loading", "uf2", devices[i].name)
|
|
|
|
|
errs[i] = picotool.Load(devices[i].uf2, devices[i].serial, 10*time.Second)
|
2026-04-07 07:43:16 +09:00
|
|
|
if errs[i] == nil {
|
2026-04-07 08:34:29 +09:00
|
|
|
log.Info("loaded", "uf2", devices[i].name)
|
2026-04-07 07:43:16 +09:00
|
|
|
}
|
2026-04-06 17:20:13 +09:00
|
|
|
})
|
2026-04-05 21:48:47 +09:00
|
|
|
}
|
|
|
|
|
wg.Wait()
|
|
|
|
|
for i, err := range errs {
|
|
|
|
|
if err != nil {
|
2026-04-07 08:34:29 +09:00
|
|
|
return fmt.Errorf("[%s] load: %w", devices[i].serial, err)
|
2026-04-05 21:48:47 +09:00
|
|
|
}
|
|
|
|
|
}
|
2026-04-03 13:25:31 +09:00
|
|
|
|
2026-04-07 07:43:16 +09:00
|
|
|
slog.Info("done")
|
2026-04-03 13:25:31 +09:00
|
|
|
return nil
|
|
|
|
|
}
|
2026-04-07 08:37:34 +09:00
|
|
|
|
2026-04-11 07:25:16 +09:00
|
|
|
func findTestDevice() (string, error) {
|
2026-04-07 08:37:34 +09:00
|
|
|
devs, err := client.ListSerial()
|
|
|
|
|
if err != nil {
|
2026-04-11 07:25:16 +09:00
|
|
|
return "", err
|
2026-04-07 08:37:34 +09:00
|
|
|
}
|
|
|
|
|
for _, dev := range devs {
|
|
|
|
|
log := slog.With("dev", dev)
|
2026-04-07 22:12:20 +09:00
|
|
|
c, err := client.NewSerial(dev, 500*time.Millisecond)
|
2026-04-07 08:37:34 +09:00
|
|
|
if err != nil {
|
|
|
|
|
log.Warn("connect error", "err", err)
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
info, err := c.Info()
|
|
|
|
|
c.Close()
|
|
|
|
|
if err != nil {
|
|
|
|
|
log.Warn("info error", "err", err)
|
|
|
|
|
continue
|
|
|
|
|
}
|
|
|
|
|
if info.FirmwareName == "picomap_test" {
|
2026-04-11 07:25:16 +09:00
|
|
|
return dev, nil
|
2026-04-07 08:37:34 +09:00
|
|
|
}
|
|
|
|
|
}
|
2026-04-11 07:25:16 +09:00
|
|
|
return "", fmt.Errorf("no picomap_test device found")
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func cmdTestGroup(args []string) error {
|
|
|
|
|
if len(args) < 1 {
|
|
|
|
|
return fmt.Errorf("usage: picomap test <list|run> [args...]")
|
|
|
|
|
}
|
|
|
|
|
switch args[0] {
|
|
|
|
|
case "list":
|
|
|
|
|
return cmdTestList(args[1:])
|
|
|
|
|
case "run":
|
|
|
|
|
return cmdTestRun(args[1:])
|
|
|
|
|
default:
|
|
|
|
|
return fmt.Errorf("usage: picomap test <list|run> [args...]")
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func cmdTestList(_ []string) error {
|
|
|
|
|
dev, err := findTestDevice()
|
|
|
|
|
if err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
c, err := client.NewSerial(dev, 10*time.Second)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
defer c.Close()
|
|
|
|
|
|
|
|
|
|
result, err := c.ListTests()
|
|
|
|
|
if err != nil {
|
|
|
|
|
return fmt.Errorf("remote: %w", err)
|
|
|
|
|
}
|
|
|
|
|
for _, name := range result.Names {
|
|
|
|
|
slog.Info("test", "dev", dev, "name", name)
|
|
|
|
|
}
|
|
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func cmdTestRun(args []string) error {
|
|
|
|
|
if len(args) < 1 {
|
|
|
|
|
return fmt.Errorf("usage: picomap test run <name>")
|
|
|
|
|
}
|
|
|
|
|
name := args[0]
|
|
|
|
|
|
|
|
|
|
dev, err := findTestDevice()
|
|
|
|
|
if err != nil {
|
|
|
|
|
return err
|
2026-04-07 08:37:34 +09:00
|
|
|
}
|
|
|
|
|
|
2026-04-11 07:25:16 +09:00
|
|
|
log := slog.With("dev", dev)
|
2026-04-07 08:37:34 +09:00
|
|
|
log.Info("running test", "name", name)
|
|
|
|
|
|
2026-04-11 07:25:16 +09:00
|
|
|
c, err := client.NewSerial(dev, 10*time.Second)
|
2026-04-07 08:37:34 +09:00
|
|
|
if err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
defer c.Close()
|
|
|
|
|
|
|
|
|
|
result, err := c.Test(name)
|
|
|
|
|
if err != nil {
|
|
|
|
|
return fmt.Errorf("remote: %w", err)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for _, msg := range result.Messages {
|
|
|
|
|
log.Info("remote", "msg", msg)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if result.Pass {
|
|
|
|
|
log.Info("PASS")
|
|
|
|
|
} else {
|
|
|
|
|
log.Error("FAIL")
|
|
|
|
|
os.Exit(1)
|
|
|
|
|
}
|
|
|
|
|
return nil
|
|
|
|
|
}
|