Oprettelse af et Ubuntu-billede til ARM "fra bunden"

Når udviklingen lige begynder, er det ofte ikke klart, hvilke pakker der går til target rootfs.

Det er med andre ord for tidligt at få fat i LFS, buildroot eller yocto (eller noget andet), men du skal allerede i gang. For de rige (jeg har 4GB eMMC på piloteksempler) er der en udvej til at distribuere en distribution til udviklere, der giver dem mulighed for hurtigt at levere noget, der mangler i øjeblikket, og så kan vi altid samle lister over pakker og oprette en liste til målroderne.

Denne artikel er ikke ny og er en simpel copy-paste instruktion.

Formålet med artiklen er at bygge Ubuntu rootfs til ARM boards (i mit tilfælde baseret på Colibri imx7d).

Opbygning af et billede

Vi samler målrootf'erne til replikering.

Udpakning af Ubuntu Base

Vi vælger selv udgivelsen ud fra behov og vores egne præferencer. Her har jeg givet 20.

$ mkdir ubuntu20
$ cd ubuntu20
$ mkdir rootfs
$ wget http://cdimage.ubuntu.com/ubuntu-base/releases/20.04/release/ubuntu-base-20.04-base-armhf.tar.gz
$ tar xf ubuntu-base-20.04-base-armhf.tar.gz -C rootfs

Kontrollerer BINFMT-understøttelse i kernen

Hvis du har en fælles distribution, så er der understøttelse af BINFMT_MISC, og alt er konfigureret, hvis ikke, så er jeg sikker på, at du ved, hvordan du aktiverer BINFMT-understøttelse i kernen.

Sørg for, at BINFMT_MISC er aktiveret i kernen:

$ zcat /proc/config.gz | grep BINFMT
CONFIG_BINFMT_ELF=y
CONFIG_COMPAT_BINFMT_ELF=y
CONFIG_BINFMT_SCRIPT=y
CONFIG_BINFMT_MISC=y

Nu skal du tjekke indstillingerne:

$ ls /proc/sys/fs/binfmt_misc
qemu-arm  register  status
$ cat /proc/sys/fs/binfmt_misc/qemu-arm
enabled
interpreter /usr/bin/qemu-arm
flags: OC
offset 0
magic 7f454c4601010100000000000000000002002800
mask ffffffffffffff00fffffffffffffffffeffffff

Du kan tilmelde dig manuelt ved hjælp af f.eks. her er disse instruktioner.

Opsætning af qemu statisk arm

Nu har vi brug for en statisk samlet qemu-instans.

!!! OPMÆRKSOMHED!!!
Hvis du planlægger at bruge en container til at bygge noget, så tjek:
https://sourceware.org/bugzilla/show_bug.cgi?id=23960
https://bugs.launchpad.net/qemu/+bug/1805913
Så for x86_64 vært og arm gæst skal du bruge i386 versionen af ​​qemu:
http://ftp.ru.debian.org/debian/pool/main/q/qemu/qemu-user-static_5.0-13_i386.deb

$ wget http://ftp.debian.org/debian/pool/main/q/qemu/qemu-user-static_5.0-13_amd64.deb
$ alient -t qemu-user-static_5.0-13_amd64.deb
# путь в rootfs и имя исполняемого файла должно совпадать с /proc/sys/fs/binfmt_misc/qemu-arm
$ mkdir qemu
$ tar xf qemu-user-static-5.0.tgz -C qemu
$ file qemu/usr/bin/qemu-arm-static
qemu/usr/bin/qemu-arm-static: ELF 64-bit LSB executable, x86-64, version 1 (GNU/Linux), statically linked, BuildID[sha1]=be45f9a321cccc5c139cc1991a4042907f9673b6, for GNU/Linux 3.2.0, stripped
$ cp qemu/usr/bin/qemu-arm-static rootfs/usr/bin/qemu-arm
$ file rootfs/usr/bin/qemu-arm
rootfs/usr/bin/qemu-arm: ELF 64-bit LSB executable, x86-64, version 1 (GNU/Linux), statically linked, BuildID[sha1]=be45f9a321cccc5c139cc1991a4042907f9673b6, for GNU/Linux 3.2.0, stripped

chroot

Simpelt script:

ch-mount.sh

#!/bin/bash

function mnt() {
    echo "MOUNTING"
    sudo mount -t proc /proc proc
    sudo mount --rbind /sys sys
    sudo mount --make-rslave sys
    sudo mount --rbind /dev dev
    sudo mount --make-rslave dev
    sudo mount -o bind /dev/pts dev/pts
    sudo chroot 
}

function umnt() {
    echo "UNMOUNTING"
    sudo umount proc
    sudo umount sys
    sudo umount dev/pts
    sudo umount dev

}

if [ "$1" == "-m" ] && [ -n "$2" ] ;
then
    mnt $1 $2
elif [ "$1" == "-u" ] && [ -n "$2" ];
then
    umnt $1 $2
else
    echo ""
    echo "Either 1'st, 2'nd or both parameters were missing"
    echo ""
    echo "1'st parameter can be one of these: -m(mount) OR -u(umount)"
    echo "2'nd parameter is the full path of rootfs directory(with trailing '/')"
    echo ""
    echo "For example: ch-mount -m /media/sdcard/"
    echo ""
    echo 1st parameter : 
    echo 2nd parameter : 
fi

Vi beundrer resultatet:

$ ./ch-mount.sh -m rootfs/
# cat /etc/os-release
NAME="Ubuntu"
VERSION="20.04 LTS (Focal Fossa)"
ID=ubuntu
ID_LIKE=debian
PRETTY_NAME="Ubuntu 20.04 LTS"
VERSION_ID="20.04"
HOME_URL="https://www.ubuntu.com/"
SUPPORT_URL="https://help.ubuntu.com/"
BUG_REPORT_URL="https://bugs.launchpad.net/ubuntu/"
PRIVACY_POLICY_URL="https://www.ubuntu.com/legal/terms-and-policies/privacy-policy"
VERSION_CODENAME=focal
UBUNTU_CODENAME=focal
# uname -a
Linux NShubin 5.5.9-gentoo-x86_64 #1 SMP PREEMPT Mon Mar 16 14:34:52 MSK 2020 armv7l armv7l armv7l GNU/Linux

Bare for sjov, lad os måle størrelsen før og efter installation af det mindste (for mig) sæt af pakker:

# du -d 0 -h / 2>/dev/null
63M     /

Lad os opdatere:

# apt update
# apt upgrade --yes

Lad os installere de pakker, vi er interesserede i:

# SYSTEMD_IGNORE_CHROOT=yes apt install --yes autoconf kmod socat ifupdown ethtool iputils-ping net-tools ssh g++ iproute2 dhcpcd5 incron ser2net udev systemd gcc minicom vim cmake make mtd-utils util-linux git strace gdb libiio-dev iiod

Kernel header filer og moduler er en separat sag. Selvfølgelig vil vi ikke installere bootloader, kerne, moduler, enhedstræet via Ubuntu. De vil komme til os udefra, eller vi samler dem selv, eller de vil blive givet til os af pladeproducenten, under alle omstændigheder er dette uden for rammerne af denne vejledning.

Til en vis grad er versionsdivergens acceptabel, men det er bedre at tage dem fra kernebygningen.

# apt install --yes linux-headers-generic

Lad os se, hvad der skete, og det viste sig en masse:

# apt clean
# du -d 0 -h / 2>/dev/null
770M    /

Glem ikke at angive en adgangskode.

Pakker billedet

$ sudo tar -C rootfs --transform "s|^./||" --numeric-owner --owner=0 --group=0 -c ./ | tar --delete ./ | gzip > rootfs.tar.gz

Derudover kan vi installere etckeeper med autopush-indstillingen

Lad os sige, at vi distribuerede vores assembly, arbejdet begyndte med, hvordan man bedst sammensætter senere forskellige versioner af vores system.

etckeeper kan komme os til hjælp.

Sikkerhed er et personligt anliggende:

  • du kan beskytte visse grene
  • generere en unik nøgle til hver enhed
  • deaktiver force push
  • etc. ...
# ssh-keygen
# apt install etckeeper
# etckeeper init
# cd /etc
# git remote add origin ...

Lad os sætte autopush op

Vi kan selvfølgelig oprette filialer på enheden på forhånd (lad os sige, at vi laver et script eller en tjeneste, der kører første gang, den lanceres).

# cat /etc/etckeeper/etckeeper.conf
PUSH_REMOTE="origin"

Eller vi kan gøre noget smartere...

doven måde

Lad os have en slags unik identifikator, f.eks. serienummeret på processoren (eller MAC - seriøse virksomheder køber rækken):

kat / proc / cpuinfo

# cat /proc/cpuinfo
processor       : 0
model name      : ARMv7 Processor rev 5 (v7l)
BogoMIPS        : 60.36
Features        : half thumb fastmult vfp edsp neon vfpv3 tls vfpv4 idiva idivt vfpd32 lpae evtstrm 
CPU implementer : 0x41
CPU architecture: 7
CPU variant     : 0x0
CPU part        : 0xc07
CPU revision    : 5

processor       : 1
model name      : ARMv7 Processor rev 5 (v7l)
BogoMIPS        : 60.36
Features        : half thumb fastmult vfp edsp neon vfpv3 tls vfpv4 idiva idivt vfpd32 lpae evtstrm 
CPU implementer : 0x41
CPU architecture: 7
CPU variant     : 0x0
CPU part        : 0xc07
CPU revision    : 5

Hardware        : Freescale i.MX7 Dual (Device Tree)
Revision        : 0000
Serial          : 06372509

Så kan vi bruge det til navnet på den gren, som vi vil skubbe til:

# cat /proc/cpuinfo | grep Serial | cut -d':' -f 2 | tr -d [:blank:]
06372509

Lad os lave et simpelt script:

# cat /etc/etckeeper/commit.d/40myown-push
#!/bin/sh
set -e

if [ "$VCS" = git ] && [ -d .git ]; then
  branch=$(cat /proc/cpuinfo | grep Serial | cut -d':' -f 2 | tr -d [:blank:])
  cd /etc/
  git push origin master:${branch}
fi

Og det er alt - efter et stykke tid kan vi se på ændringerne og oprette en liste over pakker til målfirmwaren.

Anbefalede materialer

BINFMT_MISC
Kernel Support til diverse binære formater (binfmt_misc)
Kompilere med qemu bruger chroot
Opbygning af Ubuntu rootfs til ARM
Sådan opretter du en brugerdefineret Ubuntu live fra bunden
Crossdev qemu-static-user-chroot
etc keeper

getdents64 problem

readdir() returnerer NULL (errno=EOVERFLOW) for 32-bit brugerstatisk qemu på 64-bit vært
Ext4 64 bit hash bryder 32 bit glibc 2.28+
compiler_id_detection mislykkes for armhf ved brug af QEMU brugertilstandemulering
CMake fungerer ikke korrekt under qemu-arm

Kilde: www.habr.com