Krijimi i një imazhi Ubuntu për ARM "nga e para"

Kur zhvillimi sapo fillon, shpesh nuk është e qartë se cilat paketa do të shkojnë në rootf-et e synuara.

Me fjalë të tjera, është shumë herët për të rrëmbyer LFS, buildroot ose yocto (ose diçka tjetër), por tashmë duhet të filloni. Për të pasurit (kam 4 GB eMMC në mostrat pilot) ka një rrugëdalje për t'u shpërndarë zhvilluesve një shpërndarje që do t'i lejojë ata të dorëzojnë shpejt diçka që aktualisht mungon, dhe më pas ne mund të mbledhim gjithmonë lista paketash dhe të krijojmë një listë për rrënjët e synuara.

Ky artikull nuk është i ri dhe është një udhëzim i thjeshtë copy-paste.

Qëllimi i artikullit është të ndërtojë Ubuntu rootf për bordet ARM (në rastin tim, bazuar në Colibri imx7d).

Ndërtimi i një imazhi

Ne mbledhim rootf-et e synuara për përsëritje.

Shpaketimi i bazës së Ubuntu

Ne e zgjedhim vetë lëshimin bazuar në nevojën dhe preferencat tona. Këtu kam dhënë 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

Kontrollimi i mbështetjes BINFMT në kernel

Nëse keni një shpërndarje të përbashkët, atëherë ka mbështetje për BINFMT_MISC dhe gjithçka është konfiguruar, nëse jo, atëherë jam i sigurt që ju e dini se si të aktivizoni mbështetjen BINFMT në kernel.

Sigurohuni që BINFMT_MISC është i aktivizuar në kernel:

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

Tani duhet të kontrolloni cilësimet:

$ 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

Ju mund të regjistroheni manualisht duke përdorur, për shembull, këtu janë këto udhëzime.

Vendosja e krahut statik qemu

Tani na duhet një shembull qemu i montuar në mënyrë statike.

!!! KUJDES!!!
Nëse planifikoni të përdorni një enë për të ndërtuar diçka, shikoni:
https://sourceware.org/bugzilla/show_bug.cgi?id=23960
https://bugs.launchpad.net/qemu/+bug/1805913
Pastaj për hostin x86_64 dhe mysafirin krah ju duhet të përdorni versionin i386 të 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

Skenar i thjeshtë:

ch-mali.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

Ne e admirojmë rezultatin:

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

Thjesht për argëtim, le të matim madhësinë para dhe pas instalimit të grupit minimal (për mua) të paketave:

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

Le të përditësojmë:

# apt update
# apt upgrade --yes

Le të instalojmë paketat që na interesojnë:

# 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

Skedarët dhe modulet e kokës së kernelit janë një çështje më vete. Natyrisht, ne nuk do të instalojmë ngarkuesin, kernelin, modulet, pemën e pajisjes nëpërmjet Ubuntu. Ata do të vijnë tek ne nga jashtë ose do t'i montojmë vetë ose do të na jepen nga prodhuesi i tabelës, në çdo rast kjo është jashtë objektit të këtij udhëzimi.

Në një farë mase, divergjenca e versionit është e pranueshme, por është më mirë t'i merrni ato nga ndërtimi i kernelit.

# apt install --yes linux-headers-generic

Le të shohim se çfarë ndodhi dhe doli shumë:

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

Mos harroni të vendosni një fjalëkalim.

Paketimi i imazhit

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

Për më tepër, ne mund të instalojmë etckeeper me cilësimin e autopushit

Epo, le të themi se shpërndamë asamblenë tonë, filloi puna për mënyrën më të mirë për të montuar më vonë versione të ndryshme të sistemit tonë.

etj mund të na vijë në ndihmë.

Siguria është një çështje personale:

  • ju mund të mbroni degë të caktuara
  • gjeneroni një çelës unik për çdo pajisje
  • çaktivizoni shtytjen e forcës
  • etj. ...
# ssh-keygen
# apt install etckeeper
# etckeeper init
# cd /etc
# git remote add origin ...

Le të konfigurojmë autopushimin

Natyrisht, ne mund të krijojmë degë në pajisje paraprakisht (le të themi se bëjmë një skript ose një shërbim që do të funksionojë herën e parë që do të lëshohet).

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

Ose mund të bëjmë diçka më të zgjuar...

Mënyra dembel

Le të kemi një lloj identifikuesi unik, le të themi numrin serial të procesorit (mirë, ose MAC - kompanitë serioze blejnë gamën):

mace / 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

Më pas mund ta përdorim për emrin e degës në të cilën do të shtyjmë:

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

Le të krijojmë një skenar të thjeshtë:

# 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

Dhe kjo është e gjitha - pas një kohe ne mund të shikojmë ndryshimet dhe të krijojmë një listë paketash për firmware-in e synuar.

Materialet e rekomanduara

BINFMT_MISC
Mbështetje e kernelit për formate të ndryshme binare (binfmt_misc)
Kompilimi me qemu user chroot
Ndërtimi i Ubuntu rootf për ARM
Si të krijoni një Ubuntu të personalizuar live nga e para
Crossdev qemu-static-user-chroot
mbajtës etj

problemi getdents64

readdir() kthen NULL (errno=EOVERFLOW) për qemu statike të përdoruesit 32-bit në host 64-bit
Ext4 hash 64 bit thyen 32 bit glibc 2.28+
compiler_id_detection dështon për armhf kur përdoret emulimi i modalitetit të përdoruesit QEMU
CMake nuk funksionon siç duhet nën qemu-arm

Burimi: www.habr.com