0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

如何搭建一个用于学习linux的QEMU环境?

Linux阅码场 来源:CSDN博客 作者:收信的加西亚 2021-05-21 09:17 次阅读

1.开场白

环境:

处理器架构:arm64

内核源码:linux-5.9.2

ubuntu版本:20.04.1

代码阅读工具:vim+ctags+cscope

对于Linux爱好者,你是否也有这样的困扰,为了学习Linux而去购买昂贵的开发版,这大可不必,QEMU模拟器几乎可以满足你的需求,足够你去学习Linux,它能够模拟x86, arm, riscv等各种处理器架构,本文将向你呈现的不是QEMU/虚拟化的原理解读,而是如何搭建一个用于学习linux的QEMU环境,当然对于Linux内核的学习这已经足够了。

注:这篇文章是之前写的,当时最新内核版本是linux-5.9.2,不想做重复工作,现在基本上都使用linux-5.11内核讲解代码,但是qemu环境搭建基本一致,且本文暂不涉及虚拟化讲解。

2.交叉编译工具链的安装

工欲善其事必先利其器,搭建QEMU的模拟环境首先需要下载安装对应架构的交叉编译工具链(这里是arm64架构):注:有时候需要安装一些依赖,根据提示安装即可!

$ sudo apt-get install gcc-aarch64-linux-gnu `

安装完成之后查看版本说明安装完成:

$ aarch64-linux-gnu-gcc -v

Using built-in specs.

COLLECT_GCC=aarch64-linux-gnu-gcc

COLLECT_LTO_WRAPPER=/usr/lib/gcc-cross/aarch64-linux-gnu/5/lto-wrapper

Target: aarch64-linux-gnu

Configured with: 。./src/configure -v --with-pkgversion=‘Ubuntu/Linaro 5.5.0-12ubuntu1’ --with-bugurl=file:///usr/share/doc/gcc-5/README.Bugs --enable-languages=c,ada,c++,go,d,fortran,objc,obj-c++ --prefix=/usr --program-suffix=-5 --enable-shared --enable-linker-build-id --libexecdir=/usr/lib --without-included-gettext --enable-threads=posix --libdir=/usr/lib --enable-nls --with-sysroot=/ --enable-clocale=gnu --enable-libstdcxx-debug --enable-libstdcxx-time=yes --with-default-libstdcxx-abi=new --enable-gnu-unique-object --disable-libquadmath --enable-plugin --enable-default-pie --with-system-zlib --enable-multiarch --enable-fix-cortex-a53-843419 --disable-werror --enable-checking=release --build=x86_64-linux-gnu --host=x86_64-linux-gnu --target=aarch64-linux-gnu --program-prefix=aarch64-linux-gnu- --includedir=/usr/aarch64-linux-gnu/include

Thread model: posix

gcc version 5.5.0 20171010 (Ubuntu/Linaro 5.5.0-12ubuntu1)

3.QEMU的安装

下面安装我们的QEMU,当然我们可以下载QEMU的源码通过编译的方式安装,但这里我们是直接apt-get的这种方式安装:

$ sudo apt-get install qemu-system-arm

安装完成之后同样查看版本号:

$ qemu-system-aarch64 --version

QEMU emulator version 4.2.0

Copyright (c) 2003-2019 Fabrice Bellard and the QEMU Project developers

说明安装成功!

4.根文件系统制作

这一步我们将要完成的是最小的根文件系统,虽然是最小的根文件系统,但是我们还是希望能够运行动态编译的应用程序,具体的制作过程如下:

1)首先准备下载一份Busybox源码:Busybox的官方源码下载路径为:

http://busybox.net/downloads/

这里我们使用的是busybox-1.28.1:

2)解压源码

tar xvf busybox-1.28.1.tar.bz2

3)进入源码目录

cd busybox-1.28.1

4)指定工具链

export ARCH=arm64

export CROSS_COMPILE=aarch64-linux-gnu-

5)配置

make menuconfig1静态编译:

Settings ---》

[*] Build static binary (no shared libs)

6)编译安装

make

make install

7)完善其他目录结构上面我们编译安装完成之后会在busybox源码目录的_install目录下生成必要的一些文件:

$ ls

bin linuxrc sbin usr

可以看到都是一些命令相关的文件已经作为init进程的linuxrc,缺少其他的一些配置文件等目录,所以需要进一步完善。

我们来创建其他需要的目录:

mkdir dev etc lib sys proc tmp var home root mnt

我们主要需要更新etc、dev和lib目录:

1)对于etc目录的更新:

添加profile文件:

#!/bin/sh

export HOSTNAME=liebao

export USER=root

export HOME=/home

export PS1=“[$USER@$HOSTNAME W]# ”

PATH=/bin:/sbin:/usr/bin:/usr/sbin

LD_LIBRARY_PATH=/lib:/usr/lib:$LD_LIBRARY_PATH

export PATH LD_LIBRARY_PATH

可以看到我们自定义了命令提示符,cd进入了我们制定的家目录homes,导出了环境变量。

添加inittab文件:

:/etc/init.d/rcS

:-/bin/sh

:-/bin/sh

:/bin/umount -a -r

这个是init进程解析的配置文件,通过这个配置文件决定执行哪个进程,何时执行。

添加fstab文件:

#device mount-point type options dump fsck order

proc /proc proc defaults 0 0

tmpfs /tmp tmpfs defaults 0 0

sysfs /sys sysfs defaults 0 0

tmpfs /dev tmpfs defaults 0 0

debugfs /sys/kernel/debug debugfs defaults 0 0

kmod_mount /mnt 9p trans=virtio 0 0

指定挂载的文件系统。

创建init.d目录:

mkdir init.d

init.d下添加rcS文件:

mkdir -p /sys

mkdir -p /tmp

mkdir -p /proc

mkdir -p /mnt

/bin/mount -a

mkdir -p /dev/pts

mount -t devpts devpts /dev/pts

echo /sbin/mdev 》 /proc/sys/kernel/hotplug

mdev -s

到此我们etc下的文件都制作好了,目录结构如下:

$ tree

├── fstab

├── init.d

│ └── rcS

├── inittab

└── profile

1 directory, 4 files

2)制作dev下必要文件:

cd dev/

sudo mknod console c 5 1

制作lib下必要文件:为了支持动态编译的应用程序的执行,根文件系统需要支持动态库,所以我们添加arm64相关的动态库文件到lib下:

cd lib

cp /usr/aarch64-linux-gnu/lib/*.so* -a 。

对库文件进行瘦身(去除符号表和调试信息),使得库文件变小:

aarch64-linux-gnu-strip *

至此,我们的最小的根文件系统已经全部制作完成!

5.内核源码的编译

1)下载最新的Liunx内核源码:

官网下载最新的Linux-5.9.2内核:http://www.kernel.org/pub/linux/kernel/

2)解压

tar xvf linux-5.9.2.tar.xz

3)放置根文件系统到源码目录:

cd linux-5.9.2

cp 。./。./busybox-1.31.1/_install _install_arm64 -a

4)配置

添加hotplug支持:

diff --git a/arch/arm64/configs/defconfig b/arch/arm64/configs/defconfig

index 6d04b9577..f89143b69 100644

--- a/arch/arm64/configs/defconfig

+++ b/arch/arm64/configs/defconfig

@@ -1043,3 +1043,6 @@ CONFIG_DEBUG_KERNEL=y

#CONFIG_DEBUG_PREEMPT is not set

#CONFIG_FTRACE is not set

CONFIG_MEMTEST=y

+

+CONFIG_UEVENT_HELPER=y

+CONFIG_UEVENT_HELPER_PATH=“/sbin/hotplug”

添加initramfs的支持:

diff --git a/arch/arm64/configs/defconfig b/arch/arm64/configs/defconfig

index f89143b69..caa7a34bf 100644

--- a/arch/arm64/configs/defconfig

+++ b/arch/arm64/configs/defconfig

@@ -1046,3 +1046,5 @@ CONFIG_MEMTEST=y

CONFIG_UEVENT_HELPER=y

CONFIG_UEVENT_HELPER_PATH=“/sbin/hotplug”

+

+CONFIG_INITRAMFS_SOURCE=“_install_arm64”

4)编译

现在进行漫长的编译过程,编译的快慢取决于电脑的性能:

export ARCH=arm64

export CROSS_COMPILE=aarch64-linux-gnu-

make defconfig

make all -j8

6.开始体验

1)创建共享目录

$ mkdir kmodules

用于主机和qemu运行的系统进行共享文件。

2)运行qemu模拟器:

qemu-system-aarch64 -machine virt -cpu cortex-a57 -machine type=virt

-m 1024 -smp 4 -kernel arch/arm64/boot/Image --append “rdinit=/linuxrc root=/dev/vda rw console=ttyAMA0 loglevel=8” -nographic

--fsdev local,id=kmod_dev,path=$PWD/kmodules,security_model=none

-device virtio-9p-device,fsdev=kmod_dev,mount_tag=kmod_mount

其中:-cpu指定了模拟的cpu为 cortex-a57,-m 1024 指定内存大小为1G, -smp 4指定模拟的系统为4核处理器 ,-kernel指定启动的内核镜像, --append指定传递的命令行参数,后面的选项指定共享目录已经使用的9P协议。

运行起来之后就进入了跟文件系统:

Please press Enter to activate this console.

[root@liebao ]#

[root@liebao ]#

[root@liebao ]#

[root@liebao ]# uname -a

Linux (none) 5.9.2-gacff57f0a #1 SMP PREEMPT Sat Oct 31 1125 CST 2020 aarch64 GNU/Linux

可以看到内核版本是我们编译的新的Linux-5.9.2内核,到处我们成功搭建了QEMU环境来运行我们的新内核。

3)使用模拟磁盘

上面我们使用initramfs的方式将我们的根文件系统的目录直接打包到内核源码,成为了内核的一部分,当然这个时候可以操作文件,但是文件都是在内存中,系统重启就会丢失,所以下面我们使用模拟磁盘的方式来挂载根文件系统。

制作磁盘文件:

dd if=/dev/zero of=rootfs_ext4.img bs=1M count=8192

mkfs.ext4 rootfs_ext4.img

mkdir -p tmpfs

mount -t ext4 rootfs_ext4.img tmpfs/ -o loop

cp -af _install_arm64/* tmpfs/

umount tmpfs

rm -rf tmpfs

chmod 777 rootfs_ext4.img

执行qemu命令:

qemu-system-aarch64 -machine virt -cpu cortex-a57 -machine type=virt

-m 1024 -smp 4 -kernel arch/arm64/boot/Image --append “noinitrd root=/dev/vda rw console=ttyAMA0 loglevel=8” -nographic

-driver if=none,file=rootfs_ext4.img,id=hd0

-device virtio-blk-device,drive=hd0

--fsdev local,id=kmod_dev,path=$PWD/kmodules,security_model=none

-device virtio-9p-device,fsdev=kmod_dev,mount_tag=kmod_mount

我们可以发现,传递给内核的命令行参数变化了,添加了noinitrd选项,这样就会挂载我们自己的模拟磁盘。

同样能进入到最小文件系统,但是这次我们查看:

[root@liebao ]# df -mh

Filesystem Size Used Available Use% Mounted on

/dev/root 7.8G 148.5M 7.3G 2% /

devtmpfs 468.4M 0 468.4M 0% /dev

tmpfs 489.8M 0 489.8M 0% /tmp

kmod_mount 901.1G 672.5G 182.8G 79% /mnt

可以看到大小为8G的磁盘可以我们使用,而且文件的操作重启之后不会丢失!!!

4)共享文件

前面已经支持了主机和qemu上的系统共享目录,这个目录就是kmodules目录:通过mount可以查看被挂载到了qemu上的系统的/mnt目录下

在主机的内核源码目录的kmodules目录中echo一个文件:

$ echo “Hello QEMU” 》 test.txt

然后进入到我们启动qemu的内核根文件系统的/mnt目录:

[root@liebao ]# cd /mnt/

[root@liebao mnt]# ls

[root@liebao mnt]# ls

test.txt

[root@liebao mnt]#

[root@liebao mnt]# cat test.txt

Hello QEMU

可以看到我们之前写的文件,共享目录OK!

5)测试-》应用测试:我们来写一个简单的hello world应用程序,体验一下在QEMU中的系统执行:源码如下:

#include 《stdio.h》

int main(int argc, char **argv)

{

printf(“Hello World, QEMU!!!

”);

return 0;

}

然后使用交叉编译工具链动态编译:

$ aarch64-linux-gnu-gcc test.c -o test

拷贝到共享目录下:

cp test 。./。./kmodules/

在QEMU系统中,进入/mnt目录下执行:

[root@liebao mnt]# 。/test

Hello World, QEMU!!!

可以发现被成功执行了,说明模拟出来的系统可以运行应用程序,而且可以使用动态链接库!!!

-》内核模块测试:下面写一个简单的内核模块:

Makefile文件:

export ARCH=arm64

export CROSS_COMPILE=aarch64-linux-gnu-

KERNEL_DIR ?= /home/hanch/study/kernel/linux-5.9.2

obj-m := module_test.o

modules:

$(MAKE) -C $(KERNEL_DIR) M=$(PWD) modules

clean:

$(MAKE) -C $(KERNEL_DIR) M=$(PWD) clean

install:

cp *.ko $(KERNEL_DIR)/kmodules

内核模块文件module_test.c:

#include 《linux/init.h》

#include 《linux/module.h》

#include 《linux/kernel.h》

static int __init module_test_init(void)

{

printk(“module_test_init

”);

return 0;

}

static void __exit module_test_exit(void)

{

printk(“module_test_exit

”);

}

module_init(module_test_init);

module_exit(module_test_exit);

MODULE_LICENSE(“GPL”);

编译拷贝:

$ make modules

make -C /home/hanch/study/kernel/linux-5.9.2 M=/home/hanch/study/kernel/linux-5.9.2/mydriver/module_eg modules

make[1]: 进入目录“/home/hanch/study/kernel/linux-5.9.2”

CC [M] /home/hanch/study/kernel/linux-5.9.2/mydriver/module_eg/module_test.o

MODPOST /home/hanch/study/kernel/linux-5.9.2/mydriver/module_eg/Module.symvers

CC [M] /home/hanch/study/kernel/linux-5.9.2/mydriver/module_eg/module_test.mod.o

LD [M] /home/hanch/study/kernel/linux-5.9.2/mydriver/module_eg/module_test.ko

make[1]: 离开目录“/home/hanch/study/kernel/linux-5.9.2”

$ make install

cp *.ko /home/hanch/study/kernel/linux-5.9.2/kmodules

到QEMU的内核系统中,进入/mnt目录下:执行模块的插入/移除:

[root@liebao mnt]# insmod module_test.ko

[ 1406.614976] module_test_init

[root@liebao mnt]# lsmod

module_test 16384 0 - Live 0xffff800008d40000 (O)

[root@liebao mnt]# rmmod module_test

[ 1424.748114] module_test_exit

内核模块也正常运行!!!

文章到这里关于QEMU体验最新的Linux内核已经完成了,当然QEMU有多功能在此不在描述,目前配置的QEMU环境已经足够内核的学习和实际,只有大量的内核源代码+在一种体系架构的处理器上实际内核才能真正的去理解内核的实现机理,才能更好去使用内核。

原文标题:使用QEMU搭建ARM64实验环境

文章出处:【微信公众号:Linuxer】欢迎添加关注!文章转载请注明出处。

责任编辑:haq

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 处理器
    +关注

    关注

    68

    文章

    19076

    浏览量

    228679
  • ARM
    ARM
    +关注

    关注

    134

    文章

    9020

    浏览量

    366330
  • qemu
    +关注

    关注

    0

    文章

    57

    浏览量

    5318

原文标题:使用QEMU搭建ARM64实验环境

文章出处:【微信号:LinuxDev,微信公众号:Linux阅码场】欢迎添加关注!文章转载请注明出处。

收藏 人收藏

    评论

    相关推荐

    【开源鸿蒙】使用QEMU运行OpenHarmony轻量系统

    本文将会介绍如何从源码安装QEMU 6.2.0,以及如何使用QEMU运行OpenHarmony轻量系统。通过本文,你将会对QEMU和OpenHarmony轻量系统又一个初步的认知,并对
    的头像 发表于 09-14 08:51 362次阅读
    【开源鸿蒙】使用<b class='flag-5'>QEMU</b>运行OpenHarmony轻量系统

    瑞芯微RK3566鸿蒙开发板Ubuntu虚拟机环境搭建教程,触觉智能Purple Pi OH主板

    本文适用于Ubuntu虚拟机环境搭建教程学习,设备为触觉智能开发的瑞芯微RK3566开发板,型号Purple Pi OH。是Laval官方社区主荐的
    的头像 发表于 09-08 18:25 692次阅读
    瑞芯微RK3566鸿蒙开发板Ubuntu虚拟机<b class='flag-5'>环境</b><b class='flag-5'>搭建</b>教程,触觉智能Purple Pi OH主板

    瑞芯微RK3566鸿蒙开发板Ubuntu虚拟机环境搭建教程,触觉智能Purple Pi OH主板

    本文适用于Ubuntu虚拟机环境搭建教程学习,设备为触觉智能开发的瑞芯微RK3566开发板,型号Purple Pi OH。是华为Laval官方社区主荐的
    的头像 发表于 09-05 17:47 335次阅读
    瑞芯微RK3566鸿蒙开发板Ubuntu虚拟机<b class='flag-5'>环境</b><b class='flag-5'>搭建</b>教程,触觉智能Purple Pi OH主板

    pytorch环境搭建详细步骤

    PyTorch作为广泛使用的深度学习框架,其环境搭建对于从事机器学习和深度
    的头像 发表于 08-01 15:38 602次阅读

    QT5+OpenCV4搭建应用开发环境

    作为深度学习算法工程师,必须要掌握应用开发技能吗?搞工程肯定是必须要会界面开发,QT就是很不错的选择。本文以QT5.15 + OpenCV4.8 + OpenVINO2023为例,搭建
    的头像 发表于 07-23 11:23 830次阅读
    QT5+OpenCV4<b class='flag-5'>搭建</b>应用开发<b class='flag-5'>环境</b>

    PyTorch深度学习开发环境搭建指南

    PyTorch作为种流行的深度学习框架,其开发环境搭建对于深度学习研究者和开发者来说至关重要。在Windows操作系统上
    的头像 发表于 07-16 18:29 643次阅读

    开源鸿蒙 编译OpenHarmony轻量系统QEMU RISC-V版本

    本文将介绍如何为QEMU RISC-V虚拟平台构建OpenHarmony轻量系统。得益于QEMU的CPU指令集模拟执行能力,该方法可以在没有开发板的情况下调试和运行OpenHarmony系统源码。本文介绍的该方法,可以用于Ope
    的头像 发表于 07-15 10:36 924次阅读
    开源鸿蒙 编译OpenHarmony轻量系统<b class='flag-5'>QEMU</b> RISC-V版本

    虚拟机的Linux搭建环境,执行到‘设置工具’的时候总是报错,怎么处理?

    虚拟机的Linux搭建环境,执行到‘设置工具’的时候总是报错,请问这个要怎么解决? Attachments
    发表于 06-20 07:26

    云服务器的linux中如何搭建php运行环境

    要在云服务器的Linux系统上搭建PHP运行环境,您可以按照以下步骤进行: 1、安装PHP: 使用包管理工具(如apt、yum等)安装PHP及其相关模块。例如,在Ubuntu系统上,您可以运行以下
    的头像 发表于 03-21 17:08 565次阅读

    使用VeriStand搭建MIL测试环境

    MIL(Model In The Loop)模型在环仿真测试用于在实际系统搭建完成之前进行模型测试,使用VeriStand搭建MIL测试环境,可以在不需要硬件资源的情况下测试控制模型。
    的头像 发表于 01-05 10:42 3814次阅读
    使用VeriStand<b class='flag-5'>搭建</b>MIL测试<b class='flag-5'>环境</b>

    Linux环境变量配置方法

    想必大家平时工作中也会配置Linux环境变量,但是可能也仅仅是为解决某些工具的运行环境,对于Linux环境变量本身的配置学问还没深入了解。
    的头像 发表于 01-04 09:51 480次阅读

    如何快速搭建MQTT协议的测试环境

    大家好,我是麦叔,之前有小伙伴建议出期如何快速搭建MQTT协议的测试环境,因为自己写的mqtt测试工具总是有这样那样的问题。
    的头像 发表于 12-26 09:28 1895次阅读
    如何快速<b class='flag-5'>搭建</b><b class='flag-5'>一</b><b class='flag-5'>个</b>MQTT协议的测试<b class='flag-5'>环境</b>

    如何自己搭建服务器?

    如何自己搭建服务器?自己搭建服务器涉及到硬件和软件两
    的头像 发表于 12-12 16:52 3692次阅读

    java环境搭建及配置教程

    Java是种广泛使用的编程语言,用于开发各种应用程序。在开始学习和使用Java之前,您需要搭建和配置Java开发环境。本教程将提供详细的J
    的头像 发表于 12-06 15:50 723次阅读

    php运行环境搭建方法

    的操作系统是搭建PHP运行环境的第步。常见的操作系统有Windows、Linux和Mac OS X等,选择合适的操作系统可以根据个人喜好和实际需求进行选择。Windows用户可以选择
    的头像 发表于 12-04 14:56 1179次阅读