Unified --usb/--udp/--iface target flags for all CLI commands

This commit is contained in:
Ian Gulliver
2026-04-11 22:33:55 +09:00
parent e3d97f4946
commit 5f2268f5e1
+201 -201
View File
@@ -9,6 +9,7 @@ import (
"os" "os"
"os/exec" "os/exec"
"path/filepath" "path/filepath"
"strings"
"sync" "sync"
"time" "time"
@@ -16,9 +17,103 @@ import (
"github.com/theater/picomap/lib/uf2" "github.com/theater/picomap/lib/uf2"
) )
type target struct {
name string
client *client.Client
}
type targetFlags struct {
usb string
udp string
iface string
}
func addTargetFlags(fs *flag.FlagSet) *targetFlags {
tf := &targetFlags{}
fs.StringVar(&tf.usb, "usb", "", "comma-separated USB serial devices")
fs.StringVar(&tf.udp, "udp", "", "comma-separated UDP IP addresses")
fs.StringVar(&tf.iface, "iface", "", "network interface for multicast discovery")
return tf
}
func (tf *targetFlags) connect(timeout time.Duration) ([]target, error) {
var targets []target
if tf.usb != "" {
for _, dev := range strings.Split(tf.usb, ",") {
dev = strings.TrimSpace(dev)
c, err := client.NewSerial(dev, timeout)
if err != nil {
return nil, fmt.Errorf("usb %s: %w", dev, err)
}
targets = append(targets, target{dev, c})
}
}
if tf.udp != "" {
for _, addr := range strings.Split(tf.udp, ",") {
addr = strings.TrimSpace(addr)
c, err := client.NewUDP(addr, tf.iface, timeout)
if err != nil {
return nil, fmt.Errorf("udp %s: %w", addr, err)
}
targets = append(targets, target{addr, c})
}
}
if tf.iface != "" && tf.udp == "" {
bcast, err := client.InterfaceBroadcast(tf.iface)
if err != nil {
return nil, err
}
c, err := client.NewUDP(bcast, tf.iface, timeout)
if err != nil {
return nil, err
}
infos, err := c.InfoAll()
c.Close()
if err != nil {
return nil, fmt.Errorf("discovery: %w", err)
}
for _, r := range infos {
uc, err := client.NewUDP(r.From, tf.iface, timeout)
if err != nil {
return nil, fmt.Errorf("udp %s: %w", r.From, err)
}
targets = append(targets, target{r.From, uc})
}
}
if tf.usb == "" && tf.udp == "" && tf.iface == "" {
devs, err := client.ListSerial()
if err != nil {
return nil, err
}
for _, dev := range devs {
c, err := client.NewSerial(dev, timeout)
if err != nil {
slog.Warn("connect error", "dev", dev, "err", err)
continue
}
targets = append(targets, target{dev, c})
}
}
if len(targets) == 0 {
return nil, fmt.Errorf("no devices found")
}
return targets, nil
}
func closeTargets(targets []target) {
for _, t := range targets {
t.client.Close()
}
}
func main() { func main() {
if len(os.Args) < 2 { if len(os.Args) < 2 {
slog.Error("usage: picomap <info|load|test> [args...]") slog.Error("usage: picomap <info|load|log|test> [args...]")
os.Exit(1) os.Exit(1)
} }
cmd := os.Args[1] cmd := os.Args[1]
@@ -44,16 +139,9 @@ func main() {
} }
} }
type deviceResult struct { func printInfo(via string, info *client.ResponseInfo) {
dev string
info *client.ResponseInfo
err error
}
func printInfo(via string, from string, info *client.ResponseInfo) {
slog.Info("device", slog.Info("device",
"via", via, "via", via,
"from", from,
"board_id", hex.EncodeToString(info.BoardID[:]), "board_id", hex.EncodeToString(info.BoardID[:]),
"mac", net.HardwareAddr(info.MAC[:]).String(), "mac", net.HardwareAddr(info.MAC[:]).String(),
"ip", net.IP(info.IP[:]).String(), "ip", net.IP(info.IP[:]).String(),
@@ -62,74 +150,52 @@ func printInfo(via string, from string, info *client.ResponseInfo) {
func cmdInfo(args []string) error { func cmdInfo(args []string) error {
fs := flag.NewFlagSet("info", flag.ExitOnError) fs := flag.NewFlagSet("info", flag.ExitOnError)
udpAddr := fs.String("udp", "", "connect via UDP to this IP address") tf := addTargetFlags(fs)
iface := fs.String("iface", "", "bind to this network interface (for broadcast)")
fs.Parse(args) fs.Parse(args)
if *udpAddr == "" && *iface != "" { targets, err := tf.connect(500*time.Millisecond)
bcast, err := client.InterfaceBroadcast(*iface)
if err != nil { if err != nil {
return err return err
} }
*udpAddr = bcast defer closeTargets(targets)
}
if *udpAddr != "" { for _, t := range targets {
c, err := client.NewUDP(*udpAddr, *iface, 500*time.Millisecond) info, err := t.client.Info()
if err != nil { if err != nil {
return err slog.Error("info error", "via", t.name, "err", err)
}
defer c.Close()
infos, err := c.InfoAll()
if err != nil {
return err
}
if len(infos) == 0 {
return fmt.Errorf("no devices responded")
}
for _, r := range infos {
printInfo(*udpAddr, r.From, r.Value)
}
return nil
}
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() {
c, err := client.NewSerial(dev, 500*time.Millisecond)
if err != nil {
results[i].err = err
return
}
info, err := c.Info()
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 continue
} }
printInfo(r.dev, r.dev, r.info) printInfo(t.name, info)
} }
return nil
}
func cmdLog(args []string) error {
fs := flag.NewFlagSet("log", flag.ExitOnError)
tf := addTargetFlags(fs)
fs.Parse(args)
targets, err := tf.connect(500*time.Millisecond)
if err != nil {
return err
}
defer closeTargets(targets)
for _, t := range targets {
log := slog.With("dev", t.name)
resp, err := t.client.Log()
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 return nil
} }
@@ -138,13 +204,6 @@ func boardSerial(id [8]byte) string {
id[0], id[1], id[2], id[3], id[4], id[5], id[6], id[7]) 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 { func buildFirmware(buildDir string) error {
slog.Info("configuring") slog.Info("configuring")
cmake := exec.Command("cmake", "-S", filepath.Join(filepath.Dir(buildDir)), "-B", buildDir) cmake := exec.Command("cmake", "-S", filepath.Join(filepath.Dir(buildDir)), "-B", buildDir)
@@ -164,45 +223,14 @@ func buildFirmware(buildDir string) error {
return nil return nil
} }
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)
c, err := client.NewSerial(dev, 500*time.Millisecond)
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
}
func cmdLoad(args []string) error { func cmdLoad(args []string) error {
fs := flag.NewFlagSet("load", flag.ExitOnError) fs := flag.NewFlagSet("load", flag.ExitOnError)
tf := addTargetFlags(fs)
dryRun := fs.Bool("dry-run", false, "parse UF2 and log operations without flashing") dryRun := fs.Bool("dry-run", false, "parse UF2 and log operations without flashing")
fs.Parse(args) fs.Parse(args)
target := "all" loadTarget := "all"
if fs.NArg() > 0 { if fs.NArg() > 0 {
target = fs.Arg(0) loadTarget = fs.Arg(0)
} }
wd, err := os.Getwd() wd, err := os.Getwd()
@@ -215,78 +243,66 @@ func cmdLoad(args []string) error {
return err return err
} }
allTargets := []struct { type firmwareTarget struct {
name string name string
uf2 string uf2 string
}{ }
allTargets := []firmwareTarget{
{"picomap", filepath.Join(buildDir, "picomap.uf2")}, {"picomap", filepath.Join(buildDir, "picomap.uf2")},
{"picomap_test", filepath.Join(buildDir, "picomap_test.uf2")}, {"picomap_test", filepath.Join(buildDir, "picomap_test.uf2")},
} }
var targets []struct { var fwTargets []firmwareTarget
name string switch loadTarget {
uf2 string
}
switch target {
case "all": case "all":
targets = allTargets fwTargets = allTargets
case "picomap": case "picomap":
targets = allTargets[:1] fwTargets = allTargets[:1]
case "picomap_test": case "picomap_test":
targets = allTargets[1:] fwTargets = allTargets[1:]
default: default:
return fmt.Errorf("unknown target %q", target) return fmt.Errorf("unknown target %q", loadTarget)
} }
devs, err := client.ListSerial() targets, err := tf.connect(5*time.Second)
if err != nil { if err != nil {
return err return err
} }
if len(devs) < len(targets) { defer closeTargets(targets)
return fmt.Errorf("need %d device(s), found %d", len(targets), len(devs))
if len(targets) < len(fwTargets) {
return fmt.Errorf("need %d device(s), found %d", len(fwTargets), len(targets))
} }
devices := make([]deviceInfo, len(targets)) type deviceInfo struct {
errs := make([]error, len(targets)) target target
serial string
var wg sync.WaitGroup fw firmwareTarget
for i := range targets {
log := slog.With("dev", devs[i])
wg.Go(func() {
c, err := client.NewSerial(devs[i], 500*time.Millisecond)
if err != nil {
errs[i] = err
return
} }
info, err := c.Info()
c.Close() devices := make([]deviceInfo, len(fwTargets))
for i, t := range targets[:len(fwTargets)] {
info, err := t.client.Info()
if err != nil { if err != nil {
errs[i] = err return fmt.Errorf("[%s] info: %w", t.name, err)
return
} }
devices[i] = deviceInfo{ devices[i] = deviceInfo{
dev: devs[i], target: t,
serial: boardSerial(info.BoardID), serial: boardSerial(info.BoardID),
uf2: targets[i].uf2, fw: fwTargets[i],
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)
} }
slog.Info("got info", "dev", t.name, "serial", devices[i].serial, "firmware", info.FirmwareName)
} }
errs := make([]error, len(devices))
var wg sync.WaitGroup
for i := range devices { for i := range devices {
log := slog.With("serial", devices[i].serial) log := slog.With("serial", devices[i].serial)
wg.Go(func() { wg.Go(func() {
log.Info("flashing", "uf2", devices[i].name) log.Info("flashing", "uf2", devices[i].fw.name)
errs[i] = flashDevice(devices[i].dev, devices[i].uf2, *dryRun, log) errs[i] = flashDevice(devices[i].target.client, devices[i].fw.uf2, *dryRun, log)
if errs[i] == nil { if errs[i] == nil {
log.Info("flashed", "uf2", devices[i].name) log.Info("flashed", "uf2", devices[i].fw.name)
} }
}) })
} }
@@ -301,7 +317,7 @@ func cmdLoad(args []string) error {
return nil return nil
} }
func flashDevice(dev, uf2Path string, dryRun bool, log *slog.Logger) error { func flashDevice(c *client.Client, uf2Path string, dryRun bool, log *slog.Logger) error {
blocks, err := uf2.Parse(uf2Path) blocks, err := uf2.Parse(uf2Path)
if err != nil { if err != nil {
return fmt.Errorf("parse uf2: %w", err) return fmt.Errorf("parse uf2: %w", err)
@@ -324,12 +340,6 @@ func flashDevice(dev, uf2Path string, dryRun bool, log *slog.Logger) error {
return nil return nil
} }
c, err := client.NewSerial(dev, 5*time.Second)
if err != nil {
return err
}
defer c.Close()
erased := make(map[uint32]bool) erased := make(map[uint32]bool)
for _, b := range blocks { for _, b := range blocks {
sector := b.Addr &^ (sectorSize - 1) sector := b.Addr &^ (sectorSize - 1)
@@ -349,29 +359,33 @@ func flashDevice(dev, uf2Path string, dryRun bool, log *slog.Logger) error {
return nil return nil
} }
func findTestDevice() (string, error) { func findTestDevice(args []string) (target, error) {
devs, err := client.ListSerial() fs := flag.NewFlagSet("test", flag.ExitOnError)
tf := addTargetFlags(fs)
fs.Parse(args)
targets, err := tf.connect(10*time.Second)
if err != nil { if err != nil {
return "", err return target{}, err
} }
for _, dev := range devs {
log := slog.With("dev", dev) for _, t := range targets {
c, err := client.NewSerial(dev, 500*time.Millisecond) info, err := t.client.Info()
if err != nil { if err != nil {
log.Warn("connect error", "err", err) t.client.Close()
continue
}
info, err := c.Info()
c.Close()
if err != nil {
log.Warn("info error", "err", err)
continue continue
} }
if info.FirmwareName == "picomap_test" { if info.FirmwareName == "picomap_test" {
return dev, nil for _, other := range targets {
if other.name != t.name {
other.client.Close()
} }
} }
return "", fmt.Errorf("no picomap_test device found") return t, nil
}
t.client.Close()
}
return target{}, fmt.Errorf("no picomap_test device found")
} }
func cmdTestGroup(args []string) error { func cmdTestGroup(args []string) error {
@@ -390,49 +404,40 @@ func cmdTestGroup(args []string) error {
} }
} }
func cmdTestList(_ []string) error { func cmdTestList(args []string) error {
dev, err := findTestDevice() t, err := findTestDevice(args)
if err != nil { if err != nil {
return err return err
} }
c, err := client.NewSerial(dev, 10*time.Second) defer t.client.Close()
if err != nil {
return err
}
defer c.Close()
result, err := c.ListTests() result, err := t.client.ListTests()
if err != nil { if err != nil {
return fmt.Errorf("remote: %w", err) return fmt.Errorf("remote: %w", err)
} }
for _, name := range result.Names { for _, name := range result.Names {
slog.Info("test", "dev", dev, "name", name) slog.Info("test", "dev", t.name, "name", name)
} }
return nil return nil
} }
func cmdTestAll(_ []string) error { func cmdTestAll(args []string) error {
dev, err := findTestDevice() t, err := findTestDevice(args)
if err != nil { if err != nil {
return err return err
} }
defer t.client.Close()
c, err := client.NewSerial(dev, 10*time.Second) list, err := t.client.ListTests()
if err != nil {
return err
}
defer c.Close()
list, err := c.ListTests()
if err != nil { if err != nil {
return fmt.Errorf("remote: %w", err) return fmt.Errorf("remote: %w", err)
} }
log := slog.With("dev", dev) log := slog.With("dev", t.name)
failed := 0 failed := 0
for _, name := range list.Names { for _, name := range list.Names {
log.Info("running test", "name", name) log.Info("running test", "name", name)
result, err := c.Test(name) result, err := t.client.Test(name)
if err != nil { if err != nil {
log.Error("error", "name", name, "err", err) log.Error("error", "name", name, "err", err)
failed++ failed++
@@ -462,21 +467,16 @@ func cmdTestRun(args []string) error {
} }
name := args[0] name := args[0]
dev, err := findTestDevice() t, err := findTestDevice(args[1:])
if err != nil { if err != nil {
return err return err
} }
defer t.client.Close()
log := slog.With("dev", dev) log := slog.With("dev", t.name)
log.Info("running test", "name", name) log.Info("running test", "name", name)
c, err := client.NewSerial(dev, 10*time.Second) result, err := t.client.Test(name)
if err != nil {
return err
}
defer c.Close()
result, err := c.Test(name)
if err != nil { if err != nil {
return fmt.Errorf("remote: %w", err) return fmt.Errorf("remote: %w", err)
} }