2026-04-03 13:25:31 +09:00
|
|
|
package main
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
"fmt"
|
2026-04-07 07:43:16 +09:00
|
|
|
"log/slog"
|
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() {
|
|
|
|
|
wd, err := os.Getwd()
|
|
|
|
|
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-05 21:33:19 +09:00
|
|
|
buildDir := filepath.Join(wd, "firmware", "build")
|
2026-04-03 13:25:31 +09:00
|
|
|
|
|
|
|
|
if err := run(buildDir); 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-05 21:48:47 +09:00
|
|
|
func build(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
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
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])
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func run(buildDir string) error {
|
|
|
|
|
if err := build(buildDir); err != nil {
|
|
|
|
|
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-05 21:48:47 +09:00
|
|
|
if len(devs) < 2 {
|
|
|
|
|
return fmt.Errorf("expected 2 devices, found %d", len(devs))
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
serials := make([]string, 2)
|
|
|
|
|
errs := make([]error, 2)
|
2026-04-07 07:43:16 +09:00
|
|
|
uf2Names := []string{"picomap", "picomap_test"}
|
2026-04-05 21:48:47 +09:00
|
|
|
|
|
|
|
|
var wg sync.WaitGroup
|
|
|
|
|
for i := range 2 {
|
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 07:43:16 +09:00
|
|
|
log.Info("connecting for info")
|
2026-04-05 21:48:47 +09:00
|
|
|
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
|
|
|
|
|
}
|
|
|
|
|
serials[i] = boardSerial(info.BoardID)
|
2026-04-07 07:43:16 +09:00
|
|
|
log.Info("got info", "serial", serials[i], "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
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for i := range 2 {
|
2026-04-07 07:43:16 +09:00
|
|
|
log := slog.With("serial", serials[i])
|
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-05 21:48:47 +09:00
|
|
|
c, err := client.NewSerial(devs[i], 2*time.Second)
|
|
|
|
|
if err != nil {
|
|
|
|
|
errs[i] = err
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
err = c.PICOBOOT()
|
|
|
|
|
c.Close()
|
|
|
|
|
if err != nil {
|
2026-04-07 07:43:16 +09:00
|
|
|
errs[i] = fmt.Errorf("PICOBOOT %s: %w", serials[i], err)
|
|
|
|
|
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()
|
|
|
|
|
for _, err := range errs {
|
2026-04-03 17:41:44 +09:00
|
|
|
if err != nil {
|
2026-04-05 21:48:47 +09:00
|
|
|
return err
|
2026-04-03 16:59:11 +09:00
|
|
|
}
|
2026-04-03 13:25:31 +09:00
|
|
|
}
|
|
|
|
|
|
2026-04-06 09:19:26 +09:00
|
|
|
uf2s := []string{
|
|
|
|
|
filepath.Join(buildDir, "picomap.uf2"),
|
|
|
|
|
filepath.Join(buildDir, "picomap_test.uf2"),
|
|
|
|
|
}
|
|
|
|
|
|
2026-04-05 21:48:47 +09:00
|
|
|
for i := range 2 {
|
2026-04-07 07:43:16 +09:00
|
|
|
log := slog.With("serial", serials[i])
|
2026-04-06 17:20:13 +09:00
|
|
|
wg.Go(func() {
|
2026-04-07 07:43:16 +09:00
|
|
|
log.Info("loading", "uf2", uf2Names[i])
|
2026-04-06 17:36:41 +09:00
|
|
|
errs[i] = picotool.Load(uf2s[i], serials[i], 10*time.Second)
|
2026-04-07 07:43:16 +09:00
|
|
|
if errs[i] == nil {
|
|
|
|
|
log.Info("loaded", "uf2", uf2Names[i])
|
|
|
|
|
}
|
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 07:43:16 +09:00
|
|
|
return fmt.Errorf("[%s] load: %w", serials[i], 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
|
|
|
|
|
}
|