การสร้างอิมเมจ Ubuntu สำหรับ ARM “ตั้งแต่เริ่มต้น”

เมื่อการพัฒนาเพิ่งเริ่มต้น มักไม่ชัดเจนว่าแพ็กเกจใดจะไปที่ rootf เป้าหมาย

กล่าวอีกนัยหนึ่ง ยังเร็วเกินไปที่จะคว้า LFS, buildroot หรือ yocto (หรืออย่างอื่น) แต่คุณต้องเริ่มต้นแล้ว สำหรับคนรวย (ฉันมี eMMC ขนาด 4GB ในตัวอย่างนำร่อง) มีวิธีแจกจ่ายให้กับนักพัฒนา ซึ่งจะช่วยให้พวกเขาสามารถส่งมอบสิ่งที่ขาดหายไปในปัจจุบันได้อย่างรวดเร็ว จากนั้นเราสามารถรวบรวมรายการแพ็คเกจและสร้างรายการสำหรับ rootfs เป้าหมาย

บทความนี้ไม่ใช่บทความใหม่และเป็นคำแนะนำง่ายๆ ในการคัดลอกและวาง

วัตถุประสงค์ของบทความนี้คือเพื่อสร้าง Ubuntu rootfs สำหรับบอร์ด ARM (ในกรณีของฉัน ใช้ 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 และแขกรับเชิญ 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

โชติ

สคริปต์ง่ายๆ:

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

ไฟล์ส่วนหัวและโมดูลเคอร์เนลเป็นเรื่องแยกต่างหาก แน่นอนว่าเราจะไม่ติดตั้ง bootloader, เคอร์เนล, โมดูล, แผนผังอุปกรณ์ผ่าน 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

นอกจากนี้เรายังสามารถติดตั้ง etckeeper ด้วยการตั้งค่า autopush

สมมติว่าเรากระจายการประกอบของเรา งานเริ่มต้นจากวิธีที่ดีที่สุดในการประกอบระบบเวอร์ชันต่างๆ ในภายหลังของเรา

ฯลฯkeeper สามารถเข้ามาช่วยเหลือเราได้

ความปลอดภัยเป็นเรื่องส่วนตัว:

  • คุณสามารถปกป้องบางสาขาได้
  • สร้างคีย์เฉพาะสำหรับแต่ละอุปกรณ์
  • ปิดการใช้งานการกดแรง
  • ฯลฯ ...
# 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)
คอมไพล์ด้วย chroot ผู้ใช้ qemu
การสร้าง rootfs ของ Ubuntu สำหรับ ARM
วิธีสร้าง Ubuntu แบบกำหนดเองตั้งแต่เริ่มต้น
Crossdev qemu-static-user-chroot
ผู้ดูแล ฯลฯ

ปัญหา getdents64

readdir() ส่งคืน NULL (errno=EOVERFLOW) สำหรับ qemu แบบคงที่ผู้ใช้แบบ 32 บิตบนโฮสต์ 64 บิต
แฮช Ext4 64 บิตแบ่ง glibc 32 บิต 2.28+
compiler_id_detection ล้มเหลวสำหรับ armhf เมื่อใช้การจำลองโหมดผู้ใช้ QEMU
CMake ทำงานไม่ถูกต้องภายใต้ qemu-arm

ที่มา: will.com