Ubuntu պատկերի ստեղծում ARM-ի համար «զրոյից»

Երբ զարգացումը նոր է սկսվում, հաճախ պարզ չէ, թե որ փաթեթները կուղղվեն թիրախային rootf-ներին:

Այլ կերպ ասած, դեռ վաղ է գրավել LFS-ը, buildroot-ը կամ yocto-ն (կամ այլ բան), բայց դուք արդեն պետք է սկսել: Հարուստների համար (ես ունեմ 4 ԳԲ eMMC փորձնական նմուշների վրա) կա ելք՝ ծրագրավորողներին բաժանելու բաշխում, որը թույլ կտա նրանց արագ մատուցել այն, ինչ ներկայումս բացակայում է, և այնուհետև մենք միշտ կարող ենք հավաքել փաթեթների ցուցակները և ստեղծել ցուցակ թիրախային արմատները:

Այս հոդվածը նոր չէ և պարզ copy-paste հրահանգ է։

Հոդվածի նպատակն է ստեղծել Ubuntu rootf-ներ ARM տախտակների համար (իմ դեպքում՝ հիմնված Colibri imx7d-ի վրա):

Պատկերի կառուցում

Մենք հավաքում ենք թիրախային rootf-ները կրկնօրինակման համար:

Ubuntu Base-ի բացում

Մենք ինքներս ենք ընտրում թողարկումը՝ ելնելով կարիքից և մեր նախասիրություններից: Ահա ես տվել եմ 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

Ստուգում է BINFMT-ի աջակցությունը միջուկում

Եթե ​​ունեք ընդհանուր բաշխում, ապա կա աջակցություն BINFMT_MISC-ի համար, և ամեն ինչ կազմաձևված է, եթե ոչ, ապա վստահ եմ, որ դուք գիտեք, թե ինչպես միացնել BINFMT աջակցությունը միջուկում:

Համոզվեք, որ BINFMT_MISC-ը միացված է միջուկում՝

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

Այժմ դուք պետք է ստուգեք պարամետրերը.

$ 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

Դուք կարող եք ձեռքով գրանցվել՝ օգտագործելով, օրինակ. ահա այս հրահանգները.

Կարգավորում է ստատիկ թևը

Այժմ մեզ անհրաժեշտ է ստատիկորեն հավաքված qemu օրինակ:

!!! ՈՒՇԱԴՐՈՒԹՅՈՒՆ!!!
Եթե ​​դուք նախատեսում եք օգտագործել կոնտեյներ ինչ-որ բան կառուցելու համար, ստուգեք.
https://sourceware.org/bugzilla/show_bug.cgi?id=23960
https://bugs.launchpad.net/qemu/+bug/1805913
Այնուհետև x86_64 հաղորդավարի և arm հյուրի համար դուք պետք է օգտագործեք qemu-ի i386 տարբերակը.
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

Պարզ սցենար.

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

Մենք հիանում ենք արդյունքով.

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

Պարզապես զվարճանալու համար, եկեք չափենք չափը փաթեթների նվազագույն (ինձ համար) փաթեթը տեղադրելուց առաջ և հետո.

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

Թարմացնենք.

# apt update
# apt upgrade --yes

Եկեք տեղադրենք մեզ հետաքրքրող փաթեթները.

# 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

Միջուկի վերնագրի ֆայլերը և մոդուլները առանձին խնդիր են: Իհարկե, մենք Ubuntu-ի միջոցով չենք տեղադրի bootloader-ը, միջուկը, մոդուլները, սարքի ծառը։ Նրանք մեզ մոտ կգան դրսից, կամ մենք ինքներս կհավաքենք կամ կտրամադրվեն մեզ տախտակ արտադրողի կողմից, ամեն դեպքում սա դուրս է այս հրահանգի շրջանակներից։

Որոշ չափով տարբերակների տարբերությունը ընդունելի է, բայց ավելի լավ է դրանք վերցնել միջուկի կառուցումից։

# apt install --yes linux-headers-generic

Տեսնենք, թե ինչ եղավ և շատ բան ստացվեց.

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

Մի մոռացեք գաղտնաբառ սահմանել:

Պատկերի փաթեթավորում

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

Բացի այդ, մենք կարող ենք տեղադրել etckeeper-ը autopush պարամետրով

Դե, ենթադրենք, մենք տարածեցինք մեր ժողովը, սկսվեց աշխատանքը, թե ինչպես լավագույնս հավաքել մեր համակարգի ավելի ուշ տարբեր տարբերակները:

և այլն կարող է մեզ օգնության հասնել:

Անվտանգությունը անձնական խնդիր է.

  • դուք կարող եք պաշտպանել որոշակի ճյուղեր
  • ստեղծել եզակի բանալին յուրաքանչյուր սարքի համար
  • անջատել ուժային մղումը
  • և այլն: ...
# ssh-keygen
# apt install etckeeper
# etckeeper init
# cd /etc
# git remote add origin ...

Եկեք կարգավորենք ավտոմատ հրում

Մենք, իհարկե, կարող ենք սարքի վրա նախապես մասնաճյուղեր ստեղծել (ասենք, որ մենք ստեղծում ենք սկրիպտ կամ ծառայություն, որը կաշխատի առաջին անգամ գործարկվելուց):

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

Կամ մենք կարող ենք ավելի խելացի բան անել...

Ծույլ ճանապարհ

Եկեք ունենանք ինչ-որ եզակի նույնացուցիչ, ասենք պրոցեսորի սերիական համարը (կամ MAC - լուրջ ընկերությունները գնում են տեսականին).

կատու / 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

Այնուհետև մենք կարող ենք օգտագործել այն մասնաճյուղի անվան համար, որին մենք կհրաժարվենք.

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

Եկեք ստեղծենք պարզ սցենար.

# 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

Եվ դա բոլորն է. որոշ ժամանակ անց մենք կարող ենք դիտել փոփոխությունները և ստեղծել թիրախային որոնվածի փաթեթների ցուցակ:

Առաջարկվող նյութեր

BINFMT_MISC
Միջուկի աջակցություն տարբեր Երկուական ձևաչափերի համար (binfmt_misc)
Կազմում qemu օգտվողի chroot-ով
Ubuntu rootfs-ի կառուցում ARM-ի համար
Ինչպես զրոյից ստեղծել սովորական Ubuntu ուղիղ եթեր
Crossdev qemu-static-user-chroot
և այլն պահապան

getdents64 խնդիր

readdir() վերադարձնում է NULL (errno=EOVERFLOW) 32-բիթանոց օգտագործողի ստատիկ qemu-ի համար 64-բիթանոց հոսթում
Ext4 64 բիթանոց հեշը խախտում է 32 բիթ glibc 2.28+
compiler_id_detection-ը ձախողվում է armhf-ի համար QEMU օգտագործողի ռեժիմի էմուլյացիա օգտագործելիս
CMake-ը նորմալ չի աշխատում qemu-arm-ի տակ

Source: www.habr.com