ARM üçün "sıfırdan" Ubuntu görüntüsünün yaradılması

İnkişaf yeni başlayanda, hansı paketlərin hədəf köklərə gedəcəyi çox vaxt aydın olmur.

Başqa sözlə, LFS, buildroot və ya yocto (və ya başqa bir şey) tutmaq üçün hələ tezdir, lakin siz artıq başlamaq lazımdır. Zənginlər üçün (pilot nümunələrdə 4GB eMMC var) hazırlayıcılara hazırda çatışmayan bir şeyi tez çatdırmağa imkan verəcək bir paylama yaymaq üçün çıxış yolu var və biz həmişə paketlərin siyahılarını toplayıb siyahı yarada bilərik. hədəf kökləri.

Bu məqalə yeni deyil və sadə surəti-yapışdırma təlimatıdır.

Məqalənin məqsədi ARM lövhələri üçün Ubuntu rootflarını qurmaqdır (mənim vəziyyətimdə Colibri imx7d əsasında).

Təsvirin qurulması

Replikasiya üçün hədəf kökləri yığırıq.

Ubuntu Bazasının qablaşdırılması

Ehtiyac və öz seçimlərimizə əsasən buraxılışı özümüz seçirik. Burada 20 verdim.

$ 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

Kerneldə BINFMT dəstəyi yoxlanılır

Əgər ümumi paylamanız varsa, onda BINFMT_MISC üçün dəstək var və hər şey konfiqurasiya olunub, əgər yoxsa, onda əminəm ki, nüvədə BINFMT dəstəyini necə aktivləşdirəcəyinizi bilirsiniz.

BINFMT_MISC-nin nüvədə aktiv olduğundan əmin olun:

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

İndi parametrləri yoxlamaq lazımdır:

$ 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

Siz əl ilə qeydiyyatdan keçə bilərsiniz, məsələn, burada bu təlimatlar var.

Qemu statik qolunun qurulması

İndi bizə statik olaraq yığılmış qemu nümunəsi lazımdır.

!!! DİQQƏT!!!
Bir şey qurmaq üçün konteynerdən istifadə etməyi planlaşdırırsınızsa, yoxlayın:
https://sourceware.org/bugzilla/show_bug.cgi?id=23960
https://bugs.launchpad.net/qemu/+bug/1805913
Sonra x86_64 host və arm qonağı üçün qemu-nun i386 versiyasından istifadə etməlisiniz:
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

Sadə 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

Nəticəni heyran edirik:

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

Sadəcə əylənmək üçün minimum (mənim üçün) paketlər dəstini quraşdırmadan əvvəl və sonra ölçüsünü ölçək:

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

Gəlin yeniləyək:

# apt update
# apt upgrade --yes

Bizi maraqlandıran paketləri quraşdıraq:

# 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 başlıq faylları və modulları ayrı bir məsələdir. Təbii ki, Ubuntu vasitəsilə yükləyicini, nüvəni, modulları, cihaz ağacını quraşdırmayacağıq. Onlar bizə kənardan gələcək və ya biz onları özümüz yığacağıq və ya lövhə istehsalçısı tərəfindən bizə veriləcək, hər halda bu, bu təlimatın əhatə dairəsindən kənardadır.

Müəyyən dərəcədə versiya fərqi məqbuldur, lakin onları nüvə quruluşundan götürmək daha yaxşıdır.

# apt install --yes linux-headers-generic

Gəlin görək nə baş verdi və çox şey oldu:

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

Parol təyin etməyi unutmayın.

Şəklin qablaşdırılması

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

Əlavə olaraq, autopush parametri ilə etckeeper quraşdıra bilərik

Deyək ki, montajımızı payladıq, sistemimizin daha sonra müxtəlif versiyalarını ən yaxşı şəkildə necə yığmaq üzərində iş başladı.

etckeeper köməyimizə gələ bilər.

Təhlükəsizlik şəxsi məsələdir:

  • müəyyən filialları qoruya bilərsiniz
  • hər bir cihaz üçün unikal açar yaradın
  • güc itələməsini söndürün
  • və s. ...
# ssh-keygen
# apt install etckeeper
# etckeeper init
# cd /etc
# git remote add origin ...

Gəlin avtopushu quraşdıraq

Biz, əlbəttə ki, əvvəlcədən cihazda filiallar yarada bilərik (tutaq ki, biz ilk dəfə işə salınan bir skript və ya xidmət hazırlayırıq).

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

Yoxsa daha ağıllı bir şey edə bilərik...

Tənbəl yol

Bir növ unikal identifikatorumuz olsun, deyək ki, prosessorun seriya nömrəsi (və ya MAC - ciddi şirkətlər çeşidi alır):

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

Sonra onu itələyəcəyimiz filialın adı üçün istifadə edə bilərik:

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

Sadə bir skript yaradaq:

# 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

Və hamısı budur - bir müddət sonra dəyişikliklərə baxa və hədəf proqram təminatı üçün paketlərin siyahısını yarada bilərik.

Tövsiyə olunan materiallar

BINFMT_MISC
Müxtəlif Binar Formatlar üçün Kernel Dəstəyi (binfmt_misc)
Qemu istifadəçi chroot ilə tərtib edilir
ARM üçün Ubuntu rootflarının qurulması
Sıfırdan fərdi Ubuntu canlısını necə yaratmaq olar
Crossdev qemu-static-user-chroot
sair

getdents64 problemi

readdir() 32-bit hostda 64-bit istifadəçi statik qemu üçün NULL (errno=EOVERFLOW) qaytarır
Ext4 64 bit hash 32 bit glibc 2.28+-ı pozur
QEMU istifadəçi rejimi emulyasiyasından istifadə edərkən compiler_id_detection armhf üçün uğursuz olur
CMake qemu-arm altında düzgün işləmir

Mənbə: www.habr.com