Files
2026-01-19 19:31:46 -08:00

259 lines
6.2 KiB
Go

package main
import (
"errors"
"fmt"
"io"
"log"
"os"
"os/exec"
"path/filepath"
"runtime"
"strconv"
"strings"
)
const (
uBootRepo = "https://github.com/u-boot/u-boot"
ubootRev = "ff498a3c5efb424accc1d825cc45cede2540ca13"
trustedFirmwareRepo = "https://github.com/ARM-software/arm-trusted-firmware"
trustedRepoRev = "6251d6ed1ffa7080edc55fa75f525e19ecf5edbd"
rkbinRepo = "https://github.com/friendlyarm/rkbin"
rkbinRev = "nanopi6"
)
func applyPatches(srcdir, t string) error {
patches, err := filepath.Glob(t + ".patches/*.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(ubootDir, trustedFirmwareDir string) error {
defconfig := exec.Command("make", "ARCH=arm64", "cm3588-nas-rk3588_defconfig")
defconfig.Stdout = os.Stdout
defconfig.Stderr = os.Stderr
defconfig.Dir = ubootDir
if err := defconfig.Run(); err != nil {
return fmt.Errorf("make defconfig: %v", err)
}
f, err := os.OpenFile(filepath.Join(ubootDir, ".config"), os.O_RDWR|os.O_APPEND, 0o755)
if err != nil {
return err
}
if _, err := f.Write([]byte(`
CONFIG_CMD_SETEXPR=y
CONFIG_CMD_SETEXPR_FMT=y
CONFIG_BOOTCOMMAND="setenv bootmeths script; bootflow scan -lb"
`)); err != nil {
return err
}
if err := f.Close(); err != nil {
return err
}
cmd := exec.Command("git", "show", "-s", "--date=unix", "--pretty=format:%ad", "HEAD")
cmd.Dir = ubootDir
output, err := cmd.Output()
if err != nil {
return fmt.Errorf("unable to get date of git repo: %w", err)
}
make := exec.Command("make", "-j"+strconv.Itoa(runtime.NumCPU()))
make.Dir = ubootDir
make.Env = append(os.Environ(),
"ARCH=arm64",
"CROSS_COMPILE=aarch64-linux-gnu-",
"SOURCE_DATE_EPOCH="+strings.TrimSpace(string(output)),
fmt.Sprintf("BL31=%s/build/rk3588/release/bl31/bl31.elf", trustedFirmwareDir),
fmt.Sprintf("ROCKCHIP_TPL=%s", "/usr/src/uboot.patches/rk3588_ddr_lp4_2112MHz_lp5_2400MHz_v1.16.bin"),
)
make.Stdout = os.Stdout
make.Stderr = os.Stderr
if err := make.Run(); err != nil {
return fmt.Errorf("make: %v", err)
}
return nil
}
func generateBootScr(ubootDir, bootCmdPath string) error {
mkimage := exec.Command(filepath.Join(ubootDir, "./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.Dir = ubootDir
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 clone(dir string, repo string, rev string) error {
err := os.Mkdir(dir, 0o777)
if err != nil && !errors.Is(err, os.ErrExist) {
log.Fatal(err)
}
var commands = [][]string{
{"git", "init"},
{"git", "remote", "add", "origin", repo},
{"git", "fetch", "--depth=1", "origin", rev},
{"git", "checkout", "FETCH_HEAD"},
}
if _, err = os.Stat(filepath.Join(dir, ".git")); err == nil {
commands = commands[2:]
}
for _, cmd := range commands {
log.Printf("Running %s", cmd)
cmdObj := exec.Command(cmd[0], cmd[1:]...)
cmdObj.Stdout = os.Stdout
cmdObj.Stderr = os.Stderr
cmdObj.Dir = dir
if err := cmdObj.Run(); err != nil {
return err
}
}
return nil
}
func compileATF() string {
trustedFirmwareDir, _ := filepath.Abs("arm-trusted-firmware")
if err := clone(trustedFirmwareDir, trustedFirmwareRepo, trustedRepoRev); err != nil {
log.Fatal("Failed to clone Trusted Firmware:", err)
}
log.Printf("applying patches")
if err := applyPatches(trustedFirmwareDir, "atf"); err != nil {
log.Fatal(err)
}
for _, cmd := range [][]string{
{"make", "SOURCE_DATE_EPOCH=1600000000", "CROSS_COMPILE=aarch64-linux-gnu-", "PLAT=rk3588"},
} {
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)
}
}
return trustedFirmwareDir
}
func downloadRKbin() string {
var err error = nil
rkbinDir, _ := filepath.Abs("rkbin")
if err = clone(rkbinDir, rkbinRepo, rkbinRev); err != nil {
log.Fatal("Failed to clone rkbin Firmware:", err)
}
log.Printf("applying patches")
if err := applyPatches(rkbinDir, "rkbin"); err != nil {
log.Fatal(err)
}
return rkbinDir
}
func indockerMain() {
srcFiles, err := filepath.Glob("/usr/_src/*")
if err != nil {
log.Fatalf("failed to find source files: %v", err)
}
for _, fileName := range srcFiles {
cmdObj := exec.Command("cp", "-r", "-t", "/usr/src", fileName)
cmdObj.Stdout = os.Stdout
cmdObj.Stderr = os.Stderr
if err := cmdObj.Run(); err != nil {
log.Fatal(err)
}
}
rkbinDir := compileATF()
bootCmdPath, _ := filepath.Abs("uboot.patches/boot.cmd")
ubootDir, _ := filepath.Abs("u-boot")
if err := clone(ubootDir, uBootRepo, ubootRev); err != nil {
log.Fatal("Failed to clone uboot repo:", err)
}
log.Printf("applying patches")
if err := applyPatches(ubootDir, "uboot"); err != nil {
log.Fatal(err)
}
log.Printf("compiling uboot")
if err := compile(ubootDir, rkbinDir); err != nil {
log.Fatal(err)
}
log.Printf("generating boot.scr")
if err := generateBootScr(ubootDir, bootCmdPath); err != nil {
log.Fatal(err)
}
for _, copyCfg := range []struct {
dest, src string
}{
{"boot.scr", "u-boot/boot.scr"},
{"u-boot-rockchip.bin", "u-boot/u-boot-rockchip.bin"},
} {
if err := copyFile(filepath.Join("/tmp/buildresult", copyCfg.dest), copyCfg.src); err != nil {
log.Fatal("indocker ", err)
}
}
}