171 lines
3.8 KiB
Go
171 lines
3.8 KiB
Go
package main
|
|
|
|
import (
|
|
"bufio"
|
|
"flag"
|
|
"fmt"
|
|
"io"
|
|
"io/fs"
|
|
"log"
|
|
"os"
|
|
"os/user"
|
|
"path/filepath"
|
|
"runtime"
|
|
"strconv"
|
|
"strings"
|
|
|
|
"github.com/magefile/mage/sh"
|
|
)
|
|
|
|
func main() {
|
|
if err := run(); err != nil {
|
|
log.Println(err)
|
|
|
|
os.Exit(1)
|
|
}
|
|
}
|
|
|
|
func execCmd(env map[string]string, stdout io.Writer, stderr io.Writer, cmd string, args ...string) func(args ...string) error {
|
|
return func(args2 ...string) error {
|
|
fmt.Println(cmd, args2)
|
|
_, err := sh.Exec(env, stdout, stderr, cmd, append(args, args2...)...)
|
|
return err
|
|
}
|
|
}
|
|
|
|
var kernelFolderFlag = flag.String("kernel", "./linux-sources", "folder containing the kernel to compile")
|
|
|
|
func run() error {
|
|
flag.Parse()
|
|
|
|
// TODO new version check
|
|
|
|
kernelFolder, err := filepath.Abs(*kernelFolderFlag)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
fmt.Println("[kernel]", kernelFolder)
|
|
|
|
dockerRun := execCmd(nil, os.Stdout, os.Stderr,
|
|
"docker",
|
|
"run",
|
|
"-it", // to forward interrupt signals
|
|
"--rm", // cleanup afterwards
|
|
"-v", kernelFolder+":/root/armhf",
|
|
"ghcr.io/oliverpool/crossbuild-armhf:impish-20220316",
|
|
)
|
|
// change the owner of the files inside docker to the current user
|
|
chown := func(folder string) error {
|
|
user, err := user.Current()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return dockerRun("chown", "-R", user.Uid+":"+user.Gid, folder)
|
|
}
|
|
|
|
// default raspberry pi config according to https://www.raspberrypi.com/documentation/computers/linux_kernel.html#cross-compiling-the-kernel
|
|
if err := dockerRun("make", "bcmrpi_defconfig"); err != nil {
|
|
return err
|
|
}
|
|
if err := chown(".config"); err != nil {
|
|
return err
|
|
}
|
|
|
|
// adjust config to add CONFIG_SQUASHFS
|
|
configPath := filepath.Join(kernelFolder, ".config")
|
|
err = adjustTextFile(configPath, func(line string) bool {
|
|
return strings.HasPrefix(line, "CONFIG_SQUASHFS=")
|
|
}, []string{
|
|
"CONFIG_SQUASHFS=y",
|
|
})
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// compile kernel and dtbs
|
|
if err := dockerRun("make", "zImage", "dtbs", "-j"+strconv.Itoa(runtime.NumCPU())); err != nil {
|
|
return err
|
|
}
|
|
if err := chown("arch/arm/boot"); err != nil {
|
|
return err
|
|
}
|
|
|
|
bootFolder := filepath.Join(kernelFolder, "arch", "arm", "boot")
|
|
dstFolder := filepath.Join(".", "dist")
|
|
os.RemoveAll(dstFolder) // ignore any error
|
|
if err = os.MkdirAll(dstFolder, 0755); err != nil {
|
|
return err
|
|
}
|
|
|
|
// copy and rename kernel
|
|
if err = sh.Copy(filepath.Join(dstFolder, "vmlinuz"), filepath.Join(bootFolder, "zImage")); err != nil {
|
|
return err
|
|
}
|
|
|
|
// copy dtb files
|
|
files, err := filepath.Glob(filepath.Join(bootFolder, "dts", "bcm*-rpi-*.dtb"))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
// copy config and cmdline files
|
|
files = append(files, "./gokrazy/cmdline.txt", "./gokrazy/config.txt")
|
|
for _, file := range files {
|
|
dtbName := filepath.Base(file)
|
|
if err = sh.Copy(filepath.Join(dstFolder, dtbName), file); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func adjustTextFile(path string, skipLine func(string) bool, appendLines []string) error {
|
|
b, stat, err := readFile(path) // read the whole file in memory, since we are going to overwrite it
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
dst, err := os.OpenFile(path, os.O_TRUNC|os.O_WRONLY, stat.Mode())
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer dst.Close()
|
|
w := bufio.NewWriter(dst)
|
|
|
|
for _, line := range strings.Split(string(b), "\n") {
|
|
if skipLine(line) {
|
|
continue
|
|
}
|
|
_, err = w.WriteString(line + "\n")
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
for _, line := range appendLines {
|
|
_, err = w.WriteString(line + "\n")
|
|
if err != nil {
|
|
return err
|
|
}
|
|
}
|
|
err = w.Flush()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return dst.Close()
|
|
}
|
|
|
|
func readFile(path string) ([]byte, fs.FileInfo, error) {
|
|
f, err := os.Open(path)
|
|
if err != nil {
|
|
return nil, nil, err
|
|
}
|
|
defer f.Close()
|
|
b, err := io.ReadAll(f)
|
|
if err != nil {
|
|
return b, nil, err
|
|
}
|
|
|
|
stats, err := f.Stat()
|
|
return b, stats, err
|
|
}
|