qemu模拟vexpress开发板(续)

1. 问题描述

  续接上文,利用qemu模拟cortex-a9开发板,流程:qemu ===> u-boot ===> kernel ===>  nfsroot

                                                 ||

                                                 ======> ramdisk

  进行到kernel引导根文件系统时,总是失败。经过尝试,终于解决了!!! 


 

2. 解决方法

2.1 关于bootargs

  前文说过,u-boot可以通过 bootargs 这个环境变量来传递参数给内核,那么可以传递哪些参数?格式又是如何?这个问题我找了好久,还是有些糊涂,我曾试图到u-boot的官网上去找找文档,结果登不上。。。昨天又试了一下,登上了!

  

  有提示:可以在 Documentation/initrd.txt 和 Documentation/nfsroot.txt 找到相关内容。因为我这边主要是无法挂载nfs,所以可以先看看关于这个的说明:

   按照这个说明,我们可以把参数写到u-boot中的vexpress_common.h 中,主要是IP那一项。

 1 /* Basic environment settings */
 2 #if 0
 3 #define CONFIG_BOOTCOMMAND      "run bootflash;"  //注释原有宏
 4 #endif
 5  
 6 #define CONFIG_IPADDR    192.168.222.21
 7 #define CONFIG_NETMASK   255.255.255.0
 8 #define CONFIG_SERVERIP  192.168.222.20
 9 #define CONFIG_GATEWAYIP  192.168.222.1
10  
11 /*
12 "setenv bootargs 'root=/dev/nfs nfsroot=192.168.222.20:/source/rootfs_qemu, " \
13 "proto=tcp,nfsvers=3,nolock rw console=ttyAMA0,init=/linuxrc ip=192.168.222.21'; " \
14 */
15 #define CONFIG_BOOTCOMMAND \
16     "tftp 0x60008000 uImage4; " \
17     "tftp 0x63000000 vexpress-v2p-ca9.dtb; " \
18     "setenv bootargs 'root=/dev/nfs nfsroot=192.168.222.20:/source/rootfs_qemu, " \
19     "proto=tcp, nfsver=3, nolock rw console=ttyAMA0, init=/linuxrc " \
20     "ip=192.168.222.21:192.168.222.20:192.168.222.1:255.255.255.0'; " \
21     "bootm 0x60008000 - 0x63000000; "

  编辑好后保存退出,重新编译u-boot。然后运行qemu,结果如下。我们设置的IP参数的确传递给了内核,但是仍然无法挂载根文件系统!

  那么问题到底出现在哪呢?首先根文件系统是没问题的,之前测试过,nfs服务应该也没问题(其实有问题,但我没有往这个方向想,因为我用实际的板子挂载过),我一直怀疑是虚拟机的网络设置有问题,但是不知道怎么验证。后来,我在网上看到这么一句话:用kernel挂载ramdisk,然后尝试连接主机的nfs服务。前文说过,用kernel直接挂载ramdisk,我这里是可行的,所以值得一试。

2.2 在kernel中ping主机测试

  前文用的kernel是linux3.4.4,我之前看到的博文好多说要加载设备树,但是我没加载设备树也能启动kernel,并且这个版本的设备树,我使用 make dtbs 无法编译,即使我用 make mrproper 清空了之前的编译文件。所以,我又下载了一个kernel,linux-4.20.4。按照前文的步骤来一遍,发现的确需要设备树,否则无法启动内核。下面的第8行即是附加设备树选项。设备树文件在 arch/arm/boot/dtbs/ 目录下,后缀名为 dts,这个是可以直接编辑的,使用 make dtbs 可以编译设备树文件,生成后缀名为 dtb 二进制文件。

1 sudo qemu-system-arm \
2     -M vexpress-a9 \
3     -m 256M \
4     -kernel arch/arm/boot/zImage \
5     -nographic \
6     -append "root=/dev/mmcblk0  rw console=ttyAMA0" \
7     -sd ~/qemu_linux/rootfs.ext3 \
8     -dtb /tftpboot/vexpress-v2p-ca9.dtb \
9     -net nic,vlan=0 -net tap,vlan=0,ifname=tap0 

  然后 ifconfig 看看本地网络设备,发现只有 lo 这个本地回传地址,编辑 /etc/init.d/rcS 这个脚本,文末加上

ifconfig eth0 192.168.222.21

  重启qemu,用 ifconfig 查看,可以发现 eth0 启动了,接下来,ping 主机IP,发现可以ping通,说明qemu的网络没有问题。那么显然,还是nfs有问题。

2.3 在kernel中挂载nfs测试

  挂载主机的nfs:

  然后我看了下主机 nfs 的配置,/etc/exports,终于发现了问题,之前使用实际的板子挂载/source/rootfs,所以这里设置的共享路径是/source/rootfs,但是这个虚拟机挂载的根文件是/source/rootfs_qemu,难怪挂载不上。修改成/source ,重启主机 nfs 服务。

1 sudo service nfs-kernel-server restart

  再运行qemu,终于成功了。

1 sudo qemu-system-arm \
2     -M vexpress-a9 \
3     -kernel u-boot \
4     -nographic \
5     -m 512M \
6     -net nic,vlan=0 -net tap,vlan=0,ifname=tap0

  


 

3. 总结

  关于u-boot中的bootm、bootargs等参数的设置,其实还有一些不清楚的地方。单说vexpress,在u-boot-2015-04这个版本中,CONFIG_BOOTCOMMAND这个宏是定义在vexpress_common.h中,我们可以直接修改,但是在后面的u-boot版本中,比如说u-boot-2018 中,CONFIG_BOOTCOMMAND这个宏定义在了 /include/config_distro_bootcmd.h 中,而且看起来更加复杂。

我感觉直接改这些文件并不是官方推荐的做法,我找了几个案例,是通过在u-boot交互模式中设置环境变量来进行配置,当然如果对u-boot的源码很熟悉,直接改更方便。 

  

 

posted @ 2019-07-07 13:46  bigsissy  阅读(712)  评论(0编辑  收藏  举报