草庐IT

Qemu模拟ARM Linux平台

Leo-夜空 2023-04-22 原文

文章目录

Qemu运行ARM内核

参考

Qemu搭建ARM平台(SD卡存根文件系统)

Busybox 使用总结

《奔跑吧!linux内核》

下载安装QEMU

安装交叉编译工具:
sudo apt install gcc-arm-linux-gnueabi

下载

从网址下载:download.qemu.org
解压:tar -xvf qemu-7.2.0.tar.xz

或者sudo apt install qemu-system-arm

安装

安装依赖库和软件包:

 sudo apt-get install zlib1g-dev 
 sudo apt-get install libglib2.0-0
 sudo apt-get install libglib2.0-dev
 sudo apt-get install libtool
 sudo apt-get install libsdl1.2-dev
 sudo apt-get install autoconf

进入目录,创建build

 cd qemu-7.2.0/
 mkdir build
 cd build/

配置、编译、安装

 ../configure --target-list=arm-softmmu --audio-drv-list=
 make
 make install

ERROR: Cannot find Ninja时:sudo apt-get install ninja-build
ERROR: meson setup failed时:sudo apt-get install libpixman-1-dev

查看安装结果

命令含义
qemu-system-arm --version查看qemu版本
qemu-system-arm -M help查看支持的ARM开发板

编译内核

qemu-system-arm -M help命令列出的Versatile Express是ARM公司提供,现选择改平台学习

下载

网址下载:https://www.kernel.org/

或者github稳定版仓库:git clone git@github.com:gregkh/linux.git
(https://github.com/gregkh/linux)

配置:

设置环境变量

export ARCH=arm
export CROSS_COMPILE=arm-linux-gnueabi-

或者:
在Makefile中添加:

ARCH=arm 
CROSS_COMPILE=arm-linux-gnueabi-

make vexpress_defconfig O=./object

make menuconfig -j4 O=./object

编译内核:

生成.config文件
make vexpress_defconfig
编译
make zImage -j4
编译驱动
make modules -j4
编译设备树
make dtbs

生成了编译文件:

arch/arm/boot/zImage
arch/arm/boot/dts/vexpress-v2p-ca9.dtb
 cp arch/arm/boot/zImage ~/qemu
 cp arch/arm/boot/dts/vexpress-v2p-ca9.dtb ~/qemu

Qemu试运行

在qemu目录:

qemu-system-arm -M vexpress-a9 -smp 4 -m 1024M -kernel ./zImage -dtb ./vexpress-v2p-ca9.dtb -nographic -append "rdinit=/linuxrc console=ttyAMA0 loglevel=8"
    -M          指定开发板
    -m          指定内存大小
    -kernel     指定内核文件
    -dtb        指定dtb文件
    -nographic  指定不需要图形界面
    -append     指定扩展显示界面,串口或者LCD

出现Kernel panic是因为没有挂载rootfs

制作根文件系统

下载busybox

网址下载:https://busybox.net/downloads/

配置

设置环境变量

export ARCH=arm
export CROSS_COMPILE=arm-linux-gnueabi-

make manuconfig
进入manuconfig后,配置成静态编译

Busybox Settings --->
    Build Option --->
        [*]Build Busybox as a static binary(no shared libs)

make
make install
出现如下信息表示编译成功:

--------------------------------------------------
You will probably need to make your busybox binary
setuid root to ensure all configured applets will
work properly.
--------------------------------------------------

编译完成之后,生成的目标文件默认在./_install目录,这个目标文件目录就是下边要制作根文件系统需要用到的工具:

$ ls _install/
bin  linuxrc  sbin  usr

或者直接使用CONFIG_PREFIX指定安装目录:make CONFIG_PREFIX=/.../rootfs/ install

安装根文件系统

复制_install目录到linux kernel目录
进入_install目录,执行:

mkdir etc
mkdir dev
mkdir mnt
mkdir -p etc/init.d/

在etc/init.d/新建rcS文件,写入:

mkdir -p /proc 
mkdir -p /tmp 
mkdir -p /sys 
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,新建fstab,写入:

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

在etc,新建inittab,写入:

::sysinit:/etc/init.d/rcS
::respawn:-/bin/sh
::askfirst:-/bin/sh
::ctrlaltdel:/bin/umount -a -r

在dev目录新建设备节点:sudo mknod console c 5 1 sudo mknod null c 1 3

编译内核

cd linux-4.13.304
设置环境变量

export ARCH=arm
export CROSS_COMPILE=arm-linux-gnueabi-

make vexpress_defconfig
make manuconfig

  1. 配置initramfs,在inittramfs source file 中填入_install
  2. 清空Default kernel command string
  3. 配置memory split3G/1G user/kernel split
  4. 打开高端内存
General setup --->
    [*]Initial RAM filesystem and RAM disk(initramfs/initrd) support
     (_install)Initramfs source file(s)
Boot options -->
    ()Default kernel command string
Kernel Features --->
    Memory split (3G/1G user/kernel split) ---> 
    [*]High Memory Support

编译kernel:
make
make zImage -j4
make dtbs

Qemu启动

cp arch/arm/boot/zImage ~/qemu
cp arch/arm/boot/dts/vexpress-v2p-ca9.dtb ~/qemu
qemu-system-arm -M vexpress-a9 -smp 4 -m 1024M -kernel ./zImage -dtb ./vexpress-v2p-ca9.dtb -nographic -append "rdinit=/linuxrc console=ttyAMA0 loglevel=8"

启动的命令太长,可以编成脚本:

#! /bin/sh
qemu-system-arm -M vexpress-a9 \
-smp 4 \
-m 1024M \
-kernel ./zImage \
-dtb ./vexpress-v2p-ca9.dtb \
-nographic \
-append "rdinit=/linuxrc console=ttyAMA0 loglevel=8"

退出Qemu

在另一个终端关闭QEMU平台:killall qemu-system-arm

或者:

放开Ctrl + A
再按X

使用gdb到Qemu调试内核

下载ARM gdb:
sudo apt install gdb-multiarch

旧版sudo apt-get install gdb-arm-none-eabi

确保编译的内核包含调试信息:

Kernel hacking --->
    Compile-time checks and compiler options --->
        [*] Compile the kernel with debug info

重新编译内核,使用如下命令开启模拟:

qemu-system-arm -nographic -M vexpress-a9 -m 1024M -kernel ./zImage -append "rdinit=/linuxrc console=ttyAMA0 loglevel=8" -dtb ./vexpress-v2p-ca9.dtb -S -s
  • -S:表示 OEMU 虚拟机会冻结 CPU,直到远程的 GDB输入相应控制命令。
  • -s:表示在 1234 端口接受 GDB的调试连接。

在另外一个超级终端中启动 ARM GDB。在内核源码根目录下:
gdb-multiarch --tui vmlinux
输入c,进入调试

# 通过1234端口连接到qemu
target remote localhost:1234
# 在内核的start_kernel处设置断点
b start_kernel
c

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-lOXNVtEc-1675492657489)(https://note.youdao.com/yws/api/personal/file/WEB5a79a4608fbe7fdebbf8c2096a4ef84c?method=download&shareKey=1b1e96ae4ce7e1745b77844ae797d298)]

Qemu运行ARMv8平台

Qemu运行ARMv8平台

安装sudo apt install gcc-aarch64-linux-gnu

# 修改busybox和linux kernel的Makefile
ARCH=arm64
CROSS_COMPILE=aarch64-linux-gnu-

在busybox:

执行
make menuconfig  #确保为静态编译
make install 
# cp _install ../linux-4.14.304 -rf

在linux kernel路径:

make menuconfig #编译配置,参考下述内容

# 其他配置和arm配置一样修改。
Kernel Features -->
	Page size (4KB) --->
		Virtual address space size (48-bit) --->    # 原39-bit

make -j4

启动Qemu:
qemu-system-aarch64 -machine virt -cpu cortex-a57 -machine type=virt -nographic -m 2048 -smp 2 -kernel arch/arm64/boot/Image --append "rdinit=/linuxrc console=ttyAMA0"

有关Qemu模拟ARM Linux平台的更多相关文章

  1. ruby - 如何模拟 Net::HTTP::Post? - 2

    是的,我知道最好使用webmock,但我想知道如何在RSpec中模拟此方法:defmethod_to_testurl=URI.parseurireq=Net::HTTP::Post.newurl.pathres=Net::HTTP.start(url.host,url.port)do|http|http.requestreq,foo:1endresend这是RSpec:let(:uri){'http://example.com'}specify'HTTPcall'dohttp=mock:httpNet::HTTP.stub!(:start).and_yieldhttphttp.shou

  2. ruby-on-rails - 在这种情况下我如何模拟一个对象?没有明显的方法可以用模拟替换对象 - 2

    假设我在Store的模型中有这个非常简单的方法:defgeocode_addressloc=Store.geocode(address)self.lat=loc.latself.lng=loc.lngend如果我想编写一些不受地理编码服务影响的测试脚本,这些脚本可能已关闭、有限制或取决于我的互联网连接,我该如何模拟地理编码服务?如果我可以将地理编码对象传递到该方法中,那将很容易,但我不知道在这种情况下该怎么做。谢谢!特里斯坦 最佳答案 使用内置模拟和stub的rspecs,你可以做这样的事情:setupdo@subject=MyCl

  3. ruby - "public/protected/private"方法是如何实现的,我该如何模拟它? - 2

    在ruby中,你可以这样做:classThingpublicdeff1puts"f1"endprivatedeff2puts"f2"endpublicdeff3puts"f3"endprivatedeff4puts"f4"endend现在f1和f3是公共(public)的,f2和f4是私有(private)的。内部发生了什么,允许您调用一个类方法,然后更改方法定义?我怎样才能实现相同的功能(表面上是创建我自己的java之类的注释)例如...classThingfundeff1puts"hey"endnotfundeff2puts"hey"endendfun和notfun将更改以下函数定

  4. ruby - 在 RSpec 中 stub /模拟全局常量 - 2

    我有一个gem,它有一个根据Rails.env的不同行为的方法:defself.envifdefined?(Rails)Rails.envelsif...现在我想编写一个规范来测试这个代码路径。目前我是这样做的:Kernel.const_set(:Rails,nil)Rails.should_receive(:env).and_return('production')...没关系,只是感觉很丑。另一种方法是在spec_helper中声明:moduleRails;end而且效果也很好。但也许有更好的方法?理想情况下,这应该有效:rails=double('Rails')rails.sho

  5. ruby-on-rails - rspec 模拟对象属性赋值 - 2

    我有一个rspec模拟对象,一个值赋给了属性。我正在努力在我的rspec测试中满足这种期望。只是想知道语法是什么?代码:defcreate@new_campaign=AdCampaign.new(params[:new_campaign])@new_campaign.creationDate="#{Time.now.year}/#{Time.now.mon}/#{Time.now.day}"if@new_campaign.saveflash[:status]="Success"elseflash[:status]="Failed"endend测试it"shouldabletocreat

  6. ruby - 如何使用 rspec stub /模拟对命令行的调用? - 2

    我正在尝试测试命令行工具的输出。如何使用rspec来“伪造”命令行调用?执行以下操作不起作用:it"shouldcallthecommandlineandreturn'text'"do@p=Pig.new@p.should_receive(:run).with('my_command_line_tool_call').and_return('resulttext')end如何创建stub? 最佳答案 使用newmessageexpectationsyntax:规范/虚拟规范.rbrequire"dummy"describeDummy

  7. ruby - 接收 block 作为参数的模拟方法 - 2

    我有一个或多或少这样的场景classAdefinitialize(&block)b=B.new(&block)endend我正在对A类进行单元测试,我想知道B#new是否正在接收传递给A#new的block。我使用Mocha作为模拟框架。这可能吗? 最佳答案 我用Mocha和RSpec都试过了,虽然我可以通过测试,但行为不正确。从我的实验中,我得出结论,验证block是否已通过是不可能的。问题:为什么要传递一个block作为参数?block将用于什么目的?什么时候调用?也许这确实是您应该用类似的东西测试的行为:classBlockP

  8. ruby - 如何模拟 Fixnum 变量的整数溢出? - 2

    我目前正在将一种算法从Java转换为Ruby,但由于Ruby中缺少整数溢出,我遇到了一些障碍。假设我的值为2663860877,它大于最大整数2147483648。在Java中,它环绕,我应该得到-1631106419。我找到了这段代码,但它似乎不起作用:defforce_overflow(i)ifi2147483647i&0xffffffffelseiendend并且'ing变量不会像您期望的那样强制它为负。 最佳答案 假设32位整数具有二进制补码负数,这应该可行:defforce_overflow_signed(i)force_

  9. ruby - 单元测试 ruby​​ 命令行应用程序的代码 - 如何模拟/传递 ARGV - 2

    我有一个命令行应用程序,它使用thor来处理选项的解析。我想使用test-unit和/或minitest针对代码对命令行功能进行单元测试。我似乎无法弄清楚如何确保ARGV数组(通常会保存命令行中的选项)保存我的测试选项,以便它们可以根据代码进行测试。具体应用代码:#myapp/commands/build.rbrequire'thor'moduleMyappmoduleCommands#DefinebuildcommandsforMyAppcommandlineclassBuild:test_unit#Definesourcerootofapplicationdefself.sourc

  10. sql - 如何模拟ActiveRecord Model.count.to_sql - 2

    我想显示一个计数中使用的SQL。但是,Model.count.to_sql将不起作用,因为count返回一个没有to_sql方法的FixNum。我认为最简单的解决方案是这样做:Model.where(nil).to_sql.sub(/SELECT.*FROM/,"SELECTCOUNT(*)FROM")这会创建与Model.count中使用的SQL相同的SQL,但它是否会导致进一步的问题?例如,如果我添加一个复杂的where子句和一些连接。有更好的方法吗? 最佳答案 你可以试试Model.select("count(*)asmode

随机推荐