Qemu搭建ARM vexpress开发环境(二)----u-boot启动kernel
标签(空格分隔): Qemu ARM Linux
在上文《Qemu搭建ARM vexpress开发环境(一)》中已经简单讲述了通过Qemu模拟直接启动Linux内核,并挂载SD卡根文件系统的方法,这种方法是直接启动内核,与实际应用中ARM板的启动方式:u-boot、kernel、rootfs有些不同。现在就来讲述下通过Qemu搭建通过u-boot启动Linux内核并挂载根文件系统的方法。
1. 开发环境简述
嵌入式系统要正常运行,应该包含:u-boot、kernel、rootfs、appfs。这几部分在ARM开发板Flash上的位置关系应该类似于下图所示:
BootLoader | BootParameters | Kernel | Rootfs | Appfs |
---|
rootfs可以添加到开发板的Flash,也可以不用添加到开发板,而是部署到PC机上,开发板在运行完kernel,要挂载rootfs时,可以通过NFS网络挂载到设定好的PC机上的rootfs根文件系统上(下一篇文章《Qemu搭建ARM vexpress开发环境(三)----NFS网络根文件系统》会详细讲解NFS网络挂载根文件系统的具体操作方法)。
2. 准备u-boot
用来使用u-boot启动加载Linux内核
2.1 下载u-boot
从http://ftp.denx.de/pub/u-boot网站选择一个u-boot版本源码下载,本文中使用的是u-boot-2017.05版本。
2.2 解压u-boot
解压u-boot:
# tar -xvf u-boot-2018.09.tar.bz2
2.3 编译u-boot
直接编译
如果没有修改Makefile和config.mk文件,需要在编译时加上ARCH和CROSS_COMPILE的配置:
# make vexpress_ca9x4_defconfig ARCH=arm CROSS_COMPILE=arm-linux-gnueabi-
这种方式编译比较直接,但是在调试时,需要多次编译u-boot,参数太多会带来不便,于是可以通过以下方法,修改Makefile和config.mk文件中的ARCH和CROSS_COMPILE选项,在编译的时候就可以省去了在编译命令中添加编译配置的设置。
修改Makefile
# vim Makefile
CROSS_COMPILE = arm-linux-gnueabi-
修改config.mk
# vim config.mk
ARCH = arm
编译u-boot:
# make vexpress_ca9x4_defconfig
# make -j4
scripts/kconfig/conf --syncconfig Kconfig
CHK include/config.h
GEN include/autoconf.mk.dep
CFG u-boot.cfg
GEN include/autoconf.mk
*** Your GCC is older than 6.0 and is not supported
arch/arm/config.mk:66: recipe for target 'checkgcc6' failed
make: *** [checkgcc6] Error 1
在编译u-boot时可能会出现一些问题,只要逐个问题解决就可以编译完成了,如上所示问题,可以通过修改对应文件解决。
# vim arch/arm/config.mk
ifeq ($(CONFIG_$(SPL_)SYS_THUMB_BUILD),y)
#archprepare: checkthumb checkgcc6
archprepare: checkthumb
checkthumb:
@if test "$(call cc-name)" = "gcc" -a \
"$(call cc-version)" -lt "0404"; then \
echo -n '*** Your GCC does not produce working '; \
echo 'binaries in THUMB mode.'; \
echo '*** Your board is configured for THUMB mode.'; \
false; \
fi
#else
#archprepare: checkgcc6
endif
#checkgcc6:
# @if test "$(call cc-name)" = "gcc" -a \
# "$(call cc-version)" -lt "0600"; then \
# echo '*** Your GCC is older than 6.0 and is not supported'; \
# false; \
# fi
2.4 启动u-boot
通过Qemu启动命令启动u-boot:
# qemu-system-arm -M vexpress-a9 -m 512M -nographic -kernel tools/u-boot
U-Boot 2018.09 (Sep 24 2018 - 21:50:42 +0800)
DRAM: 512 MiB
WARNING: Caches not enabled
Flash: 128 MiB
MMC: MMC: 0
*** Warning - bad CRC, using default environment
In: serial
Out: serial
Err: serial
Net: smc911x-0
Hit any key to stop autoboot: 0
=>
至此,u-boot启动成功,下一步开始配置u-boot的内核启动参数,来启动Linux内核,并挂在根文件系统。
打印u-boot的环境变量:
=> print
arch=arm
baudrate=38400
board=vexpress
board_name=vexpress
......
scriptaddr=0x88000000
stderr=serial
stdin=serial
stdout=serial
vendor=armltd
Environment size: 4201/262140 bytes
3. 准备内核
使用u-boot引导内核镜像:
需要将内核编译为uImage格式
需要指定uImage在内存中的加载地址
编译内核时指定:make LOADADDR=? uImage -j4
在内核目录直接编译:
# make LOADADDR=0x60003000 uImage -j4
或者在Makefile中添加加载地址配置:
# vim linux/arch/arm/boot/Makefile
LOADADDR ?= 0x60003000
再编译生成uImage文件:
# make uImage -j4
4. Qemu网络功能设置
Qemu虚拟机在u-boot启动时,需要将uImage加载到内存,而uImage从哪里来?可以通过TFTP服务器下载uImage到内存指定地址。而在这之前需要通过桥接方式将网络链接到Xubuntu系统。
4.1 配置Qemu与主机的网络连接
采用桥接网络连接Host主机通信
主机内核需要支持tun/tap模块
4.2 配置Xubuntu主机
安装桥接网络依赖的两个工具:
# sudo apt install uml-utilities bridge-utils
创建tun设备文件:/dev/net/tun(一般会自动创建)
修改/etc/network/interfaces文件配置网络
# sudo vim /etc/network/interfaces
auto lo
iface lo inet loopback
auto enp0s3
auto br0
iface br0 inet dhcp
bridge_ports enp0s3
配置/etc/qemu-ifup、/etc/qemu-ifdown脚本(高版本会自动配置,不需要手动配置)
4.3 重启系统,使配置生效
# reboot
或者:
# init 6
4.4 查看Qemu的网络环境
# ifconfig
br0 Link encap:Ethernet HWaddr 08:00:27:63:e5:46
inet addr:192.168.0.105 Bcast:192.168.0.255 Mask:255.255.255.0
inet6 addr: fe80::a00:27ff:fe63:e546/64 Scope:Link
UP BROADCAST RUNNING MULTICAST MTU:1500 Metric:1
RX packets:39 errors:0 dropped:0 overruns:0 frame:0
TX packets:64 errors:0 dropped:0 overruns:0 carrier:0
collisions:0 txqueuelen:1000
RX bytes:10471 (10.4 KB) TX bytes:7534 (7.5 KB)
虚拟网口br0即Qemu虚拟机与Linux主机通讯的网口
5. 安装TFTP服务器
创建TFTP服务器,用来给Qemu模拟开发板启动uImage时,下载uImage到内存中。
1) 安装Linux主机Host的TFTP服务器工具:
# sudo apt install tftp-hpa tftpd-hpa xinetd
2) 修改配置文件,设置TFTP服务器目录:
# sudo vim /etc/default/tftpd-hpa
......
TFTP_DIRECTORY="/home/mcy/tftpboot"
......
3) Linux主机上创建tftp目录:
# mkdir /home/mcy/tftpboot
# chmod 777 /home/mcy/tftpboot
4) 重启tftp服务:
# sudo /etc/init.d/tftpd-hpa restart
6. 设置u-boot中的内核启动参数
# vim include/configs/vexpress_common.h
#define CONFIG_BOOTCOMMAND \
"tftp 0x60003000 uImage; tftp 0x60500000 vexpress-v2p-ca9.dtb; \
setenv bootargs 'root=/dev/nfs rw \
nfsroot=192.168.1.104:/home/mcy/qemu/rootfs init=/linuxrc \
ip=192.168.1.110 console=ttyAMA0'; \
bootm 0x60003000 - 0x60500000;"
#define CONFIG_IPADDR 192.168.1.110
#define CONFIG_SERVERIP 192.168.1.104
#define CONFIGN_NETMASK 255.255.255.0
启动Qemu验证
# cat boot.sh
#! /bin/sh
qemu-system-arm \
-M vexpress-a9 \
-m 512M \
-kernel u-boot \
#-dtb kernel/linux-4.4.157/arch/arm/boot/dts/vexpress-v2p-ca9.dtb \
-nographic \
-net nic,vlan=0 -net tap,vlan=0,ifname=tap0 \
#-append "root=/dev/mmcblk0 rw console=ttyAMA0" \
-sd rootfs.ext3
启动验证
=> reset
resetting ...
U-Boot 2017.05 (Jul 24 2019 - 21:52:28 +0800)
DRAM: 512 MiB
WARNING: Caches not enabled
Flash: 128 MiB
MMC: MMC: 0
*** Warning - bad CRC, using default environment
In: serial
Out: serial
Err: serial
Net: smc911x-0
Hit any key to stop autoboot: 0
smc911x: MAC 52:54:00:12:34:56
smc911x: detected LAN9118 controller
smc911x: phy initialized
smc911x: MAC 52:54:00:12:34:56
Using smc911x-0 device
TFTP from server 192.168.1.104; our IP address is 192.168.1.110
Filename 'uImage'.
Load address: 0x60003000
Loading: #################################################################
#################################################################
#################################################################
##################################################
1.3 MiB/s
done
Bytes transferred = 3581920 (36a7e0 hex)
smc911x: MAC 52:54:00:12:34:56
smc911x: MAC 52:54:00:12:34:56
smc911x: detected LAN9118 controller
smc911x: phy initialized
smc911x: MAC 52:54:00:12:34:56
Using smc911x-0 device
TFTP from server 192.168.1.104; our IP address is 192.168.1.110
Filename 'vexpress-v2p-ca9.dtb'.
Load address: 0x60500000
Loading: #
250 KiB/s
done
Bytes transferred = 14360 (3818 hex)
smc911x: MAC 52:54:00:12:34:56
## Booting kernel from Legacy Image at 60003000 ...
Image Name: Linux-4.4.157
Image Type: ARM Linux Kernel Image (uncompressed)
Data Size: 3581856 Bytes = 3.4 MiB
Load Address: 60003000
Entry Point: 60003000
Verifying Checksum ... OK
## Flattened Device Tree blob at 60500000
Booting using the fdt blob at 0x60500000
Loading Kernel Image ... OK
Loading Device Tree to 7fed4000, end 7feda817 ... OK
至此,rootfs目录下就是一个简易的根文件系统,可以将它制作成一个镜像文件,将镜像文件烧写到开发板,或者通过Qemu中的u-boot启动Linux内核后挂载到镜像文件上。也可以设置为通过NFS网络文件系统启动,参考下一篇文章《Qemu搭建ARM vexpress开发环境(三)----NFS网络根文件系统》。