In Ubuntu-ôfbylding meitsje foar ARM "fanôf it begjin"

As de ûntwikkeling krekt begjint, is it faaks net dúdlik hokker pakketten nei de doelrootfs sille gean.

Mei oare wurden, it is te betiid om LFS, buildroot of yocto (of wat oars) te pakken, mar jo moatte al begjinne. Foar de rike (ik haw 4GB eMMC op pilot-samples) is d'r in útwei om in distribúsje oan ûntwikkelders te fersprieden wêrmei se fluch wat kinne leverje dat op it stuit ûntbrekt, en dan kinne wy ​​altyd listen mei pakketten sammelje en in list meitsje foar de doelgroep rootfs.

Dit artikel is net nij en is in ienfâldige copy-paste ynstruksje.

It doel fan it artikel is om Ubuntu rootfs te bouwen foar ARM-boards (yn myn gefal, basearre op Colibri imx7d).

It bouwen fan in byld

Wy sammelje de doelrootfs foar replikaasje.

Ubuntu Base útpakke

Wy kieze de frijlitting sels op basis fan need en ús eigen foarkar. Hjir haw ik 20 jûn.

$ 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

Kontrolearje BINFMT-stipe yn 'e kernel

As jo ​​​​in mienskiplike distribúsje hawwe, dan is d'r stipe foar BINFMT_MISC en alles is konfigureare, sa net, dan bin ik wis dat jo witte hoe't jo BINFMT-stipe yn 'e kearn kinne ynskeakelje.

Soargje derfoar dat BINFMT_MISC ynskeakele is yn 'e kernel:

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

No moatte jo de ynstellings kontrolearje:

$ 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

Jo kinne manuell registrearje mei, bygelyks, hjir binne dizze ynstruksjes.

It opsetten fan qemu statyske earm

No moatte wy in statysk gearstalde qemu-eksimplaar.

!!! OANDACHT!!!
As jo ​​​​plannen om in kontener te brûken om wat te bouwen, kontrolearje dan:
https://sourceware.org/bugzilla/show_bug.cgi?id=23960
https://bugs.launchpad.net/qemu/+bug/1805913
Dan moatte jo foar x86_64-host en earmgast de i386-ferzje fan qemu brûke:
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

Ienfâldich skript:

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

Wy bewûnderje it resultaat:

$ ./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

Litte wy gewoan foar de wille de grutte mjitte foar en nei it ynstallearjen fan de minimale (foar my) set pakketten:

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

Litte wy bywurkje:

# apt update
# apt upgrade --yes

Litte wy de pakketten ynstallearje wêryn wy ynteressearre binne:

# 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-headerbestannen en modules binne in aparte saak. Fansels sille wy de bootloader, kernel, modules, apparaatbeam fia Ubuntu net ynstallearje. Se komme fan bûten nei ús ta of wy sette se sels yn of se wurde oan ús jûn troch de boerdfabrikant, dit leit yn alle gefallen bûten it ramt fan dizze ynstruksje.

Yn guon mjitte, ferzje divergence is akseptabel, mar it is better te nimmen se út de kernel build.

# apt install --yes linux-headers-generic

Litte wy sjen wat der bard is en it kaam in protte út:

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

Ferjit net in wachtwurd yn te stellen.

De ôfbylding ynpakke

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

Derneist kinne wy ​​​​etckeeper ynstallearje mei de autopush-ynstelling

No, lit ús sizze dat wy ús gearkomste ferspraat hawwe, it wurk begon oer hoe't jo letter ferskate ferzjes fan ús systeem it bêste kinne gearstalle.

etckeeper kin ús te hulp komme.

Feiligens is in persoanlike saak:

  • jo kinne bepaalde tûken beskermje
  • generearje in unike kaai foar elk apparaat
  • útskeakelje krêft push
  • ensfh. ...
# ssh-keygen
# apt install etckeeper
# etckeeper init
# cd /etc
# git remote add origin ...

Litte wy autopush ynstelle

Wy kinne fansels tûken meitsje op it apparaat fan tefoaren (litte wy sizze dat wy in skript meitsje as in tsjinst dy't de earste kear dat it wurdt lansearre sil rinne).

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

Of wy kinne wat slimmer dwaan ...

Lazy manier

Lit ús in soarte fan unike identifier hawwe, sis it serienûmer fan 'e prosessor (as MAC - serieuze bedriuwen keapje it berik):

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

Dan kinne wy ​​it brûke foar de namme fan 'e tûke wêr't wy sille triuwe:

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

Litte wy in ienfâldich skript meitsje:

# 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

En dat is alles - nei in skoft kinne wy ​​​​nei de wizigingen sjen en in list mei pakketten meitsje foar de doelfirmware.

Oanrikkemandearre materialen

BINFMT_MISC
Kernel-stipe foar ferskate binêre formaten (binfmt_misc)
Kompilearjen mei qemu-brûker chroot
Bouwe Ubuntu rootfs foar ARM
Hoe kinne jo in oanpaste Ubuntu live fanôf it begjin meitsje
Crossdev qemu-statyske-brûker-chroot
etckeeper

getdents64 probleem

readdir () jout NULL (errno = EOVERFLOW) werom foar 32-bit brûker-statyske qemu op 64-bit host
Ext4 64 bit hash brekt 32 bit glibc 2.28+
compiler_id_detection mislearret foar armhf by it brûken fan QEMU brûker-modus emulaasje
CMake wurket net goed ûnder qemu-earm

Boarne: www.habr.com