Qemu 4.1linux
vexpress-ca9shell
为了减少linux内核的大小,能够把一些外设驱动编译成内核模块,可是在启动ubuntu的时候,须要读取flash,可是此时flash的驱动模块存放在ubuntu文件系统中。为了解决这样的问题,能够使用ramdisk内存文件系统,将必备的驱动模块,好比这里的flash驱动模块放到ramdisk中,当uboot引导linux时,能够将ramdisk和内核镜像先加载到内存,启动内核时,会将ramdisk在内存中的内存地址传给内核。express
当内核启动后,会先将ramdisk做为根文件系统,而后使用insmod加载flash驱动,就能够访问存放有ubuntu文件系统的块设备节点,最后再经过命令将根文件系统切换为ubuntu文件系统。ubuntu
这里为了简单起见,咱们没有把存储器驱动编译成内核模块,只是演示一下从ramdisk切到ubuntu的过程。bash
原先直接启动ubuntu使用的命令是:app
kernel_dir=./linux-4.14.13 kernel_image=${kernel_dir}/arch/arm/boot/zImage dtb_image=${kernel_dir}/arch/arm/boot/dts/vexpress-v2p-ca9.dtb sudo qemu-system-arm \ -M vexpress-a9 \ -m 1024M \ -smp 4 \ -kernel ${kernel_image} \ -append "noinitrd root=/dev/vda1 rw rootfstype=ext4 console=ttyAMA0,115200" \ -dtb ${dtb_image} \ -drive if=none,file=./ubuntu_rootfs/ubuntu.fulldisk,id=hd0 \ -device virtio-blk-device,drive=hd0 \ -nic tap \ -serial stdio
其中vda是一个虚拟磁盘,它的第一个分区中存放的是ubuntu根文件系统。oop
如今,若是使用ramdisk的话,修改启动命令:spa
kernel_dir=./linux-4.14.13 kernel_image=${kernel_dir}/arch/arm/boot/zImage dtb_image=${kernel_dir}/arch/arm/boot/dts/vexpress-v2p-ca9.dtb sudo qemu-system-arm \ -M vexpress-a9 \ -m 1024M \ -smp 4 \ -kernel ${kernel_image} \ -dtb ${dtb_image} \ -drive if=none,file=./ubuntu_rootfs/ubuntu.fulldisk,id=hd0 \ -device virtio-blk-device,drive=hd0 \ -append "root=/dev/ram0 rw rootfstype=ext4 console=ttyAMA0 init=/linuxrc ignore_loglevel" \ -initrd ./rootfs/ramdisk.img \ -nic tap \ -serial stdio
为此,须要修改ramdisk的内容,原先我用的ramdisk中的根文件系统使用busybox编译生成的,/linuxrc是指向/bin/busybox的软链接。须要对此进行修改,修改后的ramdisk内容以下(为了简单明了,我仅保留必备的一些命令):code
. ├── bin │ ├── busybox │ ├── cd -> busybox │ ├── chroot -> busybox │ ├── echo -> busybox │ ├── exec -> busybox │ ├── mdev -> busybox │ ├── mkdir -> busybox │ ├── mount -> busybox │ ├── pivot_root -> busybox │ ├── sh -> busybox │ └── umount -> busybox ├── dev ├── linuxrc └── sys
其中linuxrc是一个shell脚本,内容以下:blog
#!/bin/sh echo "hello world" echo "mount -t sysfs sysfs /sys" mount -t sysfs sysfs /sys echo "mdev -s" mdev -s echo "mkdir /newroot" mkdir /newroot echo "mount -t ext4 /dev/vda1 /newroot" mount -t ext4 /dev/vda1 /newroot echo "sys" umount /sys echo "cd /newroot" cd /newroot echo "mkdir -p oldroot" mkdir -p oldroot echo "pivot_root . oldroot" pivot_root . oldroot echo "exec chroot . /sbin/init <dev/console >dev/console 2>&1" exec chroot . /sbin/init <dev/console >dev/console 2>&1
若是ramdisk里有须要加载的flash驱动,须要先insmod,而后才能看到/dev/vda1。
ubuntu文件系统中/sbin/init是一个软链接,实际指向的是/lib/systemd/systemd。
制做ramdisk镜像的命令:
#!/bin/bash sudo dd if=/dev/zero of=ramdisk bs=1M count=32 sudo mkfs.ext4 -F ramdisk sudo mkdir -p tmpfs sudo mount -t ext4 ramdisk ./tmpfs/ -o loop sudo cp -raf rootfs/* tmpfs/ sudo umount tmpfs sudo gzip --best -c ramdisk > ramdisk.gz sudo mkimage -n "ramdisk" -A arm -O linux -T ramdisk -C gzip -d ramdisk.gz ramdisk.img
下面是启动log:
完。