Paglikha ng imahe ng Ubuntu para sa ARM "mula sa simula"

Kapag nagsisimula pa lang ang pag-unlad, madalas na hindi malinaw kung aling mga pakete ang mapupunta sa mga target na rootfs.

Sa madaling salita, masyadong maaga para kunin ang LFS, buildroot o yocto (o iba pa), ngunit kailangan mo nang magsimula. Para sa mga mayayaman (mayroon akong 4GB na eMMC sa mga pilot sample) mayroong isang paraan upang maipamahagi sa mga developer ang isang pamamahagi na magbibigay-daan sa kanila na mabilis na makapaghatid ng isang bagay na kasalukuyang nawawala, at pagkatapos ay maaari kaming palaging mangolekta ng mga listahan ng mga pakete at lumikha ng isang listahan para sa ang target na rootfs.

Ang artikulong ito ay hindi bago at ito ay isang simpleng copy-paste na pagtuturo.

Ang layunin ng artikulo ay upang bumuo ng Ubuntu rootfs para sa ARM boards (sa aking kaso, batay sa Colibri imx7d).

Pagbuo ng isang imahe

Binubuo namin ang mga target na rootfs para sa pagtitiklop.

Pag-unpack ng Ubuntu Base

Pinipili namin ang pagpapalaya sa aming sarili batay sa pangangailangan at sa aming sariling mga kagustuhan. Dito ako nagbigay ng 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

Sinusuri ang suporta ng BINFMT sa kernel

Kung mayroon kang isang karaniwang pamamahagi, pagkatapos ay mayroong suporta para sa BINFMT_MISC at lahat ay na-configure, kung hindi, pagkatapos ay sigurado ako na alam mo kung paano paganahin ang suporta ng BINFMT sa kernel.

Tiyaking pinagana ang BINFMT_MISC sa kernel:

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

Ngayon ay kailangan mong suriin ang mga setting:

$ 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

Maaari kang magrehistro nang manu-mano gamit ang, halimbawa, narito ang mga tagubiling ito.

Pagse-set up ng qemu static na braso

Ngayon kailangan namin ng statically assembled qemu instance.

!!! PANSIN!!!
Kung plano mong gumamit ng container para bumuo ng isang bagay, tingnan ang:
https://sourceware.org/bugzilla/show_bug.cgi?id=23960
https://bugs.launchpad.net/qemu/+bug/1805913
Pagkatapos para sa x86_64 host at arm guest kailangan mong gamitin ang i386 na bersyon ng 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

Simpleng script:

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

Hinahangaan namin ang resulta:

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

Para lang masaya, sukatin natin ang laki bago at pagkatapos i-install ang minimum (para sa akin) na hanay ng mga pakete:

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

I-update natin:

# apt update
# apt upgrade --yes

I-install natin ang mga package na interesado tayo:

# 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

Ang mga file at module ng kernel header ay isang hiwalay na usapin. Siyempre, hindi namin i-install ang bootloader, kernel, modules, device tree sa pamamagitan ng Ubuntu. Darating sila sa amin mula sa labas o kami mismo ang magtitipon sa kanila o sila ay ibibigay sa amin ng tagagawa ng board, sa anumang kaso ito ay lampas sa saklaw ng pagtuturo na ito.

Sa ilang mga lawak, ang pagkakaiba-iba ng bersyon ay katanggap-tanggap, ngunit ito ay mas mahusay na kunin ang mga ito mula sa kernel build.

# apt install --yes linux-headers-generic

Tingnan natin kung ano ang nangyari at ito ay naging marami:

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

Huwag kalimutang magtakda ng password.

Pag-iimpake ng imahe

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

Bukod pa rito, maaari tayong mag-install ng etckeeper gamit ang setting ng autopush

Buweno, sabihin nating ipinamahagi namin ang aming pagpupulong, nagsimula ang trabaho sa kung paano pinakamahusay na mag-assemble sa ibang pagkakataon sa ibang mga bersyon ng aming system.

etckeeper ay maaaring tumulong sa amin.

Ang kaligtasan ay isang personal na bagay:

  • maaari mong protektahan ang ilang mga sangay
  • bumuo ng natatanging key para sa bawat device
  • huwag paganahin ang puwersang pagtulak
  • atbp. ...
# ssh-keygen
# apt install etckeeper
# etckeeper init
# cd /etc
# git remote add origin ...

I-set up natin ang autopush

Siyempre, maaari tayong gumawa ng mga sangay sa device nang maaga (sabihin nating gumawa tayo ng script o serbisyo na tatakbo sa unang pagkakataong ilunsad ito).

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

O maaari tayong gumawa ng mas matalinong bagay...

Tamad na paraan

Magkaroon tayo ng ilang uri ng natatanging identifier, sabihin ang serial number ng processor (o MAC - binibili ng mga seryosong kumpanya ang hanay):

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

Pagkatapos ay magagamit natin ito para sa pangalan ng sangay kung saan natin itulak:

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

Gumawa tayo ng isang simpleng script:

# 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

At iyon lang - pagkatapos ng ilang sandali maaari naming tingnan ang mga pagbabago at lumikha ng isang listahan ng mga pakete para sa target na firmware.

Mga inirerekomendang materyales

BINFMT_MISC
Kernel Support para sa iba't ibang Binary Formats (binfmt_misc)
Pag-compile gamit ang qemu user chroot
Pagbuo ng Ubuntu rootfs para sa ARM
Paano lumikha ng isang custom na Ubuntu live mula sa simula
Crossdev qemu-static-user-chroot
etckeeper

getdents64 problema

readdir() ay nagbabalik ng NULL (errno=EOVERFLOW) para sa 32-bit user-static qemu sa 64-bit host
Ext4 64 bit hash break 32 bit glibc 2.28+
Nabigo ang compiler_id_detection para sa armhf kapag gumagamit ng QEMU user-mode emulation
Hindi gumagana nang maayos ang CMake sa ilalim ng qemu-arm

Pinagmulan: www.habr.com