ARM үшін Ubuntu кескінін «нөлден» жасау

Әзірлеу енді ғана басталған кезде, қандай пакеттердің мақсатты rootfs-ке өтетіні жиі белгісіз.

Басқаша айтқанда, LFS, buildroot немесе yocto (немесе басқа нәрсе) алуға әлі ерте, бірақ қазірдің өзінде бастау керек. Байлар үшін (менің пилоттық үлгілерде 4 ГБ eMMC бар) әзірлеушілерге қазіргі уақытта жетіспейтін нәрсені жылдам жеткізуге мүмкіндік беретін дистрибуцияны таратудың жолы бар, содан кейін біз әрқашан пакеттер тізімін жинап, тізім жасай аламыз. мақсатты түбірлер.

Бұл мақала жаңа емес және қарапайым көшіру-қою нұсқаулығы.

Мақаланың мақсаты - ARM тақталары үшін Ubuntu rootfs құру (менің жағдайда Colibri imx7d негізінде).

Сурет салу

Біз репликация үшін мақсатты rootfs жинаймыз.

Ubuntu базасын орамнан шығару

Біз қажеттілік пен өз қалауымыз бойынша шығарылымды өзіміз таңдаймыз. Міне, мен 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 статикалық қолын орнату

Енді бізге статикалық жинақталған qemu данасы қажет.

!!! НАЗАР АУДАРЫҢЫЗ!!!
Бірдеңе салу үшін контейнерді пайдалануды жоспарласаңыз, мыналарды тексеріңіз:
https://sourceware.org/bugzilla/show_bug.cgi?id=23960
https://bugs.launchpad.net/qemu/+bug/1805913
Содан кейін x86_64 хосты мен қонағы үшін 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 арқылы жүктегішті, ядроны, модульдерді, құрылғы ағашын орнатпаймыз. Олар бізге сырттан келеді немесе біз оларды өзіміз жинаймыз немесе оларды бізге тақта өндірушісі береді, кез келген жағдайда бұл осы нұсқаулықтың аясынан тыс.

Белгілі бір дәрежеде нұсқалардың алшақтығы қолайлы, бірақ оларды ядро ​​құрылымынан алған дұрыс.

# 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

Бұған қоса, autopush параметрімен etckeeper орната аламыз

Ал, біз өзіміздің жинақты тараттық делік, жүйенің кейінірек әртүрлі нұсқаларын қалай жақсы жинау керектігі туралы жұмыс басталды.

etckeeper көмекке келе алады.

Қауіпсіздік жеке мәселе болып табылады:

  • белгілі бір бұтақтарды қорғай аласыз
  • әрбір құрылғы үшін бірегей кілт жасаңыз
  • күшпен итеруді өшіру
  • және т.б. ...
# ssh-keygen
# apt install etckeeper
# etckeeper init
# cd /etc
# git remote add origin ...

Автоматты түртуді реттейік

Біз, әрине, құрылғыда тармақтарды алдын ала жасай аламыз (алғаш іске қосылғанда сценарий немесе қызметті жасадық делік).

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

Немесе біз ақылды нәрсе жасай аламыз ...

Жалқау жол

Бірегей идентификатордың қандай да бір түрін алайық, айталық процессордың сериялық нөмірі (немесе MAC - ауқымды компаниялар ауқымды сатып алады):

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

Содан кейін біз оны итеретін филиалдың атауы үшін пайдалана аламыз:

# 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 көмегімен құрастыру
ARM үшін Ubuntu rootfs құру
Пайдаланушы Ubuntu тікелей нөлден қалай жасауға болады
Crossdev qemu-static-user-chroot
т.б. сақтаушы

getdents64 мәселесі

readdir() 32 биттік хосттағы 64 биттік пайдаланушы статикалық qemu үшін NULL (errno=EOVERFLOW) мәнін қайтарады
Ext4 64 бит хэш 32 бит glibc 2.28+ үзеді
QEMU пайдаланушы режимінің эмуляциясын пайдаланған кезде compiler_id_detection armhf үшін сәтсіз аяқталады
CMake qemu-arm астында дұрыс жұмыс істемейді

Ақпарат көзі: www.habr.com