package main import ( "fmt" "io" "log" "os" "os/exec" "path/filepath" "runtime" "strconv" ) const ubootRev = "7c9c5c0562347dccb8ac89148784a34de402ea9e" const ubootTS = 1720131068 const trustedRepoRev = "10eb851f92acc67f7cdb955770e3bdced3026677" const ( uBootRepo = "https://github.com/u-boot/u-boot" trustedFirmwareRepo = "https://github.com/ARM-software/arm-trusted-firmware" ) func applyPatches(srcdir string) error { patches, err := filepath.Glob("*.patch") if err != nil { return err } for _, patch := range patches { log.Printf("applying patch %q", patch) f, err := os.Open(patch) if err != nil { return err } defer f.Close() cmd := exec.Command("patch", "-p1") cmd.Dir = srcdir cmd.Stdin = f cmd.Stdout = os.Stdout cmd.Stderr = os.Stderr if err := cmd.Run(); err != nil { return err } f.Close() } return nil } func compile(trustedFirmwareDir string) error { defconfig := exec.Command("make", "ARCH=arm64", "rock64-rk3328_defconfig") defconfig.Stdout = os.Stdout defconfig.Stderr = os.Stderr if err := defconfig.Run(); err != nil { return fmt.Errorf("make defconfig: %v", err) } f, err := os.OpenFile(".config", os.O_RDWR|os.O_APPEND, 0755) if err != nil { return err } if _, err := f.Write([]byte("CONFIG_CMD_SETEXPR=y\nCONFIG_CMD_SETEXPR_FMT=y\n")); err != nil { return err } if err := f.Close(); err != nil { return err } make := exec.Command("make", "-j"+strconv.Itoa(runtime.NumCPU())) make.Env = append(os.Environ(), "ARCH=arm64", "CROSS_COMPILE=aarch64-linux-gnu-", "SOURCE_DATE_EPOCH="+strconv.Itoa(ubootTS), fmt.Sprintf("BL31=%s/build/rk3328/release/bl31/bl31.elf", trustedFirmwareDir), ) make.Stdout = os.Stdout make.Stderr = os.Stderr if err := make.Run(); err != nil { return fmt.Errorf("make: %v", err) } return nil } func generateBootScr(bootCmdPath string) error { mkimage := exec.Command("./tools/mkimage", "-A", "arm", "-T", "script", "-C", "none", "-d", bootCmdPath, "boot.scr") mkimage.Env = append(os.Environ(), "ARCH=arm64", "CROSS_COMPILE=aarch64-linux-gnu-", "SOURCE_DATE_EPOCH=1600000000", ) mkimage.Stdout = os.Stdout mkimage.Stderr = os.Stderr if err := mkimage.Run(); err != nil { return fmt.Errorf("mkimage: %v", err) } return nil } func copyFile(dest, src string) error { out, err := os.Create(dest) if err != nil { return err } defer out.Close() in, err := os.Open(src) if err != nil { return err } defer in.Close() if _, err := io.Copy(out, in); err != nil { return err } st, err := in.Stat() if err != nil { return err } if err := out.Chmod(st.Mode()); err != nil { return err } return out.Close() } func main() { ubootDir, err := os.MkdirTemp("", "u-boot") if err != nil { log.Fatal(err) } trustedFirmwareDir, err := os.MkdirTemp("", "arm-trusted-firmware") if err != nil { log.Fatal(err) } for _, cmd := range [][]string{ {"git", "init"}, {"git", "remote", "add", "origin", trustedFirmwareRepo}, {"git", "fetch", "--depth=1", "origin", trustedRepoRev}, {"git", "checkout", "FETCH_HEAD"}, {"make", "SOURCE_DATE_EPOCH=1600000000", "CROSS_COMPILE=aarch64-linux-gnu-", "PLAT=rk3328"}, } { log.Printf("Running %s", cmd) cmdObj := exec.Command(cmd[0], cmd[1:]...) cmdObj.Stdout = os.Stdout cmdObj.Stderr = os.Stderr cmdObj.Dir = trustedFirmwareDir if err := cmdObj.Run(); err != nil { log.Fatal(err) } } var bootCmdPath string if p, err := filepath.Abs("boot.cmd"); err != nil { log.Fatal(err) } else { bootCmdPath = p } if err := os.Chdir(ubootDir); err != nil { log.Fatal(err) } for _, cmd := range [][]string{ {"git", "init"}, {"git", "remote", "add", "origin", uBootRepo}, {"git", "fetch", "--depth=1", "origin", ubootRev}, {"git", "checkout", "FETCH_HEAD"}, } { log.Printf("Running %s", cmd) cmdObj := exec.Command(cmd[0], cmd[1:]...) cmdObj.Stdout = os.Stdout cmdObj.Stderr = os.Stderr cmdObj.Dir = ubootDir if err := cmdObj.Run(); err != nil { log.Fatal(err) } } log.Printf("applying patches") if err := applyPatches(ubootDir); err != nil { log.Fatal(err) } log.Printf("compiling uboot") if err := compile(trustedFirmwareDir); err != nil { log.Fatal(err) } log.Printf("generating boot.scr") if err := generateBootScr(bootCmdPath); err != nil { log.Fatal(err) } for _, copyCfg := range []struct { dest, src string }{ {"boot.scr", "boot.scr"}, {"u-boot-rockchip.bin", "u-boot-rockchip.bin"}, } { if err := copyFile(filepath.Join("/tmp/buildresult", copyCfg.dest), copyCfg.src); err != nil { log.Fatal(err) } } }